From 201e8c815709f306db8227c1054e99888d135c6f Mon Sep 17 00:00:00 2001 From: "Dmitry V. Levin" Date: Thu, 9 Mar 2017 02:00:35 +0000 Subject: [PATCH 001/305] programs/Makefile: remove zstd-internal target zstd-internal was intended to be a helper target, but it doesn't help at all, what it does in practice is a useless rebuild of zstd every time "make zstd" is invoked. Fixes: 030ac243a0f3 ("Changed Makefile to generate zstd with .gz support by default") --- programs/Makefile | 14 +++++--------- 1 file changed, 5 insertions(+), 9 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index 5620a25a3..a935c744a 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -83,8 +83,11 @@ all: zstd $(ZSTDDECOMP_O): CFLAGS += $(ALIGN_LOOP) -zstd-internal : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) -zstd-internal : $(ZSTDLIB_OBJ) zstdcli.o fileio.o bench.o datagen.o dibio.o +zstd : CPPFLAGS += $(ZLIBCPP) +zstd : LDFLAGS += $(ZLIBLD) +zstd-nogz : HAVE_ZLIB=0 +zstd zstd-nogz : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) +zstd zstd-nogz : $(ZSTDLIB_OBJ) zstdcli.o fileio.o bench.o datagen.o dibio.o ifeq ($(HAVE_ZLIB), 1) @echo "==> building zstd with .gz decompression support " else @@ -95,13 +98,6 @@ ifneq (,$(filter Windows%,$(OS))) endif $(CC) $(FLAGS) $^ $(RES_FILE) -o zstd$(EXT) $(LDFLAGS) -zstd-nogz : HAVE_ZLIB=0 -zstd-nogz : zstd-internal - -zstd : CPPFLAGS += $(ZLIBCPP) -zstd : LDFLAGS += $(ZLIBLD) -zstd : zstd-internal - zstd-release: DEBUGFLAGS := zstd-release: zstd From daec40db242db71946dbb250b05a88beadb37008 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 7 Mar 2017 12:08:15 -0800 Subject: [PATCH 002/305] Update .travis.yml and Makefile for medium tests --- .travis.yml | 54 +++++++++++++++++++++++++------------------------- Makefile | 28 ++++++++++++++++++++++---- appveyor.yml | 5 +++++ tests/Makefile | 4 ++-- 4 files changed, 58 insertions(+), 33 deletions(-) diff --git a/.travis.yml b/.travis.yml index 9c1e10e15..a52d57af3 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,39 +1,39 @@ +# Medium Tests: Run on all commits/PRs to dev branch + language: c sudo: required dist: trusty matrix: - fast_finish: true include: # Ubuntu 14.04 - - env: Cmd="make libc6install && make -C tests test32" - - env: Cmd='make valgrindinstall arminstall ppcinstall arm-ppc-compilation && make clean lib && CFLAGS="-O1 -g" make -C zlibWrapper valgrindTest && make -C tests valgrindTest' + - env: Cmd='make gcc6install && CC=gcc-6 make clean uasan-test-zstd' + - env: Cmd='make gcc6install libc6install && CC=gcc-6 make clean uasan-test-zstd32' + - env: Cmd='make clang38install && CC=clang-3.8 make clean msan-test-zstd' - - env: Cmd='CC=gcc-6 make gcc6install uasan-test' - - env: Cmd='CC=gcc-6 make gcc6install uasan-test32' - - env: Cmd="make arminstall armtest && make clean && make aarch64test" - - env: Cmd='make ppcinstall ppctest && make clean && make ppc64test' - - env: Cmd='make gpp6install zlibwrapper && make -C tests clean test-zstd-nolegacy && make -C tests versionsTest && make clean && cd contrib/pzstd && make test-pzstd && make test-pzstd32 && make test-pzstd-tsan && make test-pzstd-asan' - install: - - export CXX="g++-6" CC="gcc-6" + - env: Cmd='make gcc6install && CC=gcc-6 make clean uasan-fuzztest' + - env: Cmd='make gcc6install libc6install && CC=gcc-6 CFLAGS=-m32 make clean uasan-fuzztest' + - env: Cmd='make clang38install && CC=clang-3.8 make clean msan-fuzztest' + - env: Cmd='make clang38install && CC=clang-3.8 make clean tsan-test-zstream' - # OS X Mavericks - - env: Cmd="make gnu90build && make clean && make test && make clean && make travis-install" - os: osx + - env: Cmd='make valgrindinstall && make -C tests clean valgrindTest' + + - env: Cmd='make arminstall && make armfuzz' + - env: Cmd='make arminstall && make aarch64fuzz' + - env: Cmd='make ppcinstall && make ppcfuzz' + - env: Cmd='make ppcinstall && make ppc64fuzz' + +git: + depth: 1 + +branches: + only: + - dev + - master script: - JOB_NUMBER=$(echo $TRAVIS_JOB_NUMBER | sed -e 's:[0-9][0-9]*\.\(.*\):\1:') - # cron & master => full tests, as this is the final step towards a Release - # pull requests => normal tests (job numbers 1-3) - # other feature branches => short tests (job numbers 1-2) - echo JOB_NUMBER=$JOB_NUMBER TRAVIS_BRANCH=$TRAVIS_BRANCH TRAVIS_EVENT_TYPE=$TRAVIS_EVENT_TYPE TRAVIS_PULL_REQUEST=$TRAVIS_PULL_REQUEST - - if [ "$TRAVIS_EVENT_TYPE" = "cron" ] || [ "$TRAVIS_BRANCH" = "master" ]; then - FUZZERTEST=-T7mn sh -c "$Cmd" || travis_terminate 1; - else - if [ "$TRAVIS_EVENT_TYPE" = "pull_request" ] && [ $JOB_NUMBER -lt 4 ]; then - sh -c "$Cmd" || travis_terminate 1; - else - if [ $JOB_NUMBER -lt 3 ]; then - sh -c "$Cmd" || travis_terminate 1; - fi - fi - fi + - export FUZZERTEST=-T2mn; + export ZSTREAM_TESTTIME=-T2mn; + export DECODECORPUS_TESTTIME=-T1mn; + sh -c "$Cmd" || travis_terminate 1; diff --git a/Makefile b/Makefile index e10d29267..0187348ee 100644 --- a/Makefile +++ b/Makefile @@ -151,6 +151,18 @@ ppcbuild: clean ppc64build: clean CC=powerpc-linux-gnu-gcc CFLAGS="-m64 -Werror" $(MAKE) allarch +armfuzz: clean + CC=arm-linux-gnueabi-gcc QEMU_SYS=qemu-arm-static MOREFLAGS="-static" $(MAKE) -C $(TESTDIR) fuzztest + +aarch64fuzz: clean + CC=aarch64-linux-gnu-gcc QEMU_SYS=qemu-aarch64-static MOREFLAGS="-static" $(MAKE) -C $(TESTDIR) fuzztest + +ppcfuzz: clean + CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc-static MOREFLAGS="-static" $(MAKE) -C $(TESTDIR) fuzztest + +ppc64fuzz: clean + CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc64-static MOREFLAGS="-m64 -static" $(MAKE) -C $(TESTDIR) fuzztest + gpptest: clean CC=g++ $(MAKE) -C $(PRGDIR) all CFLAGS="-O3 -Wall -Wextra -Wundef -Wshadow -Wcast-align -Werror" @@ -189,7 +201,7 @@ arm-ppc-compilation: $(MAKE) -C $(PRGDIR) clean zstd CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc64-static ZSTDRTTEST= MOREFLAGS="-m64 -static" usan: clean - $(MAKE) test CC=clang MOREFLAGS="-g -fsanitize=undefined" + $(MAKE) test CC=clang MOREFLAGS="-g -fno-sanitize-recover=all -fsanitize=undefined" asan: clean $(MAKE) test CC=clang MOREFLAGS="-g -fsanitize=address" @@ -197,15 +209,20 @@ asan: clean msan: clean $(MAKE) test CC=clang MOREFLAGS="-g -fsanitize=memory -fno-omit-frame-pointer" # datagen.c fails this test for no obvious reason +msan-%: clean + LDFLAGS=-fuse-ld=gold MOREFLAGS="-fno-sanitize-recover=all -fsanitize=memory -fno-omit-frame-pointer" $(MAKE) -C $(TESTDIR) $* + asan32: clean $(MAKE) -C $(TESTDIR) test32 CC=clang MOREFLAGS="-g -fsanitize=address" uasan: clean - $(MAKE) test CC=clang MOREFLAGS="-g -fsanitize=address -fsanitize=undefined" + $(MAKE) test CC=clang MOREFLAGS="-g -fno-sanitize-recover=all -fsanitize=address,undefined" uasan-%: clean - LDFLAGS=-fuse-ld=gold CFLAGS="-Og -fsanitize=address -fsanitize=undefined" $(MAKE) -C $(TESTDIR) $* + LDFLAGS=-fuse-ld=gold MOREFLAGS="-Og -fno-sanitize-recover=all -fsanitize=address,undefined" $(MAKE) -C $(TESTDIR) $* +tsan-%: clean + LDFLAGS=-fuse-ld=gold MOREFLAGS="-g -fno-sanitize-recover=all -fsanitize=thread" $(MAKE) -C $(TESTDIR) $* apt-install: sudo apt-get -yq --no-install-suggests --no-install-recommends --force-yes install $(APT_PACKAGES) @@ -217,7 +234,7 @@ ppcinstall: APT_PACKAGES="qemu-system-ppc qemu-user-static gcc-powerpc-linux-gnu" $(MAKE) apt-install arminstall: - APT_PACKAGES="qemu-system-arm qemu-user-static gcc-powerpc-linux-gnu gcc-arm-linux-gnueabi libc6-dev-armel-cross gcc-aarch64-linux-gnu libc6-dev-arm64-cross" $(MAKE) apt-install + APT_PACKAGES="qemu-system-arm qemu-user-static gcc-arm-linux-gnueabi libc6-dev-armel-cross gcc-aarch64-linux-gnu libc6-dev-arm64-cross" $(MAKE) apt-install valgrindinstall: APT_PACKAGES="valgrind" $(MAKE) apt-install @@ -231,6 +248,9 @@ gcc6install: apt-add-repo gpp6install: apt-add-repo APT_PACKAGES="libc6-dev-i386 g++-multilib gcc-6 g++-6 g++-6-multilib" $(MAKE) apt-install +clang38install: + APT_PACKAGES="clang-3.8" $(MAKE) apt-install + endif diff --git a/appveyor.yml b/appveyor.yml index 51ff488a4..9507fec6e 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -146,6 +146,11 @@ test_script: fuzzer_VS2015_%PLATFORM%_Release.exe %FUZZERTEST% ) +branches: + only: + - dev + - master + artifacts: - path: bin\zstd.exe - path: bin\zstd32.exe diff --git a/tests/Makefile b/tests/Makefile index 8b19aa3d5..39e4d1015 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -75,8 +75,6 @@ all32: fullbench32 fuzzer32 zstreamtest32 zbufftest32 dll: fuzzer-dll zstreamtest-dll zbufftest-dll - - zstd: $(MAKE) -C $(PRGDIR) $@ @@ -257,6 +255,8 @@ zstd-playTests: datagen shortest: ZSTDRTTEST= shortest: test-zstd +fuzztest: test-fuzzer test-zstream test-decodecorpus + test: test-zstd test-fullbench test-fuzzer test-zstream test-invalidDictionaries test-legacy test-decodecorpus ifeq ($(QEMU_SYS),) test: test-pool From 7c8f5d5bc7ee4c16c5de73cb2e1c0e13f2ffca79 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 9 Mar 2017 16:05:10 -0800 Subject: [PATCH 003/305] Make test times overwritable --- tests/Makefile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tests/Makefile b/tests/Makefile index 39e4d1015..59256f841 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -60,10 +60,10 @@ endif MULTITHREAD = $(MULTITHREAD_CPP) $(MULTITHREAD_LD) VOID = /dev/null -ZSTREAM_TESTTIME = -T2mn +ZSTREAM_TESTTIME ?= -T2mn FUZZERTEST ?= -T5mn ZSTDRTTEST = --test-large-data -DECODECORPUS_TESTTIME = -T30 +DECODECORPUS_TESTTIME ?= -T30 .PHONY: default all all32 dll clean test test32 test-all namespaceTest versionsTest From 2500dcfa5f6f99dc015231d5561c35d9a4016965 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 9 Mar 2017 13:59:26 -0800 Subject: [PATCH 004/305] Add testing description --- TESTING.md | 44 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 44 insertions(+) create mode 100644 TESTING.md diff --git a/TESTING.md b/TESTING.md new file mode 100644 index 000000000..1fa5fe8c2 --- /dev/null +++ b/TESTING.md @@ -0,0 +1,44 @@ +Testing +======= + +Zstandard CI testing is split up into three sections: +short, medium, and long tests. + +Short Tests +----------- +Short tests run on CircleCI for new commits on every branch and pull request. +They consist of the following tests: +- Compilation on all supported targets (x86, x86_64, ARM, AArch64, PowerPC, and PowerPC64) +- Compilation on various versions of gcc, clang, and g++ +- `tests/playTests.sh` on x86_64, without the tests on long data (CLI tests) +- Small tests (`tests/legacy.c`, `tests/longmatch.c`, `tests/symbols.c`) on x64_64 + +Medium Tests +------------ +Medium tests run on every commit and pull request to `dev` branch, on TravisCI. +They consist of the following tests: +- The following tests run with UBsan and Asan on x86_64 and x86, as well as with + Msan on x86_64 + - `tests/playTests.sh --test-long-data` + - Fuzzer tests: `tests/fuzzer.c`, `tests/zstreamtest.c`, and `tests/decodecorpus.c` +- `tests/zstreamtest.c` under Tsan (streaming mode, including multithreaded mode) +- Valgrind Test (`make -C tests valgrindTest`) (testing CLI and fuzzer under valgrind) +- Fuzzer tests (see above) on ARM, AArch64, PowerPC, and PowerPC64 + +Long Tests +---------- +Long tests run on all commits to `master` branch, +and once a day on the current version of `dev` branch, +on TravisCI. +They consist of the following tests: +- Entire test suite (including fuzzers and some other specialized tests) on: + - x86_64 and x86 with UBsan and Asan + - x86_64 with Msan + - ARM, AArch64, PowerPC, and PowerPC64 +- Streaming mode fuzzer with Tsan (for the `zstdmt` testing) +- ZlibWrapper tests, including under valgrind +- Versions test (ensuring `zstd` can decode files from all previous versions) +- `pzstd` with asan and tsan, as well as in 32-bits mode +- Testing `zstd` with legacy mode off +- Testing `zbuff` (old streaming API) +- Entire test suite and make install on OS X From caf0ee8d20663b9dce7d0ac4cd048ebc542eb473 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 9 Mar 2017 17:28:08 -0800 Subject: [PATCH 005/305] Make signed integer overflow recoverable in UBsan --- Makefile | 13 ++++++++++--- 1 file changed, 10 insertions(+), 3 deletions(-) diff --git a/Makefile b/Makefile index 0187348ee..49f29d782 100644 --- a/Makefile +++ b/Makefile @@ -200,12 +200,19 @@ arm-ppc-compilation: $(MAKE) -C $(PRGDIR) clean zstd CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc-static ZSTDRTTEST= MOREFLAGS="-Werror -Wno-attributes -static" $(MAKE) -C $(PRGDIR) clean zstd CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc64-static ZSTDRTTEST= MOREFLAGS="-m64 -static" +# run UBsan with -fsanitize-recover=signed-integer-overflow +# due to a bug in UBsan when doing pointer subtraction +# https://gcc.gnu.org/bugzilla/show_bug.cgi?id=63303 + usan: clean - $(MAKE) test CC=clang MOREFLAGS="-g -fno-sanitize-recover=all -fsanitize=undefined" + $(MAKE) test CC=clang MOREFLAGS="-g -fno-sanitize-recover=all -fsanitize-recover=signed-integer-overflow -fsanitize=undefined" asan: clean $(MAKE) test CC=clang MOREFLAGS="-g -fsanitize=address" +asan-%: clean + LDFLAGS=-fuse-ld=gold MOREFLAGS="-g -fno-sanitize-recover=all -fsanitize=address" $(MAKE) -C $(TESTDIR) $* + msan: clean $(MAKE) test CC=clang MOREFLAGS="-g -fsanitize=memory -fno-omit-frame-pointer" # datagen.c fails this test for no obvious reason @@ -216,10 +223,10 @@ asan32: clean $(MAKE) -C $(TESTDIR) test32 CC=clang MOREFLAGS="-g -fsanitize=address" uasan: clean - $(MAKE) test CC=clang MOREFLAGS="-g -fno-sanitize-recover=all -fsanitize=address,undefined" + $(MAKE) test CC=clang MOREFLAGS="-g -fno-sanitize-recover=all -fsanitize-recover=signed-integer-overflow -fsanitize=address,undefined" uasan-%: clean - LDFLAGS=-fuse-ld=gold MOREFLAGS="-Og -fno-sanitize-recover=all -fsanitize=address,undefined" $(MAKE) -C $(TESTDIR) $* + LDFLAGS=-fuse-ld=gold MOREFLAGS="-Og -fno-sanitize-recover=all -fsanitize-recover=signed-integer-overflow -fsanitize=address,undefined" $(MAKE) -C $(TESTDIR) $* tsan-%: clean LDFLAGS=-fuse-ld=gold MOREFLAGS="-g -fno-sanitize-recover=all -fsanitize=thread" $(MAKE) -C $(TESTDIR) $* From 8fe5c6862c56f937ad5c3c25d387d4a331b42775 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 9 Mar 2017 11:54:34 -0800 Subject: [PATCH 006/305] Fix undefined behaviour in decompressor --- lib/common/mem.h | 18 ++++++++++-------- lib/decompress/zstd_decompress.c | 6 +++--- 2 files changed, 13 insertions(+), 11 deletions(-) diff --git a/lib/common/mem.h b/lib/common/mem.h index 7a3f72141..3cacd216a 100644 --- a/lib/common/mem.h +++ b/lib/common/mem.h @@ -48,14 +48,15 @@ MEM_STATIC void MEM_check(void) { MEM_STATIC_ASSERT((sizeof(size_t)==4) || (size *****************************************************************/ #if !defined (__VMS) && (defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) ) # include - typedef uint8_t BYTE; - typedef uint16_t U16; - typedef int16_t S16; - typedef uint32_t U32; - typedef int32_t S32; - typedef uint64_t U64; - typedef int64_t S64; - typedef intptr_t iPtrDiff; + typedef uint8_t BYTE; + typedef uint16_t U16; + typedef int16_t S16; + typedef uint32_t U32; + typedef int32_t S32; + typedef uint64_t U64; + typedef int64_t S64; + typedef intptr_t iPtrDiff; + typedef uintptr_t uPtrDiff; #else typedef unsigned char BYTE; typedef unsigned short U16; @@ -65,6 +66,7 @@ MEM_STATIC void MEM_check(void) { MEM_STATIC_ASSERT((sizeof(size_t)==4) || (size typedef unsigned long long U64; typedef signed long long S64; typedef ptrdiff_t iPtrDiff; + typedef size_t uPtrDiff; #endif diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c index 482c334ff..516edfcc6 100644 --- a/lib/decompress/zstd_decompress.c +++ b/lib/decompress/zstd_decompress.c @@ -1033,7 +1033,7 @@ size_t ZSTD_execSequence(BYTE* op, if (sequence.offset > (size_t)(oLitEnd - base)) { /* offset beyond prefix */ if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); - match += (dictEnd-base); + match = dictEnd + (match - base); if (match + sequence.matchLength <= dictEnd) { memmove(oLitEnd, match, sequence.matchLength); return sequenceLength; @@ -1216,7 +1216,7 @@ FORCE_INLINE seq_t ZSTD_decodeSequenceLong_generic(seqState_t* seqState, int con { size_t const pos = seqState->pos + seq.litLength; seq.match = seqState->base + pos - seq.offset; /* single memory segment */ - if (seq.offset > pos) seq.match += seqState->gotoDict; /* separate memory segment */ + if (seq.offset > pos) seq.match += (uPtrDiff)seqState->gotoDict; /* separate memory segment */ seqState->pos = pos + seq.matchLength; } @@ -1356,7 +1356,7 @@ static size_t ZSTD_decompressSequencesLong( { U32 i; for (i=0; ientropy.rep[i]; } seqState.base = base; seqState.pos = (size_t)(op-base); - seqState.gotoDict = (iPtrDiff)(dictEnd - base); + seqState.gotoDict = (iPtrDiff)((uPtrDiff)dictEnd - (uPtrDiff)base); /* cast to avoid undefined behaviour */ CHECK_E(BIT_initDStream(&seqState.DStream, ip, iend-ip), corruption_detected); FSE_initDState(&seqState.stateLL, &seqState.DStream, dctx->LLTptr); FSE_initDState(&seqState.stateOffb, &seqState.DStream, dctx->OFTptr); From 784082f49cfd48c9f77db23ae7df35703b8b68b2 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Fri, 10 Mar 2017 10:34:45 -0800 Subject: [PATCH 007/305] Change gotoDict type to uPtrDiff --- lib/decompress/zstd_decompress.c | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c index 516edfcc6..943bdf94e 100644 --- a/lib/decompress/zstd_decompress.c +++ b/lib/decompress/zstd_decompress.c @@ -876,7 +876,7 @@ typedef struct { size_t prevOffset[ZSTD_REP_NUM]; const BYTE* base; size_t pos; - iPtrDiff gotoDict; + uPtrDiff gotoDict; } seqState_t; @@ -1216,7 +1216,7 @@ FORCE_INLINE seq_t ZSTD_decodeSequenceLong_generic(seqState_t* seqState, int con { size_t const pos = seqState->pos + seq.litLength; seq.match = seqState->base + pos - seq.offset; /* single memory segment */ - if (seq.offset > pos) seq.match += (uPtrDiff)seqState->gotoDict; /* separate memory segment */ + if (seq.offset > pos) seq.match += seqState->gotoDict; /* separate memory segment */ seqState->pos = pos + seq.matchLength; } @@ -1356,7 +1356,7 @@ static size_t ZSTD_decompressSequencesLong( { U32 i; for (i=0; ientropy.rep[i]; } seqState.base = base; seqState.pos = (size_t)(op-base); - seqState.gotoDict = (iPtrDiff)((uPtrDiff)dictEnd - (uPtrDiff)base); /* cast to avoid undefined behaviour */ + seqState.gotoDict = (uPtrDiff)dictEnd - (uPtrDiff)base; /* cast to avoid undefined behaviour */ CHECK_E(BIT_initDStream(&seqState.DStream, ip, iend-ip), corruption_detected); FSE_initDState(&seqState.stateLL, &seqState.DStream, dctx->LLTptr); FSE_initDState(&seqState.stateOffb, &seqState.DStream, dctx->OFTptr); From 334cb34edba6da2232701b298b61a62fc26259a4 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 13 Mar 2017 14:32:30 -0700 Subject: [PATCH 008/305] ZSTD_LEGACY_SUPPORT defines lowest supported version --- lib/Makefile | 13 +++-- lib/decompress/zstd_decompress.c | 4 +- lib/legacy/zstd_legacy.h | 97 ++++++++++++++++++++++++++++++++ programs/Makefile | 10 ++-- 4 files changed, 113 insertions(+), 11 deletions(-) diff --git a/lib/Makefile b/lib/Makefile index 58f99baf5..18b08a11d 100644 --- a/lib/Makefile +++ b/lib/Makefile @@ -31,12 +31,15 @@ FLAGS = $(CPPFLAGS) $(CFLAGS) ZSTD_FILES := $(wildcard common/*.c compress/*.c decompress/*.c dictBuilder/*.c deprecated/*.c) -ifeq ($(ZSTD_LEGACY_SUPPORT), 0) -CPPFLAGS += -DZSTD_LEGACY_SUPPORT=0 -else -CPPFLAGS += -I./legacy -DZSTD_LEGACY_SUPPORT=1 -ZSTD_FILES+= $(wildcard legacy/*.c) +ZSTD_LEGACY_SUPPORT ?= 1 + +ifneq ($(ZSTD_LEGACY_SUPPORT), 0) +ifeq ($(shell test $(ZSTD_LEGACY_SUPPORT) -lt 8; echo $$?), 0) + ZSTD_FILES += $(shell ls legacy/*.c | grep 'v0[$(ZSTD_LEGACY_SUPPORT)-7]') endif + CPPFLAGS += -I./legacy +endif +CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) ZSTD_OBJ := $(patsubst %.c,%.o,$(ZSTD_FILES)) diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c index 943bdf94e..2aaa4a3df 100644 --- a/lib/decompress/zstd_decompress.c +++ b/lib/decompress/zstd_decompress.c @@ -320,7 +320,7 @@ size_t ZSTD_getFrameParams(ZSTD_frameParams* fparamsPtr, const void* src, size_t * - ZSTD_CONTENTSIZE_ERROR if an error occurred (e.g. invalid magic number, srcSize too small) */ unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize) { -#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT==1) +#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) if (ZSTD_isLegacy(src, srcSize)) { unsigned long long const ret = ZSTD_getDecompressedSize_legacy(src, srcSize); return ret == 0 ? ZSTD_CONTENTSIZE_UNKNOWN : ret; @@ -1472,7 +1472,7 @@ size_t ZSTD_generateNxBytes(void* dst, size_t dstCapacity, BYTE byte, size_t len * @return : the compressed size of the frame starting at `src` */ size_t ZSTD_findFrameCompressedSize(const void *src, size_t srcSize) { -#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT==1) +#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) if (ZSTD_isLegacy(src, srcSize)) return ZSTD_findFrameCompressedSizeLegacy(src, srcSize); #endif if (srcSize >= ZSTD_skippableHeaderSize && diff --git a/lib/legacy/zstd_legacy.h b/lib/legacy/zstd_legacy.h index 707e76f0a..18e22e651 100644 --- a/lib/legacy/zstd_legacy.h +++ b/lib/legacy/zstd_legacy.h @@ -28,6 +28,31 @@ extern "C" { #include "zstd_v06.h" #include "zstd_v07.h" +#ifndef ZSTD_LEGACY_SUPPORT +# define ZSTD_LEGACY_SUPPORT 8 +#endif + +#if (ZSTD_LEGACY_SUPPORT <= 1) +# include "zstd_v01.h" +#endif +#if (ZSTD_LEGACY_SUPPORT <= 2) +# include "zstd_v02.h" +#endif +#if (ZSTD_LEGACY_SUPPORT <= 3) +# include "zstd_v03.h" +#endif +#if (ZSTD_LEGACY_SUPPORT <= 4) +# include "zstd_v04.h" +#endif +#if (ZSTD_LEGACY_SUPPORT <= 5) +# include "zstd_v05.h" +#endif +#if (ZSTD_LEGACY_SUPPORT <= 6) +# include "zstd_v06.h" +#endif +#if (ZSTD_LEGACY_SUPPORT <= 7) +# include "zstd_v07.h" +#endif /** ZSTD_isLegacy() : @return : > 0 if supported by legacy decoder. 0 otherwise. @@ -40,13 +65,27 @@ MEM_STATIC unsigned ZSTD_isLegacy(const void* src, size_t srcSize) magicNumberLE = MEM_readLE32(src); switch(magicNumberLE) { +#if (ZSTD_LEGACY_SUPPORT <= 1) case ZSTDv01_magicNumberLE:return 1; +#endif +#if (ZSTD_LEGACY_SUPPORT <= 2) case ZSTDv02_magicNumber : return 2; +#endif +#if (ZSTD_LEGACY_SUPPORT <= 3) case ZSTDv03_magicNumber : return 3; +#endif +#if (ZSTD_LEGACY_SUPPORT <= 4) case ZSTDv04_magicNumber : return 4; +#endif +#if (ZSTD_LEGACY_SUPPORT <= 5) case ZSTDv05_MAGICNUMBER : return 5; +#endif +#if (ZSTD_LEGACY_SUPPORT <= 6) case ZSTDv06_MAGICNUMBER : return 6; +#endif +#if (ZSTD_LEGACY_SUPPORT <= 7) case ZSTDv07_MAGICNUMBER : return 7; +#endif default : return 0; } } @@ -56,24 +95,30 @@ MEM_STATIC unsigned long long ZSTD_getDecompressedSize_legacy(const void* src, s { U32 const version = ZSTD_isLegacy(src, srcSize); if (version < 5) return 0; /* no decompressed size in frame header, or not a legacy format */ +#if (ZSTD_LEGACY_SUPPORT <= 5) if (version==5) { ZSTDv05_parameters fParams; size_t const frResult = ZSTDv05_getFrameParams(&fParams, src, srcSize); if (frResult != 0) return 0; return fParams.srcSize; } +#endif +#if (ZSTD_LEGACY_SUPPORT <= 6) if (version==6) { ZSTDv06_frameParams fParams; size_t const frResult = ZSTDv06_getFrameParams(&fParams, src, srcSize); if (frResult != 0) return 0; return fParams.frameContentSize; } +#endif +#if (ZSTD_LEGACY_SUPPORT <= 7) if (version==7) { ZSTDv07_frameParams fParams; size_t const frResult = ZSTDv07_getFrameParams(&fParams, src, srcSize); if (frResult != 0) return 0; return fParams.frameContentSize; } +#endif return 0; /* should not be possible */ } @@ -86,14 +131,23 @@ MEM_STATIC size_t ZSTD_decompressLegacy( U32 const version = ZSTD_isLegacy(src, compressedSize); switch(version) { +#if (ZSTD_LEGACY_SUPPORT <= 1) case 1 : return ZSTDv01_decompress(dst, dstCapacity, src, compressedSize); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 2) case 2 : return ZSTDv02_decompress(dst, dstCapacity, src, compressedSize); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 3) case 3 : return ZSTDv03_decompress(dst, dstCapacity, src, compressedSize); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 4) case 4 : return ZSTDv04_decompress(dst, dstCapacity, src, compressedSize); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 5) case 5 : { size_t result; ZSTDv05_DCtx* const zd = ZSTDv05_createDCtx(); @@ -102,6 +156,8 @@ MEM_STATIC size_t ZSTD_decompressLegacy( ZSTDv05_freeDCtx(zd); return result; } +#endif +#if (ZSTD_LEGACY_SUPPORT <= 6) case 6 : { size_t result; ZSTDv06_DCtx* const zd = ZSTDv06_createDCtx(); @@ -110,6 +166,8 @@ MEM_STATIC size_t ZSTD_decompressLegacy( ZSTDv06_freeDCtx(zd); return result; } +#endif +#if (ZSTD_LEGACY_SUPPORT <= 7) case 7 : { size_t result; ZSTDv07_DCtx* const zd = ZSTDv07_createDCtx(); @@ -118,6 +176,7 @@ MEM_STATIC size_t ZSTD_decompressLegacy( ZSTDv07_freeDCtx(zd); return result; } +#endif default : return ERROR(prefix_unknown); } @@ -129,20 +188,34 @@ MEM_STATIC size_t ZSTD_findFrameCompressedSizeLegacy(const void *src, U32 const version = ZSTD_isLegacy(src, compressedSize); switch(version) { +#if (ZSTD_LEGACY_SUPPORT <= 1) case 1 : return ZSTDv01_findFrameCompressedSize(src, compressedSize); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 2) case 2 : return ZSTDv02_findFrameCompressedSize(src, compressedSize); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 3) case 3 : return ZSTDv03_findFrameCompressedSize(src, compressedSize); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 4) case 4 : return ZSTDv04_findFrameCompressedSize(src, compressedSize); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 5) case 5 : return ZSTDv05_findFrameCompressedSize(src, compressedSize); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 6) case 6 : return ZSTDv06_findFrameCompressedSize(src, compressedSize); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 7) case 7 : return ZSTDv07_findFrameCompressedSize(src, compressedSize); +#endif default : return ERROR(prefix_unknown); } @@ -157,10 +230,18 @@ MEM_STATIC size_t ZSTD_freeLegacyStreamContext(void* legacyContext, U32 version) case 2 : case 3 : return ERROR(version_unsupported); +#if (ZSTD_LEGACY_SUPPORT <= 4) case 4 : return ZBUFFv04_freeDCtx((ZBUFFv04_DCtx*)legacyContext); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 5) case 5 : return ZBUFFv05_freeDCtx((ZBUFFv05_DCtx*)legacyContext); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 6) case 6 : return ZBUFFv06_freeDCtx((ZBUFFv06_DCtx*)legacyContext); +#endif +#if (ZSTD_LEGACY_SUPPORT <= 7) case 7 : return ZBUFFv07_freeDCtx((ZBUFFv07_DCtx*)legacyContext); +#endif } } @@ -176,6 +257,7 @@ MEM_STATIC size_t ZSTD_initLegacyStream(void** legacyContext, U32 prevVersion, U case 2 : case 3 : return 0; +#if (ZSTD_LEGACY_SUPPORT <= 4) case 4 : { ZBUFFv04_DCtx* dctx = (prevVersion != newVersion) ? ZBUFFv04_createDCtx() : (ZBUFFv04_DCtx*)*legacyContext; @@ -185,6 +267,8 @@ MEM_STATIC size_t ZSTD_initLegacyStream(void** legacyContext, U32 prevVersion, U *legacyContext = dctx; return 0; } +#endif +#if (ZSTD_LEGACY_SUPPORT <= 5) case 5 : { ZBUFFv05_DCtx* dctx = (prevVersion != newVersion) ? ZBUFFv05_createDCtx() : (ZBUFFv05_DCtx*)*legacyContext; @@ -193,6 +277,8 @@ MEM_STATIC size_t ZSTD_initLegacyStream(void** legacyContext, U32 prevVersion, U *legacyContext = dctx; return 0; } +#endif +#if (ZSTD_LEGACY_SUPPORT <= 6) case 6 : { ZBUFFv06_DCtx* dctx = (prevVersion != newVersion) ? ZBUFFv06_createDCtx() : (ZBUFFv06_DCtx*)*legacyContext; @@ -201,6 +287,8 @@ MEM_STATIC size_t ZSTD_initLegacyStream(void** legacyContext, U32 prevVersion, U *legacyContext = dctx; return 0; } +#endif +#if (ZSTD_LEGACY_SUPPORT <= 7) case 7 : { ZBUFFv07_DCtx* dctx = (prevVersion != newVersion) ? ZBUFFv07_createDCtx() : (ZBUFFv07_DCtx*)*legacyContext; @@ -209,6 +297,7 @@ MEM_STATIC size_t ZSTD_initLegacyStream(void** legacyContext, U32 prevVersion, U *legacyContext = dctx; return 0; } +#endif } } @@ -224,6 +313,7 @@ MEM_STATIC size_t ZSTD_decompressLegacyStream(void* legacyContext, U32 version, case 2 : case 3 : return ERROR(version_unsupported); +#if (ZSTD_LEGACY_SUPPORT <= 4) case 4 : { ZBUFFv04_DCtx* dctx = (ZBUFFv04_DCtx*) legacyContext; @@ -236,6 +326,8 @@ MEM_STATIC size_t ZSTD_decompressLegacyStream(void* legacyContext, U32 version, input->pos += readSize; return hintSize; } +#endif +#if (ZSTD_LEGACY_SUPPORT <= 5) case 5 : { ZBUFFv05_DCtx* dctx = (ZBUFFv05_DCtx*) legacyContext; @@ -248,6 +340,8 @@ MEM_STATIC size_t ZSTD_decompressLegacyStream(void* legacyContext, U32 version, input->pos += readSize; return hintSize; } +#endif +#if (ZSTD_LEGACY_SUPPORT <= 6) case 6 : { ZBUFFv06_DCtx* dctx = (ZBUFFv06_DCtx*) legacyContext; @@ -260,6 +354,8 @@ MEM_STATIC size_t ZSTD_decompressLegacyStream(void* legacyContext, U32 version, input->pos += readSize; return hintSize; } +#endif +#if (ZSTD_LEGACY_SUPPORT <= 7) case 7 : { ZBUFFv07_DCtx* dctx = (ZBUFFv07_DCtx*) legacyContext; @@ -272,6 +368,7 @@ MEM_STATIC size_t ZSTD_decompressLegacyStream(void* legacyContext, U32 version, input->pos += readSize; return hintSize; } +#endif } } diff --git a/programs/Makefile b/programs/Makefile index a935c744a..0bf194260 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -42,12 +42,14 @@ ZSTD_FILES := $(ZSTDDECOMP_FILES) $(ZSTDCOMMON_FILES) $(ZSTDCOMP_FILES) ZDICT_FILES := $(ZSTDDIR)/dictBuilder/*.c ZSTDDECOMP_O = $(ZSTDDIR)/decompress/zstd_decompress.o -ifeq ($(ZSTD_LEGACY_SUPPORT), 0) +ZSTD_LEGACY_SUPPORT ?= 1 ZSTDLEGACY_FILES:= +ifneq ($(ZSTD_LEGACY_SUPPORT), 0) +ifeq ($(shell test $(ZSTD_LEGACY_SUPPORT) -lt 8; echo $$?), 0) + ZSTDLEGACY_FILES += $(shell ls $(ZSTDDIR)/legacy/*.c | grep 'v0[$(ZSTD_LEGACY_SUPPORT)-7]') +endif + CPPFLAGS += -I$(ZSTDDIR)/legacy else -ZSTD_LEGACY_SUPPORT:=1 -CPPFLAGS += -I$(ZSTDDIR)/legacy -ZSTDLEGACY_FILES:= $(ZSTDDIR)/legacy/*.c endif ZSTDLIB_FILES := $(wildcard $(ZSTD_FILES)) $(wildcard $(ZSTDLEGACY_FILES)) $(wildcard $(ZDICT_FILES)) From 120df494e9e5df89781307776377a6e1dac16df3 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 13 Mar 2017 14:44:08 -0700 Subject: [PATCH 009/305] Update builds to not support legacy v01-v03 --- .buckconfig | 4 ++-- build/VS2005/zstd/zstd.vcproj | 8 ++++---- build/VS2005/zstdlib/zstdlib.vcproj | 8 ++++---- build/VS2008/zstd/zstd.vcproj | 8 ++++---- build/VS2008/zstdlib/zstdlib.vcproj | 8 ++++---- build/VS2010/libzstd-dll/libzstd-dll.vcxproj | 8 ++++---- build/VS2010/libzstd/libzstd.vcxproj | 8 ++++---- build/VS2010/zstd/zstd.vcxproj | 8 ++++---- build/cmake/CMakeLists.txt | 2 +- contrib/meson/meson.build | 2 +- lib/Makefile | 2 +- programs/Makefile | 2 +- tests/Makefile | 2 +- 13 files changed, 35 insertions(+), 35 deletions(-) diff --git a/.buckconfig b/.buckconfig index d698b35ba..483f6053b 100644 --- a/.buckconfig +++ b/.buckconfig @@ -1,7 +1,7 @@ [cxx] - cppflags = -DXXH_NAMESPACE=ZSTD_ -DZSTD_LEGACY_SUPPORT=1 + cppflags = -DXXH_NAMESPACE=ZSTD_ -DZSTD_LEGACY_SUPPORT=4 cflags = -Wall -Wextra -Wcast-qual -Wcast-align -Wshadow -Wstrict-aliasing=1 -Wswitch-enum -Wdeclaration-after-statement -Wstrict-prototypes -Wundef -Wpointer-arith - cxxppflags = -DXXH_NAMESPACE=ZSTD_ -DZSTD_LEGACY_SUPPORT=1 + cxxppflags = -DXXH_NAMESPACE=ZSTD_ -DZSTD_LEGACY_SUPPORT=4 cxxflags = -std=c++11 -Wno-deprecated-declarations gtest_dep = //contrib/pzstd:gtest diff --git a/build/VS2005/zstd/zstd.vcproj b/build/VS2005/zstd/zstd.vcproj index 58f254bc8..1f4febead 100644 --- a/build/VS2005/zstd/zstd.vcproj +++ b/build/VS2005/zstd/zstd.vcproj @@ -44,7 +44,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -121,7 +121,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" @@ -196,7 +196,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -274,7 +274,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" diff --git a/build/VS2005/zstdlib/zstdlib.vcproj b/build/VS2005/zstdlib/zstdlib.vcproj index f4c9950ff..8da313673 100644 --- a/build/VS2005/zstdlib/zstdlib.vcproj +++ b/build/VS2005/zstdlib/zstdlib.vcproj @@ -44,7 +44,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -120,7 +120,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" @@ -194,7 +194,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -271,7 +271,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" diff --git a/build/VS2008/zstd/zstd.vcproj b/build/VS2008/zstd/zstd.vcproj index 2dfaf3937..468d25672 100644 --- a/build/VS2008/zstd/zstd.vcproj +++ b/build/VS2008/zstd/zstd.vcproj @@ -45,7 +45,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -122,7 +122,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" @@ -197,7 +197,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -275,7 +275,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" diff --git a/build/VS2008/zstdlib/zstdlib.vcproj b/build/VS2008/zstdlib/zstdlib.vcproj index cba0ff908..857e1463e 100644 --- a/build/VS2008/zstdlib/zstdlib.vcproj +++ b/build/VS2008/zstdlib/zstdlib.vcproj @@ -45,7 +45,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -121,7 +121,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" @@ -195,7 +195,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -272,7 +272,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" diff --git a/build/VS2010/libzstd-dll/libzstd-dll.vcxproj b/build/VS2010/libzstd-dll/libzstd-dll.vcxproj index f78598fb4..866d04a04 100644 --- a/build/VS2010/libzstd-dll/libzstd-dll.vcxproj +++ b/build/VS2010/libzstd-dll/libzstd-dll.vcxproj @@ -149,7 +149,7 @@ Level4 Disabled - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) true EnableFastChecks MultiThreadedDebugDLL @@ -169,7 +169,7 @@ Level4 Disabled - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) true EnableFastChecks MultiThreadedDebugDLL @@ -189,7 +189,7 @@ MaxSpeed true true - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) false MultiThreaded ProgramDatabase @@ -211,7 +211,7 @@ MaxSpeed true true - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) false false MultiThreaded diff --git a/build/VS2010/libzstd/libzstd.vcxproj b/build/VS2010/libzstd/libzstd.vcxproj index 727795514..186b4c4da 100644 --- a/build/VS2010/libzstd/libzstd.vcxproj +++ b/build/VS2010/libzstd/libzstd.vcxproj @@ -146,7 +146,7 @@ Level4 Disabled - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) true EnableFastChecks MultiThreadedDebugDLL @@ -166,7 +166,7 @@ Level4 Disabled - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) true EnableFastChecks MultiThreadedDebugDLL @@ -186,7 +186,7 @@ MaxSpeed true true - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) false MultiThreaded ProgramDatabase @@ -208,7 +208,7 @@ MaxSpeed true true - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) false false MultiThreaded diff --git a/build/VS2010/zstd/zstd.vcxproj b/build/VS2010/zstd/zstd.vcxproj index 62c0fe10f..7568e4902 100644 --- a/build/VS2010/zstd/zstd.vcxproj +++ b/build/VS2010/zstd/zstd.vcxproj @@ -155,7 +155,7 @@ Level4 Disabled - ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions) + ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions) true false @@ -171,7 +171,7 @@ Level4 Disabled - ZSTD_LEGACY_SUPPORT=1;WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions) + ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions) true false @@ -189,7 +189,7 @@ MaxSpeed true true - ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions) + ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions) false false MultiThreaded @@ -210,7 +210,7 @@ MaxSpeed true true - ZSTD_LEGACY_SUPPORT=1;WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions) + ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions) false false MultiThreaded diff --git a/build/cmake/CMakeLists.txt b/build/cmake/CMakeLists.txt index 6b7c28925..4805cc2c9 100644 --- a/build/cmake/CMakeLists.txt +++ b/build/cmake/CMakeLists.txt @@ -16,7 +16,7 @@ OPTION(ZSTD_BUILD_CONTRIB "BUILD CONTRIB" OFF) IF (ZSTD_LEGACY_SUPPORT) MESSAGE(STATUS "ZSTD_LEGACY_SUPPORT defined!") - ADD_DEFINITIONS(-DZSTD_LEGACY_SUPPORT=1) + ADD_DEFINITIONS(-DZSTD_LEGACY_SUPPORT=4) ELSE (ZSTD_LEGACY_SUPPORT) MESSAGE(STATUS "ZSTD_LEGACY_SUPPORT not defined!") ADD_DEFINITIONS(-DZSTD_LEGACY_SUPPORT=0) diff --git a/contrib/meson/meson.build b/contrib/meson/meson.build index 369461335..8cbdcabec 100644 --- a/contrib/meson/meson.build +++ b/contrib/meson/meson.build @@ -15,7 +15,7 @@ libzstd_includes = [include_directories(common_dir, dictbuilder_dir, compress_di if get_option('legacy_support') message('Enabling legacy support') - libzstd_cflags = ['-DZSTD_LEGACY_SUPPORT=1'] + libzstd_cflags = ['-DZSTD_LEGACY_SUPPORT=4'] legacy_dir = join_paths(lib_dir, 'legacy') libzstd_includes += [include_directories(legacy_dir)] diff --git a/lib/Makefile b/lib/Makefile index 18b08a11d..197fdeeea 100644 --- a/lib/Makefile +++ b/lib/Makefile @@ -31,7 +31,7 @@ FLAGS = $(CPPFLAGS) $(CFLAGS) ZSTD_FILES := $(wildcard common/*.c compress/*.c decompress/*.c dictBuilder/*.c deprecated/*.c) -ZSTD_LEGACY_SUPPORT ?= 1 +ZSTD_LEGACY_SUPPORT ?= 4 ifneq ($(ZSTD_LEGACY_SUPPORT), 0) ifeq ($(shell test $(ZSTD_LEGACY_SUPPORT) -lt 8; echo $$?), 0) diff --git a/programs/Makefile b/programs/Makefile index 0bf194260..beeb0711c 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -42,7 +42,7 @@ ZSTD_FILES := $(ZSTDDECOMP_FILES) $(ZSTDCOMMON_FILES) $(ZSTDCOMP_FILES) ZDICT_FILES := $(ZSTDDIR)/dictBuilder/*.c ZSTDDECOMP_O = $(ZSTDDIR)/decompress/zstd_decompress.o -ZSTD_LEGACY_SUPPORT ?= 1 +ZSTD_LEGACY_SUPPORT ?= 4 ZSTDLEGACY_FILES:= ifneq ($(ZSTD_LEGACY_SUPPORT), 0) ifeq ($(shell test $(ZSTD_LEGACY_SUPPORT) -lt 8; echo $$?), 0) diff --git a/tests/Makefile b/tests/Makefile index 59256f841..9382fe80d 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -167,7 +167,7 @@ longmatch : $(ZSTD_FILES) longmatch.c invalidDictionaries : $(ZSTD_FILES) invalidDictionaries.c $(CC) $(FLAGS) $^ -o $@$(EXT) -legacy : CFLAGS+= -DZSTD_LEGACY_SUPPORT=1 +legacy : CFLAGS+= -DZSTD_LEGACY_SUPPORT=4 legacy : CPPFLAGS+= -I$(ZSTDDIR)/legacy legacy : $(ZSTD_FILES) $(wildcard $(ZSTDDIR)/legacy/*.c) legacy.c $(CC) $(FLAGS) $^ -o $@$(EXT) From 9830aeeea6458124f6086ae9cdbf79a24244e30c Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 13 Mar 2017 17:19:37 -0700 Subject: [PATCH 010/305] Fix legacy support=0 case and accidental double include of version headers --- lib/legacy/zstd_legacy.h | 10 ++-------- 1 file changed, 2 insertions(+), 8 deletions(-) diff --git a/lib/legacy/zstd_legacy.h b/lib/legacy/zstd_legacy.h index 18e22e651..3c9798f88 100644 --- a/lib/legacy/zstd_legacy.h +++ b/lib/legacy/zstd_legacy.h @@ -20,15 +20,9 @@ extern "C" { #include "mem.h" /* MEM_STATIC */ #include "error_private.h" /* ERROR */ #include "zstd.h" /* ZSTD_inBuffer, ZSTD_outBuffer */ -#include "zstd_v01.h" -#include "zstd_v02.h" -#include "zstd_v03.h" -#include "zstd_v04.h" -#include "zstd_v05.h" -#include "zstd_v06.h" -#include "zstd_v07.h" -#ifndef ZSTD_LEGACY_SUPPORT +#if !defined (ZSTD_LEGACY_SUPPORT) || (ZSTD_LEGACY_SUPPORT == 0) +# undef ZSTD_LEGACY_SUPPORT # define ZSTD_LEGACY_SUPPORT 8 #endif From aa8bcf360fe7e4da8b00c250f81893e71aa90c0c Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Mon, 13 Mar 2017 18:11:07 -0700 Subject: [PATCH 011/305] Add xz and lzma support. Finish feature started by @inikep. * Add xz and lzma compression and decompression support to target `xzstd`. * Fix bug in gzip decompression that silently accepted truncated files. * Add gzip frame composition tests. * Add xz/lzma compatibility tests. * Add xz/lzma frame composition tests. --- programs/Makefile | 32 ++++++-- programs/fileio.c | 187 +++++++++++++++++++++++++++++++++++++++------ programs/fileio.h | 6 +- programs/zstdcli.c | 14 +++- tests/playTests.sh | 58 ++++++++++++++ 5 files changed, 261 insertions(+), 36 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index a935c744a..8b97592d5 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -68,12 +68,27 @@ EXT = endif # zlib detection +NO_ZLIB_MSG := ==> no zlib, building zstd without .gz support VOID = /dev/null HAVE_ZLIB := $(shell printf '\#include \nint main(){}' | $(CC) -o have_zlib -x c - -lz 2> $(VOID) && rm have_zlib$(EXT) && echo 1 || echo 0) ifeq ($(HAVE_ZLIB), 1) +ZLIB_MSG := ==> building zstd with .gz compression support ZLIBCPP = -DZSTD_GZCOMPRESS -DZSTD_GZDECOMPRESS ZLIBLD = -lz +else +ZLIB_MSG := $(NO_ZLIB_MSG) endif +# lzma detection +NO_LZMA_MSG := ==> no liblzma, building zstd without .xz/.lzma support +HAVE_LZMA := $(shell printf '\#include \nint main(){}' | $(CC) -o have_lzma -x c - -llzma 2> $(VOID) && rm have_lzma$(EXT) && echo 1 || echo 0) +ifeq ($(HAVE_LZMA), 1) +LZMA_MSG := ==> building zstd with .xz/.lzma compression support +LZMACPP = -DZSTD_LZMACOMPRESS -DZSTD_LZMADECOMPRESS +LZMALD = -llzma +else +LZMA_MSG := $(NO_LZMA_MSG) +endif + .PHONY: default all clean clean_decomp_o install uninstall generate_res @@ -85,14 +100,15 @@ $(ZSTDDECOMP_O): CFLAGS += $(ALIGN_LOOP) zstd : CPPFLAGS += $(ZLIBCPP) zstd : LDFLAGS += $(ZLIBLD) -zstd-nogz : HAVE_ZLIB=0 -zstd zstd-nogz : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) -zstd zstd-nogz : $(ZSTDLIB_OBJ) zstdcli.o fileio.o bench.o datagen.o dibio.o -ifeq ($(HAVE_ZLIB), 1) - @echo "==> building zstd with .gz decompression support " -else - @echo "==> no zlib, building zstd with .zst support only (no .gz support) " -endif +zstd : LZMA_MSG := $(NO_LZMA_MSG) +zstd-nogz : ZLIB_MSG := $(NO_ZLIB_MSG) +zstd-nogz : LZMA_MSG := $(NO_LZMA_MSG) +xzstd : CPPFLAGS += $(ZLIBCPP) $(LZMACPP) +xzstd : LDFLAGS += $(ZLIBLD) $(LZMALD) +zstd zstd-nogz xzstd : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) +zstd zstd-nogz xzstd : $(ZSTDLIB_OBJ) zstdcli.o fileio.o bench.o datagen.o dibio.o + @echo "$(ZLIB_MSG)" + @echo "$(LZMA_MSG)" ifneq (,$(filter Windows%,$(OS))) windres/generate_res.bat endif diff --git a/programs/fileio.c b/programs/fileio.c index 41daa125e..e6481f1fa 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -44,6 +44,9 @@ # define z_const # endif #endif +#if defined(ZSTD_LZMACOMPRESS) || defined(ZSTD_LZMADECOMPRESS) +# include +#endif /*-************************************* @@ -71,7 +74,6 @@ #define MAX_DICT_SIZE (8 MB) /* protection against large input (attack scenario) */ #define FNSPACE 30 -#define GZ_EXTENSION ".gz" /*-************************************* @@ -434,6 +436,65 @@ static unsigned long long FIO_compressGzFrame(cRess_t* ress, const char* srcFile #endif +#ifdef ZSTD_LZMACOMPRESS +static unsigned long long FIO_compressLzmaFrame(cRess_t* ress, const char* srcFileName, U64 const srcFileSize, int compressionLevel, U64* readsize, int plain_lzma) +{ + unsigned long long inFileSize = 0, outFileSize = 0; + lzma_stream strm = LZMA_STREAM_INIT; + lzma_action action = LZMA_RUN; + lzma_ret ret; + + if (compressionLevel < 0) compressionLevel = 0; + if (compressionLevel > 9) compressionLevel = 9; + + if (plain_lzma) { + lzma_options_lzma opt_lzma; + if (lzma_lzma_preset(&opt_lzma, compressionLevel)) EXM_THROW(71, "zstd: %s: lzma_lzma_preset error", srcFileName); + ret = lzma_alone_encoder(&strm, &opt_lzma); /* LZMA */ + if (ret != LZMA_OK) EXM_THROW(71, "zstd: %s: lzma_alone_encoder error %d", srcFileName, ret); + } else { + ret = lzma_easy_encoder(&strm, compressionLevel, LZMA_CHECK_CRC64); /* XZ */ + if (ret != LZMA_OK) EXM_THROW(71, "zstd: %s: lzma_easy_encoder error %d", srcFileName, ret); + } + + strm.next_in = 0; + strm.avail_in = 0; + strm.next_out = ress->dstBuffer; + strm.avail_out = ress->dstBufferSize; + + while (1) { + if (strm.avail_in == 0) { + size_t const inSize = fread(ress->srcBuffer, 1, ress->srcBufferSize, ress->srcFile); + if (inSize == 0) action = LZMA_FINISH; + inFileSize += inSize; + strm.next_in = ress->srcBuffer; + strm.avail_in = inSize; + } + + ret = lzma_code(&strm, action); + + if (ret != LZMA_OK && ret != LZMA_STREAM_END) EXM_THROW(72, "zstd: %s: lzma_code encoding error %d", srcFileName, ret); + { size_t const compBytes = ress->dstBufferSize - strm.avail_out; + if (compBytes) { + if (fwrite(ress->dstBuffer, 1, compBytes, ress->dstFile) != compBytes) EXM_THROW(73, "Write error : cannot write to output file"); + outFileSize += compBytes; + strm.next_out = ress->dstBuffer; + strm.avail_out = ress->dstBufferSize; + } + } + if (!srcFileSize) DISPLAYUPDATE(2, "\rRead : %u MB ==> %.2f%%", (U32)(inFileSize>>20), (double)outFileSize/inFileSize*100) + else DISPLAYUPDATE(2, "\rRead : %u / %u MB ==> %.2f%%", (U32)(inFileSize>>20), (U32)(srcFileSize>>20), (double)outFileSize/inFileSize*100); + if (ret == LZMA_STREAM_END) break; + } + + lzma_end(&strm); + *readsize = inFileSize; + + return outFileSize; +} +#endif + + /*! FIO_compressFilename_internal() : * same as FIO_compressFilename_extRess(), with `ress.desFile` already opened. * @return : 0 : compression completed correctly, @@ -448,14 +509,26 @@ static int FIO_compressFilename_internal(cRess_t ress, U64 compressedfilesize = 0; U64 const fileSize = UTIL_getFileSize(srcFileName); - if (g_compressionType) { + switch (g_compressionType) { + case FIO_zstdCompression: + break; + case FIO_gzipCompression: #ifdef ZSTD_GZCOMPRESS - compressedfilesize = FIO_compressGzFrame(&ress, srcFileName, fileSize, compressionLevel, &readsize); + compressedfilesize = FIO_compressGzFrame(&ress, srcFileName, fileSize, compressionLevel, &readsize); #else - (void)compressionLevel; - EXM_THROW(20, "zstd: %s: file cannot be compressed as gzip (zstd compiled without ZSTD_GZCOMPRESS) -- ignored \n", srcFileName); + (void)compressionLevel; + EXM_THROW(20, "zstd: %s: file cannot be compressed as gzip (zstd compiled without ZSTD_GZCOMPRESS) -- ignored \n", srcFileName); #endif - goto finish; + goto finish; + case FIO_xzCompression: + case FIO_lzmaCompression: +#ifdef ZSTD_LZMACOMPRESS + compressedfilesize = FIO_compressLzmaFrame(&ress, srcFileName, fileSize, compressionLevel, &readsize, g_compressionType==FIO_lzmaCompression); +#else + (void)compressionLevel; + EXM_THROW(20, "zstd: %s: file cannot be compressed as xz/lzma (zstd compiled without ZSTD_LZMACOMPRESS) -- ignored \n", srcFileName); +#endif + goto finish; } /* init */ @@ -763,10 +836,10 @@ static void FIO_fwriteSparseEnd(FILE* file, unsigned storedSkips) { if (storedSkips-->0) { /* implies g_sparseFileSupport>0 */ int const seekResult = LONG_SEEK(file, storedSkips, SEEK_CUR); - if (seekResult != 0) EXM_THROW(69, "Final skip error (sparse file)\n"); + if (seekResult != 0) EXM_THROW(69, "Final skip error (sparse file)"); { const char lastZeroByte[1] = { 0 }; size_t const sizeCheck = fwrite(lastZeroByte, 1, 1, file); - if (sizeCheck != 1) EXM_THROW(69, "Write error : cannot write last zero\n"); + if (sizeCheck != 1) EXM_THROW(69, "Write error : cannot write last zero"); } } } @@ -849,6 +922,7 @@ static unsigned long long FIO_decompressGzFrame(dRess_t* ress, FILE* srcFile, co { unsigned long long outFileSize = 0; z_stream strm; + int flush = Z_NO_FLUSH; int ret; strm.zalloc = Z_NULL; @@ -866,11 +940,12 @@ static unsigned long long FIO_decompressGzFrame(dRess_t* ress, FILE* srcFile, co for ( ; ; ) { if (strm.avail_in == 0) { ress->srcBufferLoaded = fread(ress->srcBuffer, 1, ress->srcBufferSize, srcFile); - if (ress->srcBufferLoaded == 0) break; + if (ress->srcBufferLoaded == 0) flush = Z_FINISH; strm.next_in = (z_const unsigned char*)ress->srcBuffer; strm.avail_in = (uInt)ress->srcBufferLoaded; } - ret = inflate(&strm, Z_NO_FLUSH); + ret = inflate(&strm, flush); + if (ret == Z_BUF_ERROR) EXM_THROW(39, "zstd: %s: premature end", srcFileName); if (ret != Z_OK && ret != Z_STREAM_END) { DISPLAY("zstd: %s: inflate error %d \n", srcFileName, ret); return 0; } { size_t const decompBytes = ress->dstBufferSize - strm.avail_out; if (decompBytes) { @@ -886,7 +961,60 @@ static unsigned long long FIO_decompressGzFrame(dRess_t* ress, FILE* srcFile, co if (strm.avail_in > 0) memmove(ress->srcBuffer, strm.next_in, strm.avail_in); ress->srcBufferLoaded = strm.avail_in; ret = inflateEnd(&strm); - if (ret != Z_OK) EXM_THROW(32, "zstd: %s: inflateEnd error %d \n", srcFileName, ret); + if (ret != Z_OK) EXM_THROW(32, "zstd: %s: inflateEnd error %d", srcFileName, ret); + return outFileSize; +} +#endif + + +#ifdef ZSTD_LZMADECOMPRESS +static unsigned long long FIO_decompressLzmaFrame(dRess_t* ress, FILE* srcFile, const char* srcFileName, int plain_lzma) +{ + unsigned long long outFileSize = 0; + lzma_stream strm = LZMA_STREAM_INIT; + lzma_action action = LZMA_RUN; + lzma_ret ret; + + strm.next_in = 0; + strm.avail_in = 0; + if (plain_lzma) { + ret = lzma_alone_decoder(&strm, UINT64_MAX); /* LZMA */ + } else { + ret = lzma_stream_decoder(&strm, UINT64_MAX, 0); /* XZ */ + } + + if (ret != LZMA_OK) EXM_THROW(71, "zstd: %s: lzma_alone_decoder/lzma_stream_decoder error %d", srcFileName, ret); + + strm.next_out = ress->dstBuffer; + strm.avail_out = ress->dstBufferSize; + strm.avail_in = ress->srcBufferLoaded; + strm.next_in = ress->srcBuffer; + + for ( ; ; ) { + if (strm.avail_in == 0) { + ress->srcBufferLoaded = fread(ress->srcBuffer, 1, ress->srcBufferSize, srcFile); + if (ress->srcBufferLoaded == 0) action = LZMA_FINISH; + strm.next_in = ress->srcBuffer; + strm.avail_in = ress->srcBufferLoaded; + } + ret = lzma_code(&strm, action); + + if (ret == LZMA_BUF_ERROR) EXM_THROW(39, "zstd: %s: premature end", srcFileName); + if (ret != LZMA_OK && ret != LZMA_STREAM_END) { DISPLAY("zstd: %s: lzma_code decoding error %d \n", srcFileName, ret); return 0; } + { size_t const decompBytes = ress->dstBufferSize - strm.avail_out; + if (decompBytes) { + if (fwrite(ress->dstBuffer, 1, decompBytes, ress->dstFile) != decompBytes) EXM_THROW(31, "Write error : cannot write to output file"); + outFileSize += decompBytes; + strm.next_out = ress->dstBuffer; + strm.avail_out = ress->dstBufferSize; + } + } + if (ret == LZMA_STREAM_END) break; + } + + if (strm.avail_in > 0) memmove(ress->srcBuffer, strm.next_in, strm.avail_in); + ress->srcBufferLoaded = strm.avail_in; + lzma_end(&strm); return outFileSize; } #endif @@ -924,7 +1052,7 @@ static int FIO_decompressSrcFile(dRess_t ress, const char* dstFileName, const ch } readSomething = 1; /* there is at least >= 4 bytes in srcFile */ if (ress.srcBufferLoaded < toRead) { DISPLAY("zstd: %s: unknown header \n", srcFileName); fclose(srcFile); return 1; } /* srcFileName is empty */ - if (buf[0] == 31 && buf[1] == 139) { /* gz header */ + if (buf[0] == 31 && buf[1] == 139) { /* gz magic number */ #ifdef ZSTD_GZDECOMPRESS unsigned long long const result = FIO_decompressGzFrame(&ress, srcFile, srcFileName); if (result == 0) return 1; @@ -932,6 +1060,16 @@ static int FIO_decompressSrcFile(dRess_t ress, const char* dstFileName, const ch #else DISPLAYLEVEL(1, "zstd: %s: gzip file cannot be uncompressed (zstd compiled without ZSTD_GZDECOMPRESS) -- ignored \n", srcFileName); return 1; +#endif + } else if ((buf[0] == 0xFD && buf[1] == 0x37) /* xz magic number */ + || (buf[0] == 0x5D && buf[1] == 0x00)) { /* lzma header (no magic number) */ +#ifdef ZSTD_LZMADECOMPRESS + unsigned long long const result = FIO_decompressLzmaFrame(&ress, srcFile, srcFileName, buf[0] != 0xFD); + if (result == 0) return 1; + filesize += result; +#else + DISPLAYLEVEL(1, "zstd: %s: xz/lzma file cannot be uncompressed (zstd compiled without ZSTD_LZMADECOMPRESS) -- ignored \n", srcFileName); + return 1; #endif } else { if (!ZSTD_isFrame(ress.srcBuffer, toRead)) { @@ -1020,32 +1158,31 @@ int FIO_decompressMultipleFilenames(const char** srcNamesTable, unsigned nbFiles missingFiles += FIO_decompressSrcFile(ress, suffix, srcNamesTable[u]); if (fclose(ress.dstFile)) EXM_THROW(72, "Write error : cannot properly close stdout"); } else { - size_t const suffixSize = strlen(suffix); - size_t const gzipSuffixSize = strlen(GZ_EXTENSION); + size_t suffixSize; size_t dfnSize = FNSPACE; unsigned u; char* dstFileName = (char*)malloc(FNSPACE); if (dstFileName==NULL) EXM_THROW(73, "not enough memory for dstFileName"); for (u=0; u tmp + $ZSTD -f --format=gzip tmp + $ZSTD -f tmp + cat tmp.gz tmp.zst tmp.gz tmp.zst | $ZSTD -d -f -o tmp + head -c -1 tmp.gz | $ZSTD -t && die "incomplete frame not detected !" + rm tmp* +else + $ECHO "gzip mode not supported" +fi + + +$ECHO "\n**** xz compatibility tests **** " + +LZMAMODE=1 +$ZSTD --format=xz -V || LZMAMODE=0 +if [ $LZMAMODE -eq 1 ]; then + $ECHO "xz support detected" + XZEXE=1 + xz -V && lzma -V || XZEXE=0 + if [ $XZEXE -eq 1 ]; then + ./datagen > tmp + $ZSTD --format=lzma -f tmp + $ZSTD --format=xz -f tmp + xz -t -v tmp.xz + xz -t -v tmp.lzma + xz -f -k tmp + lzma -f -k --lzma1 tmp + $ZSTD -d -f -v tmp.xz + $ZSTD -d -f -v tmp.lzma + rm tmp* + else + $ECHO "xz binary not detected" + fi +else + $ECHO "xz mode not supported" +fi + + +$ECHO "\n**** xz frame tests **** " + +if [ $LZMAMODE -eq 1 ]; then + ./datagen > tmp + $ZSTD -f --format=xz tmp + $ZSTD -f --format=lzma tmp + $ZSTD -f tmp + cat tmp.xz tmp.lzma tmp.zst tmp.lzma tmp.xz tmp.zst | $ZSTD -d -f -o tmp + head -c -1 tmp.xz | $ZSTD -t && die "incomplete frame not detected !" + head -c -1 tmp.lzma | $ZSTD -t && die "incomplete frame not detected !" + rm tmp* +else + $ECHO "xz mode not supported" +fi + + $ECHO "\n**** zstd round-trip tests **** " roundTripTest From 7ae3039f41c3277a659e828752a3fe84f2a5e221 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 14 Mar 2017 04:19:51 -0700 Subject: [PATCH 012/305] updated NEWS for v1.1.4 --- NEWS | 2 ++ 1 file changed, 2 insertions(+) diff --git a/NEWS b/NEWS index 9073a8724..bbd9e1688 100644 --- a/NEWS +++ b/NEWS @@ -3,10 +3,12 @@ cli : new : can compress in *.gz format, using --format=gzip command, by Przemys cli : new : advanced benchmark command --priority=rt cli : fix : write on sparse-enabled file systems in 32-bits mode, by @ds77 cli : fix : --rm remains silent when input is stdin +cli : experimental : xzstd, with support for xz/lzma decoding, by Przemyslaw Skibinski speed : improved decompression speed in streaming mode for single shot scenarios (+5%) memory : DDict (decompression dictionary) memory usage down from 150 KB to 20 KB arch : 32-bits variant able to generate and decode very long matches (>32 MB), by Sean Purcell API : new : ZSTD_findFrameCompressedSize(), ZSTD_getFrameContentSize(), ZSTD_findDecompressedSize() +API : changed : dropped support of legacy versions <= v0.3 (can be changed by modifying ZSTD_LEGACY_SUPPORT value) build: new: meson build system in contrib/meson, by Dima Krasner build: improved cmake script, by @Majlen build: added -Wformat-security flag, as recommended by Padraig Brady From dec2b96536893f16923f149d3eab9c8474e8dd4b Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 14 Mar 2017 11:24:09 -0700 Subject: [PATCH 013/305] Add functions missing from manual, and fix parameter alignment --- contrib/gen_html/gen_html.cpp | 12 ++++--- doc/zstd_manual.html | 67 +++++++++++++++++++++++------------ lib/zstd.h | 12 ++++++- 3 files changed, 64 insertions(+), 27 deletions(-) diff --git a/contrib/gen_html/gen_html.cpp b/contrib/gen_html/gen_html.cpp index 22ff65b10..e5261c086 100644 --- a/contrib/gen_html/gen_html.cpp +++ b/contrib/gen_html/gen_html.cpp @@ -19,7 +19,7 @@ void trim(string& s, string characters) { size_t p = s.find_first_not_of(characters); s.erase(0, p); - + p = s.find_last_not_of(characters); if (string::npos != p) s.erase(p+1); @@ -48,7 +48,7 @@ vector get_lines(vector& input, int& linenum, string terminator) line = input[linenum]; if (terminator.empty() && line.empty()) { linenum--; break; } - + epos = line.find(terminator); if (!terminator.empty() && epos!=string::npos) { out.push_back(line); @@ -168,7 +168,11 @@ int main(int argc, char *argv[]) { sout << "
";
             for (l=0; l

"; for (l=0; l" << endl << "" << endl; return 0; -} \ No newline at end of file +} diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index 77e8974de..204f56ea5 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -55,8 +55,8 @@

Simple API


 
 
size_t ZSTD_compress( void* dst, size_t dstCapacity,
-                            const void* src, size_t srcSize,
-                                  int compressionLevel);
+                const void* src, size_t srcSize,
+                      int compressionLevel);
 

Compresses `src` content as a single zstd compressed frame into already allocated `dst`. Hint : compression runs faster if `dstCapacity` >= `ZSTD_compressBound(srcSize)`. @return : compressed size written into `dst` (<= `dstCapacity), @@ -64,7 +64,7 @@


size_t ZSTD_decompress( void* dst, size_t dstCapacity,
-                              const void* src, size_t compressedSize);
+                  const void* src, size_t compressedSize);
 

`compressedSize` : must be the _exact_ size of some number of compressed and/or skippable frames. `dstCapacity` is an upper bound of originalSize. If user cannot imply a maximum upper bound, it's better to use streaming mode to decompress data. @@ -118,7 +118,11 @@ size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx);

Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()).


-

Decompression context

typedef struct ZSTD_DCtx_s ZSTD_DCtx;
+

Decompression context

   When decompressing many times,
+   it is recommended to allocate a context just once, and re-use it for each successive compression operation.
+   This will make workload friendlier for system's memory.
+   Use one context per thread for parallel execution in multi-threaded environments. 
+
typedef struct ZSTD_DCtx_s ZSTD_DCtx;
 ZSTD_DCtx* ZSTD_createDCtx(void);
 size_t     ZSTD_freeDCtx(ZSTD_DCtx* dctx);
 

@@ -129,19 +133,19 @@ size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx);

Simple dictionary API


 
 
size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx,
-                                           void* dst, size_t dstCapacity,
-                                     const void* src, size_t srcSize,
-                                     const void* dict,size_t dictSize,
-                                           int compressionLevel);
+                               void* dst, size_t dstCapacity,
+                         const void* src, size_t srcSize,
+                         const void* dict,size_t dictSize,
+                               int compressionLevel);
 

Compression using a predefined Dictionary (see dictBuilder/zdict.h). Note : This function loads the dictionary, resulting in significant startup delay. Note : When `dict == NULL || dictSize < 8` no dictionary is used.


size_t ZSTD_decompress_usingDict(ZSTD_DCtx* dctx,
-                                             void* dst, size_t dstCapacity,
-                                       const void* src, size_t srcSize,
-                                       const void* dict,size_t dictSize);
+                                 void* dst, size_t dstCapacity,
+                           const void* src, size_t srcSize,
+                           const void* dict,size_t dictSize);
 

Decompression using a predefined Dictionary (see dictBuilder/zdict.h). Dictionary must be identical to the one used during compression. Note : This function loads the dictionary, resulting in significant startup delay. @@ -162,9 +166,9 @@ size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx);


size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx,
-                                            void* dst, size_t dstCapacity,
-                                      const void* src, size_t srcSize,
-                                      const ZSTD_CDict* cdict);
+                                void* dst, size_t dstCapacity,
+                          const void* src, size_t srcSize,
+                          const ZSTD_CDict* cdict);
 

Compression using a digested Dictionary. Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. Note that compression level is decided during dictionary creation. @@ -180,9 +184,9 @@ size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx);


size_t ZSTD_decompress_usingDDict(ZSTD_DCtx* dctx,
-                                              void* dst, size_t dstCapacity,
-                                        const void* src, size_t srcSize,
-                                        const ZSTD_DDict* ddict);
+                                  void* dst, size_t dstCapacity,
+                            const void* src, size_t srcSize,
+                            const ZSTD_DDict* ddict);
 

Decompression using a digested Dictionary. Faster startup than ZSTD_decompress_usingDict(), recommended when same dictionary is used multiple times.


@@ -239,6 +243,14 @@ size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx);
+

ZSTD_CStream management functions

ZSTD_CStream* ZSTD_createCStream(void);
+size_t ZSTD_freeCStream(ZSTD_CStream* zcs);
+

+

Streaming compression functions

size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel);
+size_t ZSTD_compressStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input);
+size_t ZSTD_flushStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output);
+size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output);
+

size_t ZSTD_CStreamInSize(void);    /**< recommended size for input buffer */
 

size_t ZSTD_CStreamOutSize(void);   /**< recommended size for output buffer. Guarantee to successfully flush at least one complete compressed block in all circumstances. */
@@ -264,6 +276,12 @@ size_t     ZSTD_freeDCtx(ZSTD_DCtx* dctx);
  
 
+

ZSTD_DStream management functions

ZSTD_DStream* ZSTD_createDStream(void);
+size_t ZSTD_freeDStream(ZSTD_DStream* zds);
+

+

Streaming decompression functions

size_t ZSTD_initDStream(ZSTD_DStream* zds);
+size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input);
+

size_t ZSTD_DStreamInSize(void);    /*!< recommended size for input buffer */
 

size_t ZSTD_DStreamOutSize(void);   /*!< recommended size for output buffer. Guarantee to successfully flush at least one complete block in all circumstances. */
@@ -381,7 +399,7 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v
 


ZSTD_CDict* ZSTD_createCDict_advanced(const void* dict, size_t dictSize, unsigned byReference,
-                                                  ZSTD_parameters params, ZSTD_customMem customMem);
+                                      ZSTD_parameters params, ZSTD_customMem customMem);
 

Create a ZSTD_CDict using external alloc and free, and customized compression parameters


@@ -409,10 +427,10 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v


size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx,
-                                           void* dst, size_t dstCapacity,
-                                     const void* src, size_t srcSize,
-                                     const void* dict,size_t dictSize,
-                                           ZSTD_parameters params);
+                               void* dst, size_t dstCapacity,
+                         const void* src, size_t srcSize,
+                         const void* dict,size_t dictSize,
+                               ZSTD_parameters params);
 

Same as ZSTD_compress_usingDict(), with fine-tune control of each compression parameter


@@ -443,6 +461,11 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v It is important that dictBuffer outlives DDict, it must remain read accessible throughout the lifetime of DDict


+
ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize,
+                                      unsigned byReference, ZSTD_customMem customMem);
+

Create a ZSTD_DDict using external alloc and free, optionally by reference +


+
size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict);
 

Gives the amount of memory used by a given ZSTD_DDict


diff --git a/lib/zstd.h b/lib/zstd.h index 0b48b7ec6..a3237c77e 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -139,7 +139,11 @@ ZSTDLIB_API size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx); Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). */ ZSTDLIB_API size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel); -/*= Decompression context */ +/*= Decompression context +* When decompressing many times, +* it is recommended to allocate a context just once, and re-use it for each successive compression operation. +* This will make workload friendlier for system's memory. +* Use one context per thread for parallel execution in multi-threaded environments. */ typedef struct ZSTD_DCtx_s ZSTD_DCtx; ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx(void); ZSTDLIB_API size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx); @@ -277,9 +281,11 @@ typedef struct ZSTD_outBuffer_s { * *******************************************************************/ typedef struct ZSTD_CStream_s ZSTD_CStream; +/*===== ZSTD_CStream management functions =====*/ ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream(void); ZSTDLIB_API size_t ZSTD_freeCStream(ZSTD_CStream* zcs); +/*===== Streaming compression functions =====*/ ZSTDLIB_API size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel); ZSTDLIB_API size_t ZSTD_compressStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input); ZSTDLIB_API size_t ZSTD_flushStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output); @@ -313,9 +319,11 @@ ZSTDLIB_API size_t ZSTD_CStreamOutSize(void); /**< recommended size for output * *******************************************************************************/ typedef struct ZSTD_DStream_s ZSTD_DStream; +/*===== ZSTD_DStream management functions =====*/ ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream(void); ZSTDLIB_API size_t ZSTD_freeDStream(ZSTD_DStream* zds); +/*===== Streaming decompression functions =====*/ ZSTDLIB_API size_t ZSTD_initDStream(ZSTD_DStream* zds); ZSTDLIB_API size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input); @@ -540,6 +548,8 @@ ZSTDLIB_API size_t ZSTD_sizeof_DCtx(const ZSTD_DCtx* dctx); * It is important that dictBuffer outlives DDict, it must remain read accessible throughout the lifetime of DDict */ ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_byReference(const void* dictBuffer, size_t dictSize); +/*! ZSTD_createDDict_advanced() : + * Create a ZSTD_DDict using external alloc and free, optionally by reference */ ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize, unsigned byReference, ZSTD_customMem customMem); From f5e50519e01679a2664273f420da6fb77f05ec48 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Wed, 15 Mar 2017 15:04:54 -0700 Subject: [PATCH 014/305] Prevent fuzz testers from combining large dicts with high clevel --- tests/fuzzer.c | 6 ++++-- tests/zstreamtest.c | 10 ++++++---- 2 files changed, 10 insertions(+), 6 deletions(-) diff --git a/tests/fuzzer.c b/tests/fuzzer.c index 6fb69972a..def7542b5 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -66,6 +66,7 @@ static clock_t g_displayClock = 0; * Fuzzer functions *********************************************************/ #define MIN(a,b) ((a)<(b)?(a):(b)) +#define MAX(a,b) ((a)>(b)?(a):(b)) static clock_t FUZ_clockSpan(clock_t cStart) { @@ -799,11 +800,12 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxD /*===== Streaming compression test, scattered segments and dictionary =====*/ { U32 const testLog = FUZ_rand(&lseed) % maxSrcLog; - int const cLevel = (FUZ_rand(&lseed) % (ZSTD_maxCLevel() - (testLog/3))) + 1; + U32 const dictLog = FUZ_rand(&lseed) % maxSrcLog; + int const cLevel = (FUZ_rand(&lseed) % (ZSTD_maxCLevel() - (MAX(testLog, dictLog)/3))) + 1; maxTestSize = FUZ_rLogLength(&lseed, testLog); if (maxTestSize >= dstBufferSize) maxTestSize = dstBufferSize-1; - dictSize = FUZ_randomLength(&lseed, maxSampleLog); /* needed also for decompression */ + dictSize = FUZ_rLogLength(&lseed, dictLog); /* needed also for decompression */ dict = srcBuffer + (FUZ_rand(&lseed) % (srcBufferSize - dictSize)); if (FUZ_rand(&lseed) & 0xF) { diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index 54b890266..aa7367bcf 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -645,11 +645,12 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, double compres } } else { U32 const testLog = FUZ_rand(&lseed) % maxSrcLog; - U32 const cLevel = (FUZ_rand(&lseed) % (ZSTD_maxCLevel() - (testLog/3))) + 1; + U32 const dictLog = FUZ_rand(&lseed) % maxSrcLog; + U32 const cLevel = (FUZ_rand(&lseed) % (ZSTD_maxCLevel() - (MAX(testLog, dictLog)/3))) + 1; maxTestSize = FUZ_rLogLength(&lseed, testLog); oldTestLog = testLog; /* random dictionary selection */ - dictSize = ((FUZ_rand(&lseed)&63)==1) ? FUZ_randomLength(&lseed, maxSampleLog) : 0; + dictSize = ((FUZ_rand(&lseed)&63)==1) ? FUZ_rLogLength(&lseed, dictLog) : 0; { size_t const dictStart = FUZ_rand(&lseed) % (srcBufferSize - dictSize); dict = srcBuffer + dictStart; } @@ -886,11 +887,12 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp } } else { U32 const testLog = FUZ_rand(&lseed) % maxSrcLog; - U32 const cLevel = (FUZ_rand(&lseed) % (ZSTD_maxCLevel() - (testLog/3))) + 1; + U32 const dictLog = FUZ_rand(&lseed) % maxSrcLog; + U32 const cLevel = (FUZ_rand(&lseed) % (ZSTD_maxCLevel() - (MAX(testLog, dictLog)/3))) + 1; maxTestSize = FUZ_rLogLength(&lseed, testLog); oldTestLog = testLog; /* random dictionary selection */ - dictSize = ((FUZ_rand(&lseed)&63)==1) ? FUZ_randomLength(&lseed, maxSampleLog) : 0; + dictSize = ((FUZ_rand(&lseed)&63)==1) ? FUZ_rLogLength(&lseed, dictLog) : 0; { size_t const dictStart = FUZ_rand(&lseed) % (srcBufferSize - dictSize); dict = srcBuffer + dictStart; } From 72a243e3bdcd73cb2b646fa91c969820c98662c5 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 16 Mar 2017 15:33:21 -0700 Subject: [PATCH 015/305] Convert manpage to Markdown, converted with ronn --- programs/Makefile | 25 ++ programs/zstd.1 | 629 +++++++++++++++++++-------------------------- programs/zstd.1.md | 299 +++++++++++++++++++++ 3 files changed, 583 insertions(+), 370 deletions(-) create mode 100644 programs/zstd.1.md diff --git a/programs/Makefile b/programs/Makefile index 1475cb610..bb91e069a 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -18,6 +18,19 @@ ZSTDDIR = ../lib +# Version numbers +LIBVER_SRC := $(ZSTDDIR)/zstd.h +LIBVER_MAJOR_SCRIPT:=`sed -n '/define ZSTD_VERSION_MAJOR/s/.*[[:blank:]]\([0-9][0-9]*\).*/\1/p' < $(LIBVER_SRC)` +LIBVER_MINOR_SCRIPT:=`sed -n '/define ZSTD_VERSION_MINOR/s/.*[[:blank:]]\([0-9][0-9]*\).*/\1/p' < $(LIBVER_SRC)` +LIBVER_PATCH_SCRIPT:=`sed -n '/define ZSTD_VERSION_RELEASE/s/.*[[:blank:]]\([0-9][0-9]*\).*/\1/p' < $(LIBVER_SRC)` +LIBVER_SCRIPT:= $(LIBVER_MAJOR_SCRIPT).$(LIBVER_MINOR_SCRIPT).$(LIBVER_PATCH_SCRIPT) +LIBVER_MAJOR := $(shell echo $(LIBVER_MAJOR_SCRIPT)) +LIBVER_MINOR := $(shell echo $(LIBVER_MINOR_SCRIPT)) +LIBVER_PATCH := $(shell echo $(LIBVER_PATCH_SCRIPT)) +LIBVER := $(shell echo $(LIBVER_SCRIPT)) + +ZSTD_VERSION=$(LIBVER) + ifeq ($(shell $(CC) -v 2>&1 | grep -c "gcc version "), 1) ALIGN_LOOP = -falign-loops=32 else @@ -91,6 +104,8 @@ else LZMA_MSG := $(NO_LZMA_MSG) endif +MD2ROFF =ronn +MD2ROFF_FLAGS = --roff --warnings --manual="User Commands" --organization="zstd $(ZSTD_VERSION)" .PHONY: default all clean clean_decomp_o install uninstall generate_res @@ -174,6 +189,16 @@ clean: clean_decomp_o: @$(RM) $(ZSTDDECOMP_O) +zstd.1: zstd.1.md + cat $^ | $(MD2ROFF) $(MD2ROFF_FLAGS) | sed -n '/^\.\\\".*/!p' > $@ + +man: zstd.1 + +clean-man: + rm zstd.1 + +preview-man: clean-man man + man ./zstd.1 #----------------------------------------------------------------------------- # make install is validated only for Linux, OSX, BSD, Hurd and Solaris targets diff --git a/programs/zstd.1 b/programs/zstd.1 index 684fb868a..9ac4ca0a8 100644 --- a/programs/zstd.1 +++ b/programs/zstd.1 @@ -1,408 +1,297 @@ -\" -\" zstd.1: This is a manual page for 'zstd' program. This file is part of the -\" zstd project. -\" Author: Yann Collet -\" - -\" No hyphenation -.hy 0 -.nr HY 0 - -.TH zstd "1" "2015-08-22" "zstd" "User Commands" -.SH NAME -\fBzstd, unzstd, zstdcat\fR - Compress or decompress .zst files - -.SH SYNOPSIS -.TP 5 -\fBzstd\fR [\fBOPTIONS\fR] [-|INPUT-FILE] [-o ] -.PP -.B unzstd -is equivalent to -.BR "zstd \-d" -.br -.B zstdcat -is equivalent to -.BR "zstd \-dcf" -.br - -.SH DESCRIPTION -.PP -\fBzstd\fR is a fast lossless compression algorithm -and data compression tool, -with command line syntax similar to \fB gzip (1) \fR and \fB xz (1) \fR . -It is based on the \fBLZ77\fR family, with further FSE & huff0 entropy stages. -\fBzstd\fR offers highly configurable compression speed, -with fast modes at > 200 MB/s per core, -and strong modes nearing lzma compression ratios. -It also features a very fast decoder, with speeds > 500 MB/s per core. - -\fBzstd\fR command line syntax is generally similar to gzip, -but features the following differences : - - Source files are preserved by default. - It's possible to remove them automatically by using \fB--rm\fR command. - - When compressing a single file, \fBzstd\fR displays progress notifications and result summary by default. - Use \fB-q\fR to turn them off - -.PP -.B zstd -compresses or decompresses each -.I file -according to the selected operation mode. -If no -.I files -are given or -.I file -is -.BR \- , -.B zstd -reads from standard input and writes the processed data -to standard output. -.B zstd -will refuse (display an error and skip the -.IR file ) -to write compressed data to standard output if it is a terminal. -Similarly, -.B zstd -will refuse to read compressed data -from standard input if it is a terminal. - -.PP -Unless -.B \-\-stdout -or -.B \-o -is specified, -.I files -are written to a new file whose name is derived from the source -.I file -name: -.IP \(bu 3 -When compressing, the suffix -.B .zst -is appended to the source filename to get the target filename. -.IP \(bu 3 -When decompressing, the -.B .zst -suffix is removed from the filename to get the target filename. - -.SS "Concatenation with .zst files" -It is possible to concatenate -.B .zst -files as is. -.B zstd -will decompress such files as if they were a single -.B .zst -file. - - - -.SH OPTIONS - +. +.TH "ZSTD" "1" "March 2017" "zstd 1.1.4" "User Commands" +. +.SH "NAME" +\fBzstd\fR \- zstd, unzstd, zstdcat \- Compress or decompress \.zst files +. +.SH "SYNOPSIS" +\fBzstd\fR [\fIOPTIONS\fR] [\-|] [\-o ] +. +.P +\fBunzstd\fR is equivalent to \fBzstd \-d\fR \fBzstdcat\fR is equivalent to \fBzstd \-dcf\fR +. +.SH "DESCRIPTION" +\fBzstd\fR is a fast lossless compression algorithm and data compression tool, with command line syntax similar to \fBgzip (1)\fR and \fBxz (1)\fR\. It is based on the \fBLZ77\fR family, with further FSE & huff0 entropy stages\. \fBzstd\fR offers highly configurable compression speed, with fast modes at > 200 MB/s per code, and strong modes nearing lzma compression ratios\. It also features a very fast decoder, with speeds > 500 MB/s per core\. +. +.P +\fBzstd\fR command line syntax is generally similar to gzip, but features the following differences : +. +.IP "\(bu" 4 +Source files are preserved by default\. It\'s possible to remove them automatically by using the \fB\-\-rm\fR command\. +. +.IP "\(bu" 4 +When compressing a single file, \fBzstd\fR displays progress notifications and result summary by default\. Use \fB\-q\fR to turn them off\. +. +.IP "" 0 +. +.P +\fBzstd\fR compresses or decompresses each \fIfile\fR according to the selected operation mode\. If no \fIfiles\fR are given or \fIfile\fR is \fB\-\fR, \fBzstd\fR reads from standard input and writes the processed data to standard output\. \fBzstd\fR will refuse to (display an error and skip the \fIfile\fR) to write compressed data to standard output if it is a terminal\. Similarly, \fBzstd\fR will refuse to read compressed data from standard input if it is a terminal\. +. +.P +Unless \fB\-\-stdout\fR or \fB\-o\fR is specified, \fIfiles\fR are written to a new file whose name is derived from the source \fIfile\fR name: +. +.IP "\(bu" 4 +When compressing, the suffix \fB\.zst\fR is appended to the source filename to get the target filename\. +. +.IP "\(bu" 4 +When decompressing, the \fB\.zst\fR suffix is removed from the source filename to get the target filename +. +.IP "" 0 +. +.SS "Concatenation with \.zst files" +It is possible to concatenate \fB\.zst\fR files as is\. \fBzstd\fR will decompress such files as if they were a single \fB\.zst\fR file\. +. +.SH "OPTIONS" . .SS "Integer suffixes and special values" -In most places where an integer argument is expected, -an optional suffix is supported to easily indicate large integers. -There must be no space between the integer and the suffix. +In most places where an integer argument is expected, an optional suffix is supported to easily indicate large integers\. There must be no space between the integer and the suffix\. +. .TP -.B KiB -Multiply the integer by 1,024 (2^10). -.BR Ki , -.BR K , -and -.B KB -are accepted as synonyms for -.BR KiB . +\fBKiB\fR +Multiply the integer by 1,024 (2^10)\. \fBKi\fR, \fBK\fR, and \fBKB\fR are accepted as synonyms for \fBKiB\fR\. +. .TP -.B MiB -Multiply the integer by 1,048,576 (2^20). -.BR Mi , -.BR M , -and -.B MB -are accepted as synonyms for -.BR MiB . - +\fBMiB\fR +Multiply the integer by 1,048,576 (2^20)\. \fBMi\fR, \fBM\fR, and \fBMB\fR are accepted as synonyms for \fBMiB\fR\. . .SS "Operation mode" -If multiple operation mode options are given, -the last one takes effect. +If multiple operation mode options are given, the last one takes effect\. +. .TP -.BR \-z ", " \-\-compress -Compress. -This is the default operation mode when no operation mode option -is specified and no other operation mode is implied from -the command name (for example, -.B unzstd -implies -.BR \-\-decompress ). +\fB\-z\fR, \fB\-\-compress\fR +Compress\. This is the default operation mode when no operation mode option is specified and no other operation mode is implied from the command name (for example, \fBunzstd\fR implies \fB\-\-decompress\fR)\. +. .TP -.BR \-d ", " \-\-decompress ", " \-\-uncompress -Decompress. +\fB\-d\fR, \fB\-\-decompress\fR, \fB\-\-uncompress\fR +Decompress\. +. .TP -.BR \-t ", " \-\-test -Test the integrity of compressed -.IR files . -This option is equivalent to -.B "\-\-decompress \-\-stdout" -except that the decompressed data is discarded instead of being -written to standard output. -No files are created or removed. +\fB\-t\fR, \fB\-\-test\fR +Test the integrity of compressed \fIfiles\fR\. This option is equivalent to \fB\-\-decompress \-\-stdout\fR except that the decompressed data is discarded instead of being written to standard output\. No files are created or removed\. +. .TP -.B \-b# - benchmark file(s) using compression level # +\fB\-b#\fR +Benchmark file(s) using compression level # +. .TP -.B \--train FILEs - use FILEs as training set to create a dictionary. The training set should contain a lot of small files (> 100). - +\fB\-\-train FILEs\fR +Use FILEs as a training set to create a dictionary\. The training set should contain a lot of small files (> 100)\. . .SS "Operation modifiers" +. .TP -.B \-# - # compression level [1-19] (default:3) +\fB\-#\fR +\fB#\fR compression level [1\-19] (default: 3) +. .TP -.BR \--ultra - unlocks high compression levels 20+ (maximum 22), using a lot more memory. -Note that decompression will also require more memory when using these levels. +\fB\-\-ultra\fR +unlocks high compression levels 20+ (maximum 22), using a lot more memory\. Note that decompression will also require more memory when using these levels\. +. .TP -.B \-D file - use `file` as Dictionary to compress or decompress FILE(s) +\fB\-D file\fR +use \fBfile\fR as Dictionary to compress or decompress FILE(s) +. .TP -.BR \--no-dictID - do not store dictionary ID within frame header (dictionary compression). - The decoder will have to rely on implicit knowledge about which dictionary to use, -it won't be able to check if it's correct. +\fB\-\-nodictID\fR +do not store dictionary ID within frame header (dictionary compression)\. The decoder will have to rely on implicit knowledge about which dictionary to use, it won\'t be able to check if it\'s correct\. +. .TP -.B \-o file - save result into `file` (only possible with a single INPUT-FILE) +\fB\-o file\fR +save result into \fBfile\fR (only possible with a single INPUT\-FILE) +. .TP -.BR \-f ", " --force - overwrite output without prompting +\fB\-f\fR, \fB\-\-force\fR +overwrite output without prompting +. .TP -.BR \-c ", " --stdout - force write to standard output, even if it is the console +\fB\-c\fR, \fB\-\-stdout\fR +force write to standard output, even if it is the console +. .TP -.BR \--[no-]sparse - enable / disable sparse FS support, to make files with many zeroes smaller on disk. - Creating sparse files may save disk space and speed up the decompression -by reducing the amount of disk I/O. - default : enabled when output is into a file, and disabled when output is stdout. - This setting overrides default and can force sparse mode over stdout. +\fB\-\-[no\-]sparse\fR +enable / disable sparse FS support, to make files with many zeroes smaller on disk\. Creating sparse files may save disk space and speed up decompression by reducing the amount of disk I/O\. default : enabled when output is into a file, and disabled when output is stdout\. This setting overrides default and can force sparse mode over stdout\. +. .TP -.BR \--rm - remove source file(s) after successful compression or decompression +\fB\-\-rm\fR +remove source file(s) after successful compression or decompression +. .TP -.BR \-k ", " --keep - keep source file(s) after successful compression or decompression. - This is the default behavior. +\fB\-k\fR, \fB\-\-keep\fR +keep source file(s) after successful compression or decompression\. This is the default behaviour\. +. .TP -.BR \-r - operate recursively on directories +\fB\-r\fR +operate recursively on dictionaries +. .TP -.BR \-h/\-H ", " --help - display help/long help and exit +\fB\-h\fR/\fB\-H\fR, \fB\-\-help\fR +display help/long help and exit +. .TP -.BR \-V ", " --version - display Version number and exit +\fB\-V\fR, \fB\-\-version\fR +display version number and exit +. .TP -.BR \-v ", " --verbose - verbose mode +\fB\-v\fR +verbose mode +. .TP -.BR \-q ", " --quiet - suppress warnings, interactivity and notifications. - specify twice to suppress errors too. +\fB\-q\fR, \fB\-\-quiet\fR +suppress warnings, interactivity, and notifications\. specify twice to suppress errors too\. +. .TP -.BR \-C ", " --[no-]check - add integrity check computed from uncompressed data (default : enabled) +\fB\-C\fR, \fB\-\-[no\-]check\fR +add integrety check computed from uncompressed data (default : enabled) +. .TP -.BR \-t ", " --test - Test the integrity of compressed files. This option is equivalent to \fB--decompress --stdout > /dev/null\fR. - No files are created or removed. +\fB\-\-\fR +All arguments after \fB\-\-\fR are treated as files +. +.SH "DICTIONARY BUILDER" +\fBzstd\fR offers \fIdictionary\fR compression, useful for very small files and messages\. It\'s possible to train \fBzstd\fR with some samples, the result of which is saved into a file called a \fBdictionary\fR\. Then during compression and decompression, reference the same dictionary\. It will improve compression ratio of small files\. Typical gains range from 10% (at 64KB) to x5 better (at <1KB)\. +. .TP -.BR -- - All arguments after -- are treated as files - - -.SH DICTIONARY BUILDER -.PP -\fBzstd\fR offers \fIdictionary\fR compression, useful for very small files and messages. -It's possible to train \fBzstd\fR with some samples, the result of which is saved into a file called `dictionary`. -Then during compression and decompression, make reference to the same dictionary. -It will improve compression ratio of small files. -Typical gains range from ~10% (at 64KB) to x5 better (at <1KB). +\fB\-\-train FILEs\fR +use FILEs as training set to create a dictionary\. The training set should contain a lot of small files (> 100), and weight typically 100x the target dictionary size (for example, 10 MB for a 100 KB dictionary)\. +. .TP -.B \--train FILEs - use FILEs as training set to create a dictionary. The training set should contain a lot of small files (> 100), -and weight typically 100x the target dictionary size (for example, 10 MB for a 100 KB dictionary) +\fB\-o file\fR +dictionary saved into \fBfile\fR (default: dictionary) +. .TP -.B \-o file - dictionary saved into `file` (default: dictionary) +\fB\-\-maxdict #\fR +limit dictionary to specified size (default : (112640) +. .TP -.B \--maxdict # - limit dictionary to specified size (default : 112640) +\fB\-\-dictID #\fR +A dictionary ID is a locally unique ID that a decoder can use to verify it is using the right dictionary\. By default, zstd will create a 4\-bytes random number ID\. It\'s possible to give a precise number instead\. Short numbers have an advantage : an ID < 256 will only need 1 byte in the compressed frame header, and an ID < 65536 will only need 2 bytes\. This compares favorably to 4 bytes default\. However, it\'s up to the dictionary manager to not assign twice the same ID to 2 different dictionaries\. +. .TP -.B \--dictID # - A dictionary ID is a locally unique ID that a decoder can use to verify it is using the right dictionary. - By default, zstd will create a 4-bytes random number ID. - It's possible to give a precise number instead. - Short numbers have an advantage : an ID < 256 will only need 1 byte in the compressed frame header, - and an ID < 65536 will only need 2 bytes. This compares favorably to 4 bytes default. - However, it's up to the dictionary manager to not assign twice the same ID to 2 different dictionaries. +\fB\-s#\fR +dictionary selectivity level (default: 9) the smaller the value, the denser the dictionary, improving its efficiency but reducing its possible maximum size\. +. .TP -.B \-s# - dictionary selectivity level (default: 9) - the smaller the value, the denser the dictionary, improving its efficiency but reducing its possible maximum size. -.TP -.B \--cover=k=#,d=# - Use alternate dictionary builder algorithm named cover with parameters \fIk\fR and \fId\fR with \fId\fR <= \fIk\fR. - Selects segments of size \fIk\fR with the highest score to put in the dictionary. - The score of a segment is computed by the sum of the frequencies of all the subsegments of of size \fId\fR. - Generally \fId\fR should be in the range [6, 24]. - Good values for \fIk\fR vary widely based on the input data, but a safe range is [32, 2048]. - Example: \fB--train --cover=k=64,d=8 FILEs\fR. -.TP -.B \--optimize-cover[=steps=#,k=#,d=#] - If \fIsteps\fR is not specified, the default value of 32 is used. - If \fIk\fR is not specified, \fIsteps\fR values in [16, 2048] are checked for each value of \fId\fR. - If \fId\fR is not specified, the values checked are [6, 8, ..., 16]. - - Runs the cover dictionary builder for each parameter set saves the optimal parameters and dictionary. - Prints the optimal parameters and writes the optimal dictionary to the output file. - Supports multithreading if \fBzstd\fR is compiled with threading support. - - The parameter \fIk\fR is more sensitve than \fId\fR, and is faster to optimize over. - Suggested use is to run with a \fIsteps\fR <= 32 with neither \fIk\fR nor \fId\fR set. - Once it completes, use the value of \fId\fR it selects with a higher \fIsteps\fR (in the range [256, 1024]). - \fBzstd --train --optimize-cover FILEs - \fBzstd --train --optimize-cover=d=d,steps=512 FILEs -.TP - -.SH BENCHMARK -.TP -.B \-b# - benchmark file(s) using compression level # -.TP -.B \-e# - benchmark file(s) using multiple compression levels, from -b# to -e# (included). -.TP -.B \-i# - minimum evaluation time, in seconds (default : 3s), benchmark mode only -.TP -.B \-B# - cut file into independent blocks of size # (default: no block) -.B \--priority=rt - set process priority to real-time - -.SH ADVANCED COMPRESSION OPTIONS -.TP -.B \--zstd[=\fIoptions\fR] -.PD -\fBzstd\fR provides 22 predefined compression levels. The selected or default predefined compression level can be changed with advanced compression options. -The \fIoptions\fR are provided as a comma-separated list. You may specify only the \fIoptions\fR you want to change and the rest will be taken from the selected or default compression level. -The list of available \fIoptions\fR: -.RS - -.TP -.BI strategy= strat -.PD 0 -.TP -.BI strat= strat -.PD -Specify a strategy used by a match finder. -.IP "" -There are 8 strategies numbered from 0 to 7, from faster to stronger: -0=ZSTD_fast, 1=ZSTD_dfast, 2=ZSTD_greedy, 3=ZSTD_lazy, 4=ZSTD_lazy2, 5=ZSTD_btlazy2, 6=ZSTD_btopt, 7=ZSTD_btopt2. -.IP "" - -.TP -.BI windowLog= wlog -.PD 0 -.TP -.BI wlog= wlog -.PD -Specify the maximum number of bits for a match distance. -.IP "" -The higher number of bits increases the chance to find a match what usually improves compression ratio. -It also increases memory requirements for compressor and decompressor. -.IP "" -The minimum \fIwlog\fR is 10 (1 KiB) and the maximum is 25 (32 MiB) for 32-bit compilation and 27 (128 MiB) for 64-bit compilation. -.IP "" - -.TP -.BI hashLog= hlog -.PD 0 -.TP -.BI hlog= hlog -.PD -Specify the maximum number of bits for a hash table. -.IP "" -The bigger hash table causes less collisions what usually make compression faster but requires more memory during compression. -.IP "" -The minimum \fIhlog\fR is 6 (64 B) and the maximum is 25 (32 MiB) for 32-bit compilation and 27 (128 MiB) for 64-bit compilation. - -.TP -.BI chainLog= clog -.PD 0 -.TP -.BI clog= clog -.PD -Specify the maximum number of bits for a hash chain or a binary tree. -.IP "" -The higher number of bits increases the chance to find a match what usually improves compression ratio. -It also slows down compression speed and increases memory requirements for compression. -This option is ignored for the ZSTD_fast strategy. -.IP "" -The minimum \fIclog\fR is 6 (64 B) and the maximum is 26 (64 MiB) for 32-bit compilation and 28 (256 MiB) for 64-bit compilation. -.IP "" - -.TP -.BI searchLog= slog -.PD 0 -.TP -.BI slog= slog -.PD -Specify the maximum number of searches in a hash chain or a binary tree using logarithmic scale. -.IP "" -The bigger number of searches increases the chance to find a match what usually improves compression ratio but decreases compression speed. -.IP "" -The minimum \fIslog\fR is 1 and the maximum is 24 for 32-bit compilation and 26 for 64-bit compilation. -.IP "" - -.TP -.BI searchLength= slen -.PD 0 -.TP -.BI slen= slen -.PD -Specify the minimum searched length of a match in a hash table. -.IP "" -The bigger search length usually decreases compression ratio but improves decompression speed. -.IP "" -The minimum \fIslen\fR is 3 and the maximum is 7. -.IP "" - -.TP -.BI targetLength= tlen -.PD 0 -.TP -.BI tlen= tlen -.PD -Specify the minimum match length that causes a match finder to interrupt searching of better matches. -.IP "" -The bigger minimum match length usually improves compression ratio but decreases compression speed. -This option is used only with ZSTD_btopt and ZSTD_btopt2 strategies. -.IP "" -The minimum \fItlen\fR is 4 and the maximum is 999. -.IP "" - -.PP -.B An example +\fB\-\-cover=k#,d=#\fR +Use alternate dictionary builder algorithm named cover with parameters \fIk\fR and \fId\fR with \fId\fR <= \fIk\fR\. Selects segments of size \fIk\fR with the highest score to put in the dictionary\. The score of a segment is computed by the sum of the frequencies of all the subsegments of of size \fId\fR\. Generally \fId\fR should be in the range [6, 24]\. Good values for \fIk\fR vary widely based on the input data, but a safe range is [32, 2048]\. +. .br -The following parameters sets advanced compression options to predefined level 19 for files bigger than 256 KB: -.IP "" -\fB--zstd=\fRwindowLog=23,chainLog=23,hashLog=22,searchLog=6,searchLength=3,targetLength=48,strategy=6 - -.SH BUGS -Report bugs at:- https://github.com/facebook/zstd/issues - -.SH AUTHOR +Example: \fB\-\-train \-\-cover=k=64,d=8 FILEs\fR\. +. +.TP +\fB\-\-optimize\-cover[=steps=#,k=#,d=#]\fR +If \fIsteps\fR is not specified, the default value of 32 is used\. If \fIk\fR is not specified, the \fIk\fR values in [16, 2048] are checked for each value of \fId\fR\. If \fId\fR is not specified, the values checked are [6, 8, \.\.\., 16]\. +. +.IP +Runs the cover dictionary builder for each parameter set and saves the optimal parameters and dictionary\. Prints the optimal parameters and writes the optimal dictionary to the output file\. Supports multithreading if \fBzstd\fR is compiled with threading support\. +. +.IP +The parameter \fIk\fR is more sensitive than \fId\fR, and is faster to optimize over\. Suggested use is to run with a \fIsteps\fR <= 32 with neither \fIk\fR nor \fId\fR set\. Once it completes, use the value of \fId\fR it selects with a higher \fIsteps\fR (in the range [256, 1024])\. +. +.IP +\fBzstd \-\-train \-\-optimize\-cover FILEs\fR +. +.br +\fBzstd \-\-train \-\-optimize\-cover=d=d,steps=512 FILEs\fR +. +.SH "BENCHMARK" +. +.TP +\fB\-b#\fR +benchmark file(s) using compression level # +. +.TP +\fB\-e#\fR +benchmark file(s) using multiple compression levels, from \fB\-b#\fR to \fB\-e#\fR (inclusive) +. +.TP +\fB\-i#\fR +minimum evaluation time, in seconds (default : 3s), benchmark mode only +. +.TP +\fB\-B#\fR +cut file into independent blocks of size # (default: no block) +. +.TP +\fB\-\-priority=rt\fR +set process priority to real\-time +. +.SH "ADVANCED COMPRESSION OPTIONS" +. +.SS "\-\-zstd[=options]:" +\fBzstd\fR provides 22 predefined compression levels\. The selected or default predefined compression level can be changed with advanced compression options\. The \fIoptions\fR are provided as a comma\-separated list\. You may specify only the options you want to change and the rest will be taken from the selected or default compression level\. The list of available \fIoptions\fR: +. +.TP +\fBstrategy\fR=\fIstrat\fR, \fBstrat\fR=\fIstrat\fR +Specify a strategy used by a match finder\. +. +.IP +There are 8 strategies numbered from 0 to 7, from faster to stronger: 0=ZSTD_fast, 1=ZSTD_dfast, 2=ZSTD_greedy, 3=ZSTD_lazy, 4=ZSTD_lazy2, 5=ZSTD_btlazy2, 6=ZSTD_btopt, 7=ZSTD_btopt2\. +. +.TP +\fBwindowLog\fR=\fIwlog\fR, \fBwlog\fR=\fIwlog\fR +Specify the maximum number of bits for a match distance\. +. +.IP +The higher number of increases the chance to find a match which usually improves compression ratio\. It also increases memory requirements for the compressor and decompressor\. The minimum \fIwlog\fR is 10 (1 KiB) and the maximum is 27 (128 MiB)\. +. +.TP +\fBhashLog\fR=\fIhlog\fR, \fBhlog\fR=\fIhlog\fR +Specify the maximum number of bits for a hash table\. +. +.IP +Bigger hash tables cause less collisions which usually makes compression faster, but requires more memory during compression\. +. +.IP +The minimum \fIhlog\fR is 6 (64 B) and the maximum is 26 (128 MiB)\. +. +.TP +\fBchainLog\fR=\fIclog\fR, \fBclog\fR=\fIclog\fR +Specify the maximum number of bits for a hash chain or a binary tree\. +. +.IP +Higher numbers of bits increases the chance to find a match which usually improves compression ratio\. It also slows down compression speed and increases memory requirements for compression\. This option is ignored for the ZSTD_fast strategy\. +. +.IP +The minimum \fIclog\fR is 6 (64 B) and the maximum is 28 (256 MiB)\. +. +.TP +\fBsearchLog\fR=\fIslog\fR, \fBslog\fR=\fIslog\fR +Specify the maximum number of searches in a hash chain or a binary tree using logarithmic scale\. +. +.IP +More searches increases the chance to find a match which usually increases compression ratio but decreases compression speed\. +. +.IP +The minimum \fIslog\fR is 1 and the maximum is 26\. +. +.TP +\fBsearchLength\fR=\fIslen\fR, \fBslen\fR=\fIslen\fR +Specify the minimum searched length of a match in a hash table\. +. +.IP +Larger search lengths usually decrease compression ratio but improve decompression speed\. +. +.IP +The minimum \fIslen\fR is 3 and the maximum is 7\. +. +.TP +\fBtargetLen\fR=\fItlen\fR, \fBtlen\fR=\fItlen\fR +Specify the minimum match length that causes a match finder to stop searching for better matches\. +. +.IP +A larger minimum match length usually improves compression ratio but decreases compression speed\. This option is only used with strategies ZSTD_btopt and ZSTD_btopt2\. +. +.IP +The minimum \fItlen\fR is 4 and the maximum is 999\. +. +.SS "Example" +The following parameters sets advanced compression options to those of predefined level 19 for files bigger than 256 KB: +. +.P +\fB\-\-zstd\fR=windowLog=23,chainLog=23,hashLog=22,searchLog=6,searchLength=3,targetLength=48,strategy=6 +. +.SH "BUGS" +Report bugs at: https://github\.com/facebook/zstd/issues +. +.SH "AUTHOR" Yann Collet diff --git a/programs/zstd.1.md b/programs/zstd.1.md new file mode 100644 index 000000000..d170dcc8c --- /dev/null +++ b/programs/zstd.1.md @@ -0,0 +1,299 @@ +zstd(1) -- zstd, unzstd, zstdcat - Compress or decompress .zst files +==================================================================== + +SYNOPSIS +-------- + +`zstd` [*OPTIONS*] [-|<INPUT-FILE>] [-o <OUTPUT-FILE>] + +`unzstd` is equivalent to `zstd -d` +`zstdcat` is equivalent to `zstd -dcf` + +DESCRIPTION +----------- +`zstd` is a fast lossless compression algorithm and data compression tool, +with command line syntax similar to `gzip (1)` and `xz (1)`. +It is based on the **LZ77** family, with further FSE & huff0 entropy stages. +`zstd` offers highly configurable compression speed, +with fast modes at > 200 MB/s per code, +and strong modes nearing lzma compression ratios. +It also features a very fast decoder, with speeds > 500 MB/s per core. + +`zstd` command line syntax is generally similar to gzip, +but features the following differences : + + - Source files are preserved by default. + It's possible to remove them automatically by using the `--rm` command. + - When compressing a single file, `zstd` displays progress notifications + and result summary by default. + Use `-q` to turn them off. + +`zstd` compresses or decompresses each _file_ according to the selected +operation mode. +If no _files_ are given or _file_ is `-`, `zstd` reads from standard input +and writes the processed data to standard output. +`zstd` will refuse to (display an error and skip the _file_) to write +compressed data to standard output if it is a terminal. +Similarly, `zstd` will refuse to read compressed data from standard input if it +is a terminal. + +Unless `--stdout` or `-o` is specified, _files_ are written to a new file +whose name is derived from the source _file_ name: + +* When compressing, the suffix `.zst` is appended to the source filename to + get the target filename. +* When decompressing, the `.zst` suffix is removed from the source filename to + get the target filename + +### Concatenation with .zst files +It is possible to concatenate `.zst` files as is. +`zstd` will decompress such files as if they were a single `.zst` file. + +OPTIONS +------- + +### Integer suffixes and special values +In most places where an integer argument is expected, +an optional suffix is supported to easily indicate large integers. +There must be no space between the integer and the suffix. + +* `KiB`: + Multiply the integer by 1,024 (2\^10). + `Ki`, `K`, and `KB` are accepted as synonyms for `KiB`. +* `MiB`: + Multiply the integer by 1,048,576 (2\^20). + `Mi`, `M`, and `MB` are accepted as synonyms for `MiB`. + +### Operation mode +If multiple operation mode options are given, +the last one takes effect. + +* `-z`, `--compress`: + Compress. + This is the default operation mode when no operation mode option is specified + and no other operation mode is implied from the command name + (for example, `unzstd` implies `--decompress`). +* `-d`, `--decompress`, `--uncompress`: + Decompress. +* `-t`, `--test`: + Test the integrity of compressed _files_. + This option is equivalent to `--decompress --stdout` except that the + decompressed data is discarded instead of being written to standard output. + No files are created or removed. +* `-b#`: + Benchmark file(s) using compression level # +* `--train FILEs`: + Use FILEs as a training set to create a dictionary. + The training set should contain a lot of small files (> 100). + +### Operation modifiers + +* `-#`: + `#` compression level \[1-19] (default: 3) +* `--ultra`: + unlocks high compression levels 20+ (maximum 22), using a lot more memory. + Note that decompression will also require more memory when using these levels. +* `-D file`: + use `file` as Dictionary to compress or decompress FILE(s) +* `--nodictID`: + do not store dictionary ID within frame header (dictionary compression). + The decoder will have to rely on implicit knowledge about which dictionary to use, + it won't be able to check if it's correct. +* `-o file`: + save result into `file` (only possible with a single INPUT-FILE) +* `-f`, `--force`: + overwrite output without prompting +* `-c`, `--stdout`: + force write to standard output, even if it is the console +* `--[no-]sparse`: + enable / disable sparse FS support, + to make files with many zeroes smaller on disk. + Creating sparse files may save disk space and speed up decompression by + reducing the amount of disk I/O. + default : enabled when output is into a file, + and disabled when output is stdout. + This setting overrides default and can force sparse mode over stdout. +* `--rm`: + remove source file(s) after successful compression or decompression +* `-k`, `--keep`: + keep source file(s) after successful compression or decompression. + This is the default behaviour. +* `-r`: + operate recursively on dictionaries +* `-h`/`-H`, `--help`: + display help/long help and exit +* `-V`, `--version`: + display version number and exit +* `-v`: + verbose mode +* `-q`, `--quiet`: + suppress warnings, interactivity, and notifications. + specify twice to suppress errors too. +* `-C`, `--[no-]check`: + add integrety check computed from uncompressed data (default : enabled) +* `--`: + All arguments after `--` are treated as files + +DICTIONARY BUILDER +------------------ +`zstd` offers _dictionary_ compression, +useful for very small files and messages. +It's possible to train `zstd` with some samples, +the result of which is saved into a file called a `dictionary`. +Then during compression and decompression, reference the same dictionary. +It will improve compression ratio of small files. +Typical gains range from 10% (at 64KB) to x5 better (at <1KB). + +* `--train FILEs`: + use FILEs as training set to create a dictionary. + The training set should contain a lot of small files (> 100), + and weight typically 100x the target dictionary size + (for example, 10 MB for a 100 KB dictionary). +* `-o file`: + dictionary saved into `file` (default: dictionary) +* `--maxdict #`: + limit dictionary to specified size (default : (112640) +* `--dictID #`: + A dictionary ID is a locally unique ID that a decoder can use to verify it is + using the right dictionary. + By default, zstd will create a 4-bytes random number ID. + It's possible to give a precise number instead. + Short numbers have an advantage : an ID < 256 will only need 1 byte in the + compressed frame header, and an ID < 65536 will only need 2 bytes. + This compares favorably to 4 bytes default. + However, it's up to the dictionary manager to not assign twice the same ID to + 2 different dictionaries. +* `-s#`: + dictionary selectivity level (default: 9) + the smaller the value, the denser the dictionary, + improving its efficiency but reducing its possible maximum size. +* `--cover=k#,d=#`: + Use alternate dictionary builder algorithm named cover with parameters + _k_ and _d_ with _d_ <= _k_. + Selects segments of size _k_ with the highest score to put in the dictionary. + The score of a segment is computed by the sum of the frequencies of all the + subsegments of of size _d_. + Generally _d_ should be in the range [6, 24]. + Good values for _k_ vary widely based on the input data, + but a safe range is [32, 2048].
+ Example: `--train --cover=k=64,d=8 FILEs`. + +* `--optimize-cover[=steps=#,k=#,d=#]`: + If _steps_ is not specified, the default value of 32 is used. + If _k_ is not specified, the _k_ values in [16, 2048] are checked for each + value of _d_. + If _d_ is not specified, the values checked are [6, 8, ..., 16]. + + Runs the cover dictionary builder for each parameter set and saves the + optimal parameters and dictionary. + Prints the optimal parameters and writes the optimal dictionary to the output file. + Supports multithreading if `zstd` is compiled with threading support. + + The parameter _k_ is more sensitive than _d_, and is faster to optimize over. + Suggested use is to run with a _steps_ <= 32 with neither _k_ nor _d_ set. + Once it completes, use the value of _d_ it selects with a higher _steps_ + (in the range [256, 1024]). + + `zstd --train --optimize-cover FILEs`
+ `zstd --train --optimize-cover=d=d,steps=512 FILEs` + + +BENCHMARK +--------- + +* `-b#`: + benchmark file(s) using compression level # +* `-e#`: + benchmark file(s) using multiple compression levels, from `-b#` to `-e#` (inclusive) +* `-i#`: + minimum evaluation time, in seconds (default : 3s), benchmark mode only +* `-B#`: + cut file into independent blocks of size # (default: no block) +* `--priority=rt`: + set process priority to real-time + +ADVANCED COMPRESSION OPTIONS +---------------------------- +### --zstd[=options]: +`zstd` provides 22 predefined compression levels. +The selected or default predefined compression level can be changed with +advanced compression options. +The _options_ are provided as a comma-separated list. +You may specify only the options you want to change and the rest will be +taken from the selected or default compression level. +The list of available _options_: + +- `strategy`=_strat_, `strat`=_strat_: + Specify a strategy used by a match finder. + + There are 8 strategies numbered from 0 to 7, from faster to stronger: + 0=ZSTD\_fast, 1=ZSTD\_dfast, 2=ZSTD\_greedy, 3=ZSTD\_lazy, + 4=ZSTD\_lazy2, 5=ZSTD\_btlazy2, 6=ZSTD\_btopt, 7=ZSTD\_btopt2. + +- `windowLog`=_wlog_, `wlog`=_wlog_: + Specify the maximum number of bits for a match distance. + + The higher number of increases the chance to find a match which usually + improves compression ratio. + It also increases memory requirements for the compressor and decompressor. + The minimum _wlog_ is 10 (1 KiB) and the maximum is 27 (128 MiB). + +- `hashLog`=_hlog_, `hlog`=_hlog_: + Specify the maximum number of bits for a hash table. + + Bigger hash tables cause less collisions which usually makes compression + faster, but requires more memory during compression. + + The minimum _hlog_ is 6 (64 B) and the maximum is 26 (128 MiB). + +- `chainLog`=_clog_, `clog`=_clog_: + Specify the maximum number of bits for a hash chain or a binary tree. + + Higher numbers of bits increases the chance to find a match which usually + improves compression ratio. + It also slows down compression speed and increases memory requirements for + compression. + This option is ignored for the ZSTD_fast strategy. + + The minimum _clog_ is 6 (64 B) and the maximum is 28 (256 MiB). + +- `searchLog`=_slog_, `slog`=_slog_: + Specify the maximum number of searches in a hash chain or a binary tree + using logarithmic scale. + + More searches increases the chance to find a match which usually increases + compression ratio but decreases compression speed. + + The minimum _slog_ is 1 and the maximum is 26. + +- `searchLength`=_slen_, `slen`=_slen_: + Specify the minimum searched length of a match in a hash table. + + Larger search lengths usually decrease compression ratio but improve + decompression speed. + + The minimum _slen_ is 3 and the maximum is 7. + +- `targetLen`=_tlen_, `tlen`=_tlen_: + Specify the minimum match length that causes a match finder to stop + searching for better matches. + + A larger minimum match length usually improves compression ratio but + decreases compression speed. + This option is only used with strategies ZSTD_btopt and ZSTD_btopt2. + + The minimum _tlen_ is 4 and the maximum is 999. + +### Example +The following parameters sets advanced compression options to those of +predefined level 19 for files bigger than 256 KB: + +`--zstd`=windowLog=23,chainLog=23,hashLog=22,searchLog=6,searchLength=3,targetLength=48,strategy=6 + +BUGS +---- +Report bugs at: https://github.com/facebook/zstd/issues + +AUTHOR +------ +Yann Collet From b8e52d3c83eeeeec088836b5840ee1ca307a612d Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 16 Mar 2017 16:06:03 -0700 Subject: [PATCH 016/305] Fix zstd not erroring on compressing to terminal without forceStdout --- programs/zstdcli.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/programs/zstdcli.c b/programs/zstdcli.c index ae49da7b1..fc7ca9e36 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -635,7 +635,7 @@ int main(int argCount, const char* argv[]) /* Check if input/output defined as console; trigger an error in this case */ if (!strcmp(filenameTable[0], stdinmark) && IS_CONSOLE(stdin) ) CLEAN_RETURN(badusage(programName)); - if (outFileName && !strcmp(outFileName, stdoutmark) && IS_CONSOLE(stdout) && strcmp(filenameTable[0], stdinmark) && !(forceStdout && (operation==zom_decompress))) + if (outFileName && !strcmp(outFileName, stdoutmark) && IS_CONSOLE(stdout) && !strcmp(filenameTable[0], stdinmark) && !forceStdout && operation!=zom_decompress) CLEAN_RETURN(badusage(programName)); /* user-selected output filename, only possible with a single file */ From d973071e907ba2b9f99db6ea14c625a1307c1ac9 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 16 Mar 2017 16:25:19 -0700 Subject: [PATCH 017/305] Add tests for compressed data on console --- tests/playTests.sh | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/tests/playTests.sh b/tests/playTests.sh index c493fed55..c584fe560 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -72,6 +72,10 @@ cp tmp tmp2 $ZSTD tmp2 -fo && die "-o must be followed by filename " $ECHO "test : implied stdout when input is stdin" $ECHO bob | $ZSTD | $ZSTD -d +$ECHO "test : compressed data to terminal" +$ECHO bob | $ZSTD && die "should have refused : compressed data to terminal" +$ECHO "test : compressed data from terminal (a hang here is a test fail, zstd is wrongly waiting on data from terminal)" +$ZSTD -d > $INTOVOID && die "should have refused : compressed data from terminal" $ECHO "test : null-length file roundtrip" $ECHO -n '' | $ZSTD - --stdout | $ZSTD -d --stdout $ECHO "test : decompress file with wrong suffix (must fail)" From 0a189b63fee91518796bc225eb249dd4e927fcf1 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 16 Mar 2017 16:33:53 -0700 Subject: [PATCH 018/305] fix minor details in man page --- programs/zstd.1 | 13 +++++++++++-- programs/zstd.1.md | 22 ++++++++++++++-------- programs/zstdcli.c | 2 +- 3 files changed, 26 insertions(+), 11 deletions(-) diff --git a/programs/zstd.1 b/programs/zstd.1 index 9ac4ca0a8..f79b3ce16 100644 --- a/programs/zstd.1 +++ b/programs/zstd.1 @@ -8,7 +8,10 @@ \fBzstd\fR [\fIOPTIONS\fR] [\-|] [\-o ] . .P -\fBunzstd\fR is equivalent to \fBzstd \-d\fR \fBzstdcat\fR is equivalent to \fBzstd \-dcf\fR +\fBunzstd\fR is equivalent to \fBzstd \-d\fR +. +.P +\fBzstdcat\fR is equivalent to \fBzstd \-dcf\fR . .SH "DESCRIPTION" \fBzstd\fR is a fast lossless compression algorithm and data compression tool, with command line syntax similar to \fBgzip (1)\fR and \fBxz (1)\fR\. It is based on the \fBLZ77\fR family, with further FSE & huff0 entropy stages\. \fBzstd\fR offers highly configurable compression speed, with fast modes at > 200 MB/s per code, and strong modes nearing lzma compression ratios\. It also features a very fast decoder, with speeds > 500 MB/s per core\. @@ -22,10 +25,16 @@ Source files are preserved by default\. It\'s possible to remove them automatica .IP "\(bu" 4 When compressing a single file, \fBzstd\fR displays progress notifications and result summary by default\. Use \fB\-q\fR to turn them off\. . +.IP "\(bu" 4 +\fBzstd\fR does not accept input from console, but it properly accepts \fBstdin\fR when it\'s not the console\. +. +.IP "\(bu" 4 +\fBzstd\fR displays a short help page when command line is an error\. Use \fB\-q\fR to turn it off\. +. .IP "" 0 . .P -\fBzstd\fR compresses or decompresses each \fIfile\fR according to the selected operation mode\. If no \fIfiles\fR are given or \fIfile\fR is \fB\-\fR, \fBzstd\fR reads from standard input and writes the processed data to standard output\. \fBzstd\fR will refuse to (display an error and skip the \fIfile\fR) to write compressed data to standard output if it is a terminal\. Similarly, \fBzstd\fR will refuse to read compressed data from standard input if it is a terminal\. +\fBzstd\fR compresses or decompresses each \fIfile\fR according to the selected operation mode\. If no \fIfiles\fR are given or \fIfile\fR is \fB\-\fR, \fBzstd\fR reads from standard input and writes the processed data to standard output\. \fBzstd\fR will refuse to write compressed data to standard output if it is a terminal : it will display an error message and skip the \fIfile\fR\. Similarly, \fBzstd\fR will refuse to read compressed data from standard input if it is a terminal\. . .P Unless \fB\-\-stdout\fR or \fB\-o\fR is specified, \fIfiles\fR are written to a new file whose name is derived from the source \fIfile\fR name: diff --git a/programs/zstd.1.md b/programs/zstd.1.md index d170dcc8c..bb5b22664 100644 --- a/programs/zstd.1.md +++ b/programs/zstd.1.md @@ -7,8 +7,10 @@ SYNOPSIS `zstd` [*OPTIONS*] [-|<INPUT-FILE>] [-o <OUTPUT-FILE>] `unzstd` is equivalent to `zstd -d` + `zstdcat` is equivalent to `zstd -dcf` + DESCRIPTION ----------- `zstd` is a fast lossless compression algorithm and data compression tool, @@ -27,15 +29,19 @@ but features the following differences : - When compressing a single file, `zstd` displays progress notifications and result summary by default. Use `-q` to turn them off. + - `zstd` does not accept input from console, + but it properly accepts `stdin` when it's not the console. + - `zstd` displays a short help page when command line is an error. + Use `-q` to turn it off. `zstd` compresses or decompresses each _file_ according to the selected operation mode. If no _files_ are given or _file_ is `-`, `zstd` reads from standard input and writes the processed data to standard output. -`zstd` will refuse to (display an error and skip the _file_) to write -compressed data to standard output if it is a terminal. -Similarly, `zstd` will refuse to read compressed data from standard input if it -is a terminal. +`zstd` will refuse to write compressed data to standard output +if it is a terminal : it will display an error message and skip the _file_. +Similarly, `zstd` will refuse to read compressed data from standard input +if it is a terminal. Unless `--stdout` or `-o` is specified, _files_ are written to a new file whose name is derived from the source _file_ name: @@ -162,8 +168,8 @@ Typical gains range from 10% (at 64KB) to x5 better (at <1KB). compressed frame header, and an ID < 65536 will only need 2 bytes. This compares favorably to 4 bytes default. However, it's up to the dictionary manager to not assign twice the same ID to - 2 different dictionaries. -* `-s#`: + 2 different dictionaries. +* `-s#`: dictionary selectivity level (default: 9) the smaller the value, the denser the dictionary, improving its efficiency but reducing its possible maximum size. @@ -194,7 +200,7 @@ Typical gains range from 10% (at 64KB) to x5 better (at <1KB). Once it completes, use the value of _d_ it selects with a higher _steps_ (in the range [256, 1024]). - `zstd --train --optimize-cover FILEs`
+ `zstd --train --optimize-cover FILEs`
`zstd --train --optimize-cover=d=d,steps=512 FILEs` @@ -226,7 +232,7 @@ The list of available _options_: - `strategy`=_strat_, `strat`=_strat_: Specify a strategy used by a match finder. - There are 8 strategies numbered from 0 to 7, from faster to stronger: + There are 8 strategies numbered from 0 to 7, from faster to stronger: 0=ZSTD\_fast, 1=ZSTD\_dfast, 2=ZSTD\_greedy, 3=ZSTD\_lazy, 4=ZSTD\_lazy2, 5=ZSTD\_btlazy2, 6=ZSTD\_btopt, 7=ZSTD\_btopt2. diff --git a/programs/zstdcli.c b/programs/zstdcli.c index ae49da7b1..50ff21d53 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -167,7 +167,7 @@ static int usage_advanced(const char* programName) static int badusage(const char* programName) { DISPLAYLEVEL(1, "Incorrect parameters\n"); - if (displayLevel >= 1) usage(programName); + if (displayLevel >= 2) usage(programName); return 1; } From 9a38dfa7948c8ea80b07a2f3af152d02b481c4b9 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Fri, 17 Mar 2017 12:32:18 -0700 Subject: [PATCH 019/305] Only run IS_CONSOLE tests with a TTY --- tests/playTests.sh | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/tests/playTests.sh b/tests/playTests.sh index c584fe560..38a66d327 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -20,6 +20,12 @@ roundTripTest() { $DIFF -q tmp1 tmp2 } +isTerminal=false +if [ -t 0 ] && [ -t 1 ] +then + isTerminal=true +fi + isWindows=false ECHO="echo" INTOVOID="/dev/null" @@ -27,6 +33,7 @@ case "$OS" in Windows*) isWindows=true ECHO="echo -e" + INTOVOID="NUL" ;; esac @@ -72,10 +79,12 @@ cp tmp tmp2 $ZSTD tmp2 -fo && die "-o must be followed by filename " $ECHO "test : implied stdout when input is stdin" $ECHO bob | $ZSTD | $ZSTD -d +if [ "$isTerminal" = true ]; then $ECHO "test : compressed data to terminal" $ECHO bob | $ZSTD && die "should have refused : compressed data to terminal" $ECHO "test : compressed data from terminal (a hang here is a test fail, zstd is wrongly waiting on data from terminal)" $ZSTD -d > $INTOVOID && die "should have refused : compressed data from terminal" +fi $ECHO "test : null-length file roundtrip" $ECHO -n '' | $ZSTD - --stdout | $ZSTD -d --stdout $ECHO "test : decompress file with wrong suffix (must fail)" @@ -244,7 +253,7 @@ $ZSTD -f tmp -D tmpDict1 --no-dictID $ZSTD -d tmp.zst -D tmpDict -fo result $DIFF $TESTFILE result $ECHO "- Compress with wrong argument order (must fail)" -$ZSTD tmp -Df tmpDict1 -c > /dev/null && die "-D must be followed by dictionary name " +$ZSTD tmp -Df tmpDict1 -c > $INTOVOID && die "-D must be followed by dictionary name " $ECHO "- Compress multiple files with dictionary" rm -rf dirTestDict mkdir dirTestDict From e5c4f0403397e32a63baadd701816d5720c0d2a1 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 17 Mar 2017 14:25:57 -0700 Subject: [PATCH 020/305] updated compression graph --- README.md | 11 ++++++++--- doc/images/Cspeed4.png | Bin 35361 -> 71276 bytes doc/images/Dspeed4.png | Bin 8984 -> 29425 bytes 3 files changed, 8 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index 6de5a1079..f1d8eda1b 100644 --- a/README.md +++ b/README.md @@ -14,11 +14,11 @@ you can consult a list of known ports on [Zstandard homepage](http://www.zstd.ne As a reference, several fast compression algorithms were tested and compared on a server running Linux Mint Debian Edition (`Linux version 4.8.0-1-amd64`), with a Core i7-6700K CPU @ 4.0GHz, -using [lzbench v1.6], an open-source in-memory benchmark by @inikep +using [lzbench], an open-source in-memory benchmark by @inikep compiled with GCC 6.3.0, on the [Silesia compression corpus]. -[lzbench v1.6]: https://github.com/inikep/lzbench +[lzbench]: https://github.com/inikep/lzbench [Silesia compression corpus]: http://sun.aei.polsl.pl/~sdeor/index.php?page=silesia | Compressor name | Ratio | Compression| Decompress.| @@ -38,7 +38,12 @@ on the [Silesia compression corpus]. Zstd can also offer stronger compression ratios at the cost of compression speed. Speed vs Compression trade-off is configurable by small increments. Decompression speed is preserved and remains roughly the same at all settings, a property shared by most LZ compression algorithms, such as [zlib] or lzma. -The following tests were run on a Core i7-3930K CPU @ 4.5GHz, using [lzbench], an open-source in-memory benchmark by @inikep compiled with GCC 5.2.1, on the [Silesia compression corpus]. +The following tests were run +on a server running Linux Mint Debian Edition (`Linux version 4.8.0-1-amd64`) +with a Core i7-6700K CPU @ 4.0GHz, +using [lzbench], an open-source in-memory benchmark by @inikep +compiled with GCC 6.3.0, +on the [Silesia compression corpus]. Compression Speed vs Ratio | Decompression Speed ---------------------------|-------------------- diff --git a/doc/images/Cspeed4.png b/doc/images/Cspeed4.png index f0ca0ffba9c4413cc2a8a19dd5768e03e236c095..318204c00e96311529ed76351481db81d8212770 100644 GIT binary patch literal 71276 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�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 Sv^=p6ess^_w2Cxs!~YMALH64K literal 35361 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 KryRX}>;C{7W5ebE diff --git a/doc/images/Dspeed4.png b/doc/images/Dspeed4.png index eba485d0d1ed779b61c1dd1698049ad6628a8a5e..a0c7f65c25312279b4b25294cbcdb6ecfa51ae4e 100644 GIT binary patch literal 29425 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 w>=O?O?_zidM(_WRU$OtcAGmD}WHL2Zn Date: Fri, 17 Mar 2017 15:33:09 -0700 Subject: [PATCH 021/305] fixed typo in server type --- README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index f1d8eda1b..7caee5fd3 100644 --- a/README.md +++ b/README.md @@ -12,7 +12,7 @@ you can consult a list of known ports on [Zstandard homepage](http://www.zstd.ne |dev | [![Build Status](https://travis-ci.org/facebook/zstd.svg?branch=dev)](https://travis-ci.org/facebook/zstd) | As a reference, several fast compression algorithms were tested and compared -on a server running Linux Mint Debian Edition (`Linux version 4.8.0-1-amd64`), +on a server running Linux Debian (`Linux version 4.8.0-1-amd64`), with a Core i7-6700K CPU @ 4.0GHz, using [lzbench], an open-source in-memory benchmark by @inikep compiled with GCC 6.3.0, @@ -39,7 +39,7 @@ Zstd can also offer stronger compression ratios at the cost of compression speed Speed vs Compression trade-off is configurable by small increments. Decompression speed is preserved and remains roughly the same at all settings, a property shared by most LZ compression algorithms, such as [zlib] or lzma. The following tests were run -on a server running Linux Mint Debian Edition (`Linux version 4.8.0-1-amd64`) +on a server running Linux Debian (`Linux version 4.8.0-1-amd64`) with a Core i7-6700K CPU @ 4.0GHz, using [lzbench], an open-source in-memory benchmark by @inikep compiled with GCC 6.3.0, From 8086d623ca8c2d4fe4bd63e853e088d0ff7ccfa1 Mon Sep 17 00:00:00 2001 From: Przemyslaw Skibinski Date: Sat, 18 Mar 2017 11:19:09 +0100 Subject: [PATCH 022/305] updated build of Windows packages --- lib/dll/example/README.md | 10 +++++----- lib/dll/example/build_package.bat | 1 + 2 files changed, 6 insertions(+), 5 deletions(-) diff --git a/lib/dll/example/README.md b/lib/dll/example/README.md index 957a29f35..e231f59c5 100644 --- a/lib/dll/example/README.md +++ b/lib/dll/example/README.md @@ -4,11 +4,11 @@ ZSTD Windows binary package #### The package contents - `zstd.exe` : Command Line Utility, supporting gzip-like arguments -- `dll\libzstd.dll` : The DLL of ZSTD library -- `dll\libzstd.lib` : The import library of ZSTD library for Visual C++ -- `example\` : The example of usage of ZSTD library -- `include\` : Header files required with ZSTD library -- `static\libzstd_static.lib` : The static ZSTD library +- `dll\libzstd.dll` : The ZSTD dynamic library (DLL) +- `dll\libzstd.lib` : The import library of the ZSTD dynamic library (DLL) for Visual C++ +- `example\` : The example of usage of the ZSTD library +- `include\` : Header files required by the ZSTD library +- `static\libzstd_static.lib` : The static ZSTD library (LIB) #### Usage of Command Line Interface diff --git a/lib/dll/example/build_package.bat b/lib/dll/example/build_package.bat index b225af8d8..cae0a15cb 100644 --- a/lib/dll/example/build_package.bat +++ b/lib/dll/example/build_package.bat @@ -9,6 +9,7 @@ COPY lib\common\mem.h bin\example\ COPY lib\common\zstd_errors.h bin\example\ COPY lib\common\zstd_internal.h bin\example\ COPY lib\common\error_private.h bin\example\ +COPY lib\common\xxhash.h bin\example\ COPY lib\zstd.h bin\include\ COPY lib\libzstd.a bin\static\libzstd_static.lib COPY lib\dll\libzstd.* bin\dll\ From 7ebf2de02d5b919c68e5e8b68fc98ab6e13c2996 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 20 Mar 2017 11:25:00 -0700 Subject: [PATCH 023/305] Add ability to strongly limit fuzzer test size with flag --- tests/fuzzer.c | 21 +++++++++++++++++---- tests/zstreamtest.c | 22 ++++++++++++++++------ 2 files changed, 33 insertions(+), 10 deletions(-) diff --git a/tests/fuzzer.c b/tests/fuzzer.c index def7542b5..1d8be17af 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -616,7 +616,7 @@ static size_t FUZ_randomLength(U32* seed, U32 maxLog) #define CHECK(cond, ...) if (cond) { DISPLAY("Error => "); DISPLAY(__VA_ARGS__); \ DISPLAY(" (seed %u, test nb %u) \n", seed, testNb); goto _output_error; } -static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxDurationS, double compressibility) +static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxDurationS, double compressibility, int bigTests) { static const U32 maxSrcLog = 23; static const U32 maxSampleLog = 22; @@ -636,6 +636,7 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxD U32 coreSeed = seed, lseed = 0; clock_t const startClock = clock(); clock_t const maxClockSpan = maxDurationS * CLOCKS_PER_SEC; + int const cLevelLimiter = bigTests ? 3 : 2; /* allocation */ cNoiseBuffer[0] = (BYTE*)malloc (srcBufferSize); @@ -701,7 +702,12 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxD crcOrig = XXH64(sampleBuffer, sampleSize, 0); /* compression tests */ - { unsigned const cLevel = (FUZ_rand(&lseed) % (ZSTD_maxCLevel() - (FUZ_highbit32((U32)sampleSize)/3))) + 1; + { + unsigned const cLevel = + (FUZ_rand(&lseed) % + (ZSTD_maxCLevel() - + (FUZ_highbit32((U32)sampleSize) / cLevelLimiter))) + + 1; cSize = ZSTD_compressCCtx(ctx, cBuffer, cBufferSize, sampleBuffer, sampleSize, cLevel); CHECK(ZSTD_isError(cSize), "ZSTD_compressCCtx failed : %s", ZSTD_getErrorName(cSize)); @@ -801,7 +807,10 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxD { U32 const testLog = FUZ_rand(&lseed) % maxSrcLog; U32 const dictLog = FUZ_rand(&lseed) % maxSrcLog; - int const cLevel = (FUZ_rand(&lseed) % (ZSTD_maxCLevel() - (MAX(testLog, dictLog)/3))) + 1; + int const cLevel = (FUZ_rand(&lseed) % + (ZSTD_maxCLevel() - + (MAX(testLog, dictLog) / cLevelLimiter))) + + 1; maxTestSize = FUZ_rLogLength(&lseed, testLog); if (maxTestSize >= dstBufferSize) maxTestSize = dstBufferSize-1; @@ -927,6 +936,7 @@ int main(int argc, const char** argv) int result=0; U32 mainPause = 0; U32 maxDuration = 0; + int bigTests = 1; const char* programName = argv[0]; /* Check command line */ @@ -936,6 +946,9 @@ int main(int argc, const char** argv) /* Handle commands. Aggregated commands are allowed */ if (argument[0]=='-') { + + if (!strcmp(argument, "--no-big-tests")) { bigTests=0; continue; } + argument++; while (*argument!=0) { switch(*argument) @@ -1030,7 +1043,7 @@ int main(int argc, const char** argv) if (testNb==0) result = basicUnitTests(0, ((double)proba) / 100); /* constant seed for predictability */ if (!result) - result = fuzzerTests(seed, nbTests, testNb, maxDuration, ((double)proba) / 100); + result = fuzzerTests(seed, nbTests, testNb, maxDuration, ((double)proba) / 100, bigTests); if (mainPause) { int unused; DISPLAY("Press Enter \n"); diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index aa7367bcf..a03ee9b1f 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -552,7 +552,7 @@ static size_t FUZ_randomLength(U32* seed, U32 maxLog) #define CHECK(cond, ...) if (cond) { DISPLAY("Error => "); DISPLAY(__VA_ARGS__); \ DISPLAY(" (seed %u, test nb %u) \n", seed, testNb); goto _output_error; } -static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, double compressibility) +static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, double compressibility, int bigTests) { static const U32 maxSrcLog = 24; static const U32 maxSampleLog = 19; @@ -574,6 +574,7 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, double compres const BYTE* dict=NULL; /* can keep same dict on 2 consecutive tests */ size_t dictSize = 0; U32 oldTestLog = 0; + int const cLevelLimiter = bigTests ? 3 : 2; /* allocations */ cNoiseBuffer[0] = (BYTE*)malloc (srcBufferSize); @@ -646,7 +647,10 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, double compres } else { U32 const testLog = FUZ_rand(&lseed) % maxSrcLog; U32 const dictLog = FUZ_rand(&lseed) % maxSrcLog; - U32 const cLevel = (FUZ_rand(&lseed) % (ZSTD_maxCLevel() - (MAX(testLog, dictLog)/3))) + 1; + U32 const cLevel = (FUZ_rand(&lseed) % + (ZSTD_maxCLevel() - + (MAX(testLog, dictLog) / cLevelLimiter))) + + 1; maxTestSize = FUZ_rLogLength(&lseed, testLog); oldTestLog = testLog; /* random dictionary selection */ @@ -785,7 +789,7 @@ _output_error: /* Multi-threading version of fuzzer Tests */ -static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double compressibility) +static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double compressibility, int bigTests) { static const U32 maxSrcLog = 24; static const U32 maxSampleLog = 19; @@ -807,6 +811,7 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp const BYTE* dict=NULL; /* can keep same dict on 2 consecutive tests */ size_t dictSize = 0; U32 oldTestLog = 0; + int const cLevelLimiter = bigTests ? 3 : 2; /* allocations */ cNoiseBuffer[0] = (BYTE*)malloc (srcBufferSize); @@ -888,7 +893,10 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp } else { U32 const testLog = FUZ_rand(&lseed) % maxSrcLog; U32 const dictLog = FUZ_rand(&lseed) % maxSrcLog; - U32 const cLevel = (FUZ_rand(&lseed) % (ZSTD_maxCLevel() - (MAX(testLog, dictLog)/3))) + 1; + U32 const cLevel = (FUZ_rand(&lseed) % + (ZSTD_maxCLevel() - + (MAX(testLog, dictLog) / cLevelLimiter))) + + 1; maxTestSize = FUZ_rLogLength(&lseed, testLog); oldTestLog = testLog; /* random dictionary selection */ @@ -1063,6 +1071,7 @@ int main(int argc, const char** argv) int result=0; int mainPause = 0; int mtOnly = 0; + int bigTests = 1; const char* const programName = argv[0]; ZSTD_customMem const customMem = { allocFunction, freeFunction, NULL }; ZSTD_customMem const customNULL = { NULL, NULL, NULL }; @@ -1076,6 +1085,7 @@ int main(int argc, const char** argv) if (argument[0]=='-') { if (!strcmp(argument, "--mt")) { mtOnly=1; continue; } + if (!strcmp(argument, "--no-big-tests")) { bigTests=0; continue; } argument++; while (*argument!=0) { @@ -1181,8 +1191,8 @@ int main(int argc, const char** argv) result = basicUnitTests(0, ((double)proba) / 100, customMem); /* use custom memory allocation functions */ } } - if (!result && !mtOnly) result = fuzzerTests(seed, nbTests, testNb, ((double)proba) / 100); - if (!result) result = fuzzerTests_MT(seed, nbTests, testNb, ((double)proba) / 100); + if (!result && !mtOnly) result = fuzzerTests(seed, nbTests, testNb, ((double)proba) / 100, bigTests); + if (!result) result = fuzzerTests_MT(seed, nbTests, testNb, ((double)proba) / 100, bigTests); if (mainPause) { int unused; From 0b0b10ec83ee7b9739f8f1954746f1e46fab2683 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 20 Mar 2017 11:32:24 -0700 Subject: [PATCH 024/305] Add --no-big-tests flag for qemu fuzz-tests --- Makefile | 16 ++++++++-------- tests/Makefile | 8 ++++---- 2 files changed, 12 insertions(+), 12 deletions(-) diff --git a/Makefile b/Makefile index 49f29d782..631b884ad 100644 --- a/Makefile +++ b/Makefile @@ -152,16 +152,16 @@ ppc64build: clean CC=powerpc-linux-gnu-gcc CFLAGS="-m64 -Werror" $(MAKE) allarch armfuzz: clean - CC=arm-linux-gnueabi-gcc QEMU_SYS=qemu-arm-static MOREFLAGS="-static" $(MAKE) -C $(TESTDIR) fuzztest + CC=arm-linux-gnueabi-gcc QEMU_SYS=qemu-arm-static MOREFLAGS="-static" FUZZER_FLAGS=--no-big-tests $(MAKE) -C $(TESTDIR) fuzztest aarch64fuzz: clean - CC=aarch64-linux-gnu-gcc QEMU_SYS=qemu-aarch64-static MOREFLAGS="-static" $(MAKE) -C $(TESTDIR) fuzztest + CC=aarch64-linux-gnu-gcc QEMU_SYS=qemu-aarch64-static MOREFLAGS="-static" FUZZER_FLAGS=--no-big-tests $(MAKE) -C $(TESTDIR) fuzztest ppcfuzz: clean - CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc-static MOREFLAGS="-static" $(MAKE) -C $(TESTDIR) fuzztest + CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc-static MOREFLAGS="-static" FUZZER_FLAGS=--no-big-tests $(MAKE) -C $(TESTDIR) fuzztest ppc64fuzz: clean - CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc64-static MOREFLAGS="-m64 -static" $(MAKE) -C $(TESTDIR) fuzztest + CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc64-static MOREFLAGS="-m64 -static" FUZZER_FLAGS=--no-big-tests $(MAKE) -C $(TESTDIR) fuzztest gpptest: clean CC=g++ $(MAKE) -C $(PRGDIR) all CFLAGS="-O3 -Wall -Wextra -Wundef -Wshadow -Wcast-align -Werror" @@ -180,19 +180,19 @@ clangtest: clean armtest: clean $(MAKE) -C $(TESTDIR) datagen # use native, faster - $(MAKE) -C $(TESTDIR) test CC=arm-linux-gnueabi-gcc QEMU_SYS=qemu-arm-static ZSTDRTTEST= MOREFLAGS="-Werror -static" + $(MAKE) -C $(TESTDIR) test CC=arm-linux-gnueabi-gcc QEMU_SYS=qemu-arm-static ZSTDRTTEST= MOREFLAGS="-Werror -static" FUZZER_FLAGS=--no-big-tests aarch64test: $(MAKE) -C $(TESTDIR) datagen # use native, faster - $(MAKE) -C $(TESTDIR) test CC=aarch64-linux-gnu-gcc QEMU_SYS=qemu-aarch64-static ZSTDRTTEST= MOREFLAGS="-Werror -static" + $(MAKE) -C $(TESTDIR) test CC=aarch64-linux-gnu-gcc QEMU_SYS=qemu-aarch64-static ZSTDRTTEST= MOREFLAGS="-Werror -static" FUZZER_FLAGS=--no-big-tests ppctest: clean $(MAKE) -C $(TESTDIR) datagen # use native, faster - $(MAKE) -C $(TESTDIR) test CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc-static ZSTDRTTEST= MOREFLAGS="-Werror -Wno-attributes -static" + $(MAKE) -C $(TESTDIR) test CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc-static ZSTDRTTEST= MOREFLAGS="-Werror -Wno-attributes -static" FUZZER_FLAGS=--no-big-tests ppc64test: clean $(MAKE) -C $(TESTDIR) datagen # use native, faster - $(MAKE) -C $(TESTDIR) test CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc64-static ZSTDRTTEST= MOREFLAGS="-m64 -static" + $(MAKE) -C $(TESTDIR) test CC=powerpc-linux-gnu-gcc QEMU_SYS=qemu-ppc64-static ZSTDRTTEST= MOREFLAGS="-m64 -static" FUZZER_FLAGS=--no-big-tests arm-ppc-compilation: $(MAKE) -C $(PRGDIR) clean zstd CC=arm-linux-gnueabi-gcc QEMU_SYS=qemu-arm-static ZSTDRTTEST= MOREFLAGS="-Werror -static" diff --git a/tests/Makefile b/tests/Makefile index 9382fe80d..8a12a7324 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -300,10 +300,10 @@ test-fullbench32: fullbench32 datagen $(QEMU_SYS) ./fullbench32 -i1 -P0 test-fuzzer: fuzzer - $(QEMU_SYS) ./fuzzer $(FUZZERTEST) + $(QEMU_SYS) ./fuzzer $(FUZZERTEST) $(FUZZER_FLAGS) test-fuzzer32: fuzzer32 - $(QEMU_SYS) ./fuzzer32 $(FUZZERTEST) + $(QEMU_SYS) ./fuzzer32 $(FUZZERTEST) $(FUZZER_FLAGS) test-zbuff: zbufftest $(QEMU_SYS) ./zbufftest $(ZSTREAM_TESTTIME) @@ -312,10 +312,10 @@ test-zbuff32: zbufftest32 $(QEMU_SYS) ./zbufftest32 $(ZSTREAM_TESTTIME) test-zstream: zstreamtest - $(QEMU_SYS) ./zstreamtest $(ZSTREAM_TESTTIME) + $(QEMU_SYS) ./zstreamtest $(ZSTREAM_TESTTIME) $(FUZZER_FLAGS) test-zstream32: zstreamtest32 - $(QEMU_SYS) ./zstreamtest32 $(ZSTREAM_TESTTIME) + $(QEMU_SYS) ./zstreamtest32 $(ZSTREAM_TESTTIME) $(FUZZER_FLAGS) test-longmatch: longmatch $(QEMU_SYS) ./longmatch From 2f6c7e6a53501a5285e13d67721ae738140a175a Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 20 Mar 2017 12:38:19 -0700 Subject: [PATCH 025/305] Fix windows compiler warnings for decodecorpus --- tests/decodecorpus.c | 26 +++++++++++++------------- 1 file changed, 13 insertions(+), 13 deletions(-) diff --git a/tests/decodecorpus.c b/tests/decodecorpus.c index 183d20f74..4acd055af 100644 --- a/tests/decodecorpus.c +++ b/tests/decodecorpus.c @@ -325,7 +325,7 @@ static void writeFrameHeader(U32* seed, frame_t* frame) } } - DISPLAYLEVEL(2, " frame content size:\t%zu\n", fh.contentSize); + DISPLAYLEVEL(2, " frame content size:\t%u\n", (U32)fh.contentSize); DISPLAYLEVEL(2, " frame window size:\t%u\n", fh.windowSize); DISPLAYLEVEL(2, " content size flag:\t%d\n", contentSizeFlag); DISPLAYLEVEL(2, " single segment flag:\t%d\n", singleSegment); @@ -542,8 +542,8 @@ static size_t writeLiteralsBlockCompressed(U32* seed, frame_t* frame, size_t con op += compressedSize; compressedSize += hufHeaderSize; - DISPLAYLEVEL(5, " regenerated size: %zu\n", litSize); - DISPLAYLEVEL(5, " compressed size: %zu\n", compressedSize); + DISPLAYLEVEL(5, " regenerated size: %u\n", (U32)litSize); + DISPLAYLEVEL(5, " compressed size: %u\n", (U32)compressedSize); if (compressedSize >= litSize) { DISPLAYLEVEL(5, " trying again\n"); /* if we have to try again, reset the stats so we don't accidentally @@ -611,7 +611,7 @@ static U32 generateSequences(U32* seed, frame_t* frame, seqStore_t* seqStore, size_t const remainingMatch = contentSize - literalsSize; size_t excessMatch = 0; U32 numSequences = 0; - + U32 i; @@ -628,7 +628,7 @@ static U32 generateSequences(U32* seed, frame_t* frame, seqStore_t* seqStore, excessMatch = remainingMatch - numSequences * MIN_SEQ_LEN; } - DISPLAYLEVEL(5, " total match lengths: %zu\n", remainingMatch); + DISPLAYLEVEL(5, " total match lengths: %u\n", (U32)remainingMatch); for (i = 0; i < numSequences; i++) { /* Generate match and literal lengths by exponential distribution to @@ -694,8 +694,8 @@ static U32 generateSequences(U32* seed, frame_t* frame, seqStore_t* seqStore, } DISPLAYLEVEL(6, " LL: %5u OF: %5u ML: %5u", literalLen, offset, matchLen); - DISPLAYLEVEL(7, " srcPos: %8tu seqNb: %3u", - (BYTE*)srcPtr - (BYTE*)frame->srcStart, i); + DISPLAYLEVEL(7, " srcPos: %8u seqNb: %3u", + (U32)((BYTE*)srcPtr - (BYTE*)frame->srcStart), i); DISPLAYLEVEL(6, "\n"); if (offsetCode < 3) { DISPLAYLEVEL(7, " repeat offset: %d\n", repIndex); @@ -711,8 +711,8 @@ static U32 generateSequences(U32* seed, frame_t* frame, seqStore_t* seqStore, memcpy(srcPtr, literals, literalsSize); srcPtr += literalsSize; - DISPLAYLEVEL(6, " excess literals: %5zu", literalsSize); - DISPLAYLEVEL(7, " srcPos: %8tu", (BYTE*)srcPtr - (BYTE*)frame->srcStart); + DISPLAYLEVEL(6, " excess literals: %5u", (U32)literalsSize); + DISPLAYLEVEL(7, " srcPos: %8u", (U32)((BYTE*)srcPtr - (BYTE*)frame->srcStart)); DISPLAYLEVEL(6, "\n"); return numSequences; @@ -957,11 +957,11 @@ static size_t writeCompressedBlock(U32* seed, frame_t* frame, size_t contentSize literalsSize = writeLiteralsBlock(seed, frame, contentSize); - DISPLAYLEVEL(4, " literals size: %zu\n", literalsSize); + DISPLAYLEVEL(4, " literals size: %u\n", (U32)literalsSize); nbSeq = writeSequencesBlock(seed, frame, contentSize, literalsSize); - DISPLAYLEVEL(4, " number of sequences: %zu\n", nbSeq); + DISPLAYLEVEL(4, " number of sequences: %u\n", (U32)nbSeq); return (BYTE*)frame->data - blockStart; } @@ -977,7 +977,7 @@ static void writeBlock(U32* seed, frame_t* frame, size_t contentSize, BYTE *op = header + 3; DISPLAYLEVEL(3, " block:\n"); - DISPLAYLEVEL(3, " block content size: %zu\n", contentSize); + DISPLAYLEVEL(3, " block content size: %u\n", (U32)contentSize); DISPLAYLEVEL(3, " last block: %s\n", lastBlock ? "yes" : "no"); if (blockTypeDesc == 0) { @@ -1025,7 +1025,7 @@ static void writeBlock(U32* seed, frame_t* frame, size_t contentSize, frame->src = (BYTE*)frame->src + contentSize; DISPLAYLEVEL(3, " block type: %s\n", BLOCK_TYPES[blockType]); - DISPLAYLEVEL(3, " block size field: %zu\n", blockSize); + DISPLAYLEVEL(3, " block size field: %u\n", (U32)blockSize); header[0] = (lastBlock | (blockType << 1) | (blockSize << 3)) & 0xff; MEM_writeLE16(header + 1, blockSize >> 5); From c771977489d9971ab2b2811778ad819771159043 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 20 Mar 2017 16:02:24 -0700 Subject: [PATCH 026/305] Fix decodecorpus clang compile errors --- tests/decodecorpus.c | 34 +++++++++++++++++----------------- 1 file changed, 17 insertions(+), 17 deletions(-) diff --git a/tests/decodecorpus.c b/tests/decodecorpus.c index 4acd055af..f7b3c854f 100644 --- a/tests/decodecorpus.c +++ b/tests/decodecorpus.c @@ -98,7 +98,7 @@ static void RAND_bufferMaxSymb(U32* seed, void* ptr, size_t size, int maxSymb) BYTE* op = ptr; for (i = 0; i < size; i++) { - op[i] = RAND(seed) % (maxSymb + 1); + op[i] = (BYTE) (RAND(seed) % (maxSymb + 1)); } } @@ -134,8 +134,8 @@ static void RAND_genDist(U32* seed, BYTE* dist, double weight) { size_t i = 0; size_t statesLeft = DISTSIZE; - BYTE symb = RAND(seed) % 256; - BYTE step = (RAND(seed) % 256) | 1; /* force it to be odd so it's relatively prime to 256 */ + BYTE symb = (BYTE) (RAND(seed) % 256); + BYTE step = (BYTE) ((RAND(seed) % 256) | 1); /* force it to be odd so it's relatively prime to 256 */ while (i < DISTSIZE) { size_t states = ((size_t)(weight * statesLeft)) + 1; @@ -259,7 +259,7 @@ static void writeFrameHeader(U32* seed, frame_t* frame) /* Follow window algorithm from specification */ int const exponent = RAND(seed) % (MAX_WINDOW_LOG - 10); int const mantissa = RAND(seed) % 8; - windowByte = (exponent << 3) | mantissa; + windowByte = (BYTE) ((exponent << 3) | mantissa); fh.windowSize = (1U << (exponent + 10)); fh.windowSize += fh.windowSize / 8 * mantissa; } @@ -284,7 +284,7 @@ static void writeFrameHeader(U32* seed, frame_t* frame) if (contentSizeFlag && (fh.contentSize == 0 || !(RAND(seed) & 7))) { /* do single segment sometimes */ - fh.windowSize = fh.contentSize; + fh.windowSize = (U32) fh.contentSize; singleSegment = 1; } } @@ -307,7 +307,7 @@ static void writeFrameHeader(U32* seed, frame_t* frame) { BYTE const frameHeaderDescriptor = - (fcsCode << 6) | (singleSegment << 5) | (1 << 2); + (BYTE) ((fcsCode << 6) | (singleSegment << 5) | (1 << 2)); op[pos++] = frameHeaderDescriptor; } @@ -318,10 +318,10 @@ static void writeFrameHeader(U32* seed, frame_t* frame) if (contentSizeFlag) { switch (fcsCode) { default: /* Impossible */ - case 0: op[pos++] = fh.contentSize; break; - case 1: MEM_writeLE16(op + pos, fh.contentSize - 256); pos += 2; break; - case 2: MEM_writeLE32(op + pos, fh.contentSize); pos += 4; break; - case 3: MEM_writeLE64(op + pos, fh.contentSize); pos += 8; break; + case 0: op[pos++] = (BYTE) fh.contentSize; break; + case 1: MEM_writeLE16(op + pos, (U16) (fh.contentSize - 256)); pos += 2; break; + case 2: MEM_writeLE32(op + pos, (U32) fh.contentSize); pos += 4; break; + case 3: MEM_writeLE64(op + pos, (U64) fh.contentSize); pos += 8; break; } } @@ -385,7 +385,7 @@ static size_t writeLiteralsBlockSimple(U32* seed, frame_t* frame, size_t content op += litSize; } else { /* RLE literals */ - BYTE const symb = RAND(seed) % 256; + BYTE const symb = (BYTE) (RAND(seed) % 256); DISPLAYLEVEL(4, " rle literals: 0x%02x\n", (U32)symb); @@ -647,10 +647,10 @@ static U32 generateSequences(U32* seed, frame_t* frame, seqStore_t* seqStore, U32 offset, offsetCode, repIndex; /* bounds checks */ - matchLen = MIN(matchLen, excessMatch + MIN_SEQ_LEN); - literalLen = MIN(literalLen, literalsSize); + matchLen = (U32) MIN(matchLen, excessMatch + MIN_SEQ_LEN); + literalLen = MIN(literalLen, (U32) literalsSize); if (i == 0 && srcPtr == frame->srcStart && literalLen == 0) literalLen = 1; - if (i + 1 == numSequences) matchLen = MIN_SEQ_LEN + excessMatch; + if (i + 1 == numSequences) matchLen = MIN_SEQ_LEN + (U32) excessMatch; memcpy(srcPtr, literals, literalLen); srcPtr += literalLen; @@ -1027,8 +1027,8 @@ static void writeBlock(U32* seed, frame_t* frame, size_t contentSize, DISPLAYLEVEL(3, " block type: %s\n", BLOCK_TYPES[blockType]); DISPLAYLEVEL(3, " block size field: %u\n", (U32)blockSize); - header[0] = (lastBlock | (blockType << 1) | (blockSize << 3)) & 0xff; - MEM_writeLE16(header + 1, blockSize >> 5); + header[0] = (BYTE) ((lastBlock | (blockType << 1) | (blockSize << 3)) & 0xff); + MEM_writeLE16(header + 1, (U16) (blockSize >> 5)); frame->data = op; } @@ -1300,7 +1300,7 @@ static int generateCorpus(U32 seed, unsigned numFiles, const char* const path, *********************************************************/ static U32 makeSeed(void) { - U32 t = time(NULL); + U32 t = (U32) time(NULL); return XXH32(&t, sizeof(t), 0) % 65536; } From 5a2b85702931e0d37e4c633f22328b18ea434c1f Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 14 Mar 2017 15:59:57 -0700 Subject: [PATCH 027/305] Update appveyor.yml to have short and long tests --- appveyor.yml | 246 +++++++++++++++++++++++++++++------------------ appveyor_old.yml | 187 +++++++++++++++++++++++++++++++++++ tests/Makefile | 6 +- 3 files changed, 342 insertions(+), 97 deletions(-) create mode 100644 appveyor_old.yml diff --git a/appveyor.yml b/appveyor.yml index 9507fec6e..26ad31002 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -1,66 +1,81 @@ -version: 1.0.{build} -environment: - matrix: - - COMPILER: "gcc" - PLATFORM: "mingw64" - MAKE_PARAMS: '"make test && make lib && make -C tests test-symbols fullbench-dll fullbench-lib"' - - COMPILER: "gcc" - PLATFORM: "mingw32" - MAKE_PARAMS: '"make -C tests test-zstd test-fullbench test-fuzzer test-invalidDictionaries"' - - COMPILER: "gcc" - PLATFORM: "clang" - MAKE_PARAMS: '"make -C tests zstd fullbench fuzzer paramgrill datagen CC=clang MOREFLAGS="--target=x86_64-w64-mingw32 -Werror -Wconversion -Wno-sign-conversion""' - - COMPILER: "visual" - CONFIGURATION: "Debug" - PLATFORM: "x64" - - COMPILER: "visual" - CONFIGURATION: "Debug" - PLATFORM: "Win32" - - COMPILER: "visual" - CONFIGURATION: "Release" - PLATFORM: "x64" - - COMPILER: "visual" - CONFIGURATION: "Release" - PLATFORM: "Win32" +- + version: 1.0.{build} + branches: + only: + - dev + - master + environment: + matrix: + - COMPILER: "gcc" + HOST: "mingw" + PLATFORM: "x64" + SCRIPT: "make allarch && make -C tests test-symbols fullbench-dll fullbench-lib" + ARTIFACT: "true" + - COMPILER: "gcc" + HOST: "mingw" + PLATFORM: "x86" + SCRIPT: "make lib && make zstd && make -C tests allnothread" + ARTIFACT: "true" + - COMPILER: "clang" + HOST: "mingw" + PLATFORM: "x64" + SCRIPT: "MOREFLAGS='--target=x86_64-w64-mingw32 -Werror -Wconversion -Wno-sign-conversion' make allarch" -install: + - COMPILER: "gcc" + HOST: "mingw" + PLATFORM: "x64" + SCRIPT: "" + TEST: "cmake" + + - COMPILER: "gcc" + HOST: "mingw" + PLATFORM: "x64" + SCRIPT: "" + TEST: "pzstd" + + - COMPILER: "visual" + HOST: "visual" + PLATFORM: "x64" + CONFIGURATION: "Debug" + - COMPILER: "visual" + HOST: "visual" + PLATFORM: "Win32" + CONFIGURATION: "Debug" + - COMPILER: "visual" + HOST: "visual" + PLATFORM: "x64" + CONFIGURATION: "Release" + - COMPILER: "visual" + HOST: "visual" + PLATFORM: "Win32" + CONFIGURATION: "Release" + + install: - ECHO Installing %COMPILER% %PLATFORM% %CONFIGURATION% - - MKDIR bin - - if [%COMPILER%]==[gcc] SET PATH_ORIGINAL=%PATH% - - if [%COMPILER%]==[gcc] ( - SET "PATH_MINGW32=c:\MinGW\bin;c:\MinGW\usr\bin" && - SET "PATH_MINGW64=c:\msys64\mingw64\bin;c:\msys64\usr\bin" && + - SET PATH_ORIGINAL=%PATH% + - if [%HOST%]==[mingw] ( + SET "PATH_MINGW32=C:\MinGW\bin;C:\MinGW\usr\bin" && + SET "PATH_MINGW64=C:\msys64\mingw64\bin;C:\msys64\usr\bin" && COPY C:\msys64\usr\bin\make.exe C:\MinGW\bin\make.exe && COPY C:\MinGW\bin\gcc.exe C:\MinGW\bin\cc.exe - ) else ( - IF [%PLATFORM%]==[x64] (SET ADDITIONALPARAM=/p:LibraryPath="C:\Program Files\Microsoft SDKs\Windows\v7.1\lib\x64;c:\Program Files (x86)\Microsoft Visual Studio 10.0\VC\lib\amd64;C:\Program Files (x86)\Microsoft Visual Studio 10.0\;C:\Program Files (x86)\Microsoft Visual Studio 10.0\lib\amd64;") + ) + - IF [%HOST%]==[visual] IF [%PLATFORM%]==[x64] ( + SET ADDITIONALPARAM=/p:LibraryPath="C:\Program Files\Microsoft SDKs\Windows\v7.1\lib\x64;c:\Program Files (x86)\Microsoft Visual Studio 10.0\VC\lib\amd64;C:\Program Files (x86)\Microsoft Visual Studio 10.0\;C:\Program Files (x86)\Microsoft Visual Studio 10.0\lib\amd64;" ) -build_script: - - ECHO Building %COMPILER% %PLATFORM% %CONFIGURATION% - - if [%PLATFORM%]==[mingw32] SET PATH=%PATH_MINGW32%;%PATH_ORIGINAL% - - if [%PLATFORM%]==[mingw64] SET PATH=%PATH_MINGW64%;%PATH_ORIGINAL% - - if [%PLATFORM%]==[clang] SET PATH=%PATH_MINGW64%;%PATH_ORIGINAL% - - if [%COMPILER%]==[gcc] ( - ECHO *** && - ECHO *** Building %PLATFORM% && - ECHO *** && + build_script: + - if [%HOST%]==[mingw] ( + ( if [%PLATFORM%]==[x64] ( + SET "PATH=%PATH_MINGW64%;%PATH_ORIGINAL%" + ) else if [%PLATFORM%]==[x86] ( + SET "PATH=%PATH_MINGW32%;%PATH_ORIGINAL%" + ) ) && make -v && - cc -v && - ECHO %MAKE_PARAMS% && - sh -c %MAKE_PARAMS% + sh -c "%COMPILER% -v" && + sh -c "CC=%COMPILER% %SCRIPT%" && + ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] COPY programs\zstd.exe zstd_%PLATFORM%.exe && appveyor PushArtifact zstd_%PLATFORM%.exe ) ) - - if [%PLATFORM%]==[clang] COPY tests\fuzzer.exe tests\fuzzer_clang.exe - - if [%COMPILER%]==[gcc] if [%PLATFORM%]==[mingw64] ( - COPY programs\zstd.exe bin\zstd.exe && - appveyor PushArtifact bin\zstd.exe - ) - - if [%COMPILER%]==[gcc] if [%PLATFORM%]==[mingw32] ( - COPY programs\zstd.exe bin\zstd32.exe && - appveyor PushArtifact bin\zstd32.exe - ) - - if [%COMPILER%]==[gcc] make clean - - if [%COMPILER%]==[visual] ( + - if [%HOST%]==[visual] ( ECHO *** && ECHO *** Building Visual Studio 2008 %PLATFORM%\%CONFIGURATION% in %APPVEYOR_BUILD_FOLDER% && ECHO *** && @@ -111,29 +126,24 @@ build_script: COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe tests\ ) -test_script: + test_script: - ECHO Testing %COMPILER% %PLATFORM% %CONFIGURATION% - - SET FUZZERTEST=-T1mn - - if [%COMPILER%]==[gcc] if [%PLATFORM%]==[clang] ( - tests\fuzzer_clang.exe %FUZZERTEST% && - ECHO *** && - ECHO *** Building cmake for %PLATFORM% && - ECHO *** && + - if [%TEST%]==[cmake] ( mkdir build\cmake\build && cd build\cmake\build && cmake -G "Visual Studio 14 2015 Win64" .. && cd ..\..\.. && - make clean && - ECHO *** && - ECHO *** Building pzstd for %PLATFORM% && - ECHO *** && + make clean + ) + - if [%TEST%]==[pzstd] ( make -C contrib\pzstd googletest-mingw64 && make -C contrib\pzstd pzstd.exe && make -C contrib\pzstd tests && make -C contrib\pzstd check && make -C contrib\pzstd clean ) - - if [%COMPILER%]==[visual] if [%CONFIGURATION%]==[Release] ( + - if [%HOST%]==[visual] if [%CONFIGURATION%]==[Release] ( + SET FUZZERTEST=-T1mn && CD tests && SET ZSTD=./zstd.exe && sh -e playTests.sh --test-large-data && @@ -146,33 +156,79 @@ test_script: fuzzer_VS2015_%PLATFORM%_Release.exe %FUZZERTEST% ) -branches: - only: - - dev - - master +- + version: 1.0.{build} + branches: + except: + - dev + - master + environment: + matrix: + - COMPILER: "gcc" + HOST: "mingw" + PLATFORM: "x64" + SCRIPT: "make allarch" + - COMPILER: "gcc" + HOST: "mingw" + PLATFORM: "x86" + SCRIPT: "make lib && make zstd && make -C tests allnothread" + - COMPILER: "clang" + HOST: "mingw" + PLATFORM: "x64" + SCRIPT: "MOREFLAGS='--target=x86_64-w64-mingw32 -Werror -Wconversion -Wno-sign-conversion' make allarch" -artifacts: - - path: bin\zstd.exe - - path: bin\zstd32.exe + - COMPILER: "visual" + HOST: "visual" + PLATFORM: "x64" + CONFIGURATION: "Debug" + - COMPILER: "visual" + HOST: "visual" + PLATFORM: "Win32" + CONFIGURATION: "Debug" + - COMPILER: "visual" + HOST: "visual" + PLATFORM: "x64" + CONFIGURATION: "Release" + - COMPILER: "visual" + HOST: "visual" + PLATFORM: "Win32" + CONFIGURATION: "Release" -deploy: -- provider: GitHub - auth_token: - secure: LgJo8emYc3sFnlNWkGl4/VYK3nk/8+RagcsqDlAi3xeqNGNutnKjcftjg84uJoT4 - artifact: bin\zstd.exe - force_update: true - on: - branch: autobuild - COMPILER: gcc - PLATFORM: "mingw64" - appveyor_repo_tag: true -- provider: GitHub - auth_token: - secure: LgJo8emYc3sFnlNWkGl4/VYK3nk/8+RagcsqDlAi3xeqNGNutnKjcftjg84uJoT4 - artifact: bin\zstd32.exe - force_update: true - on: - branch: autobuild - COMPILER: gcc - PLATFORM: "mingw32" - appveyor_repo_tag: true + install: + - ECHO Installing %COMPILER% %PLATFORM% %CONFIGURATION% + - SET PATH_ORIGINAL=%PATH% + - if [%HOST%]==[mingw] ( + SET "PATH_MINGW32=C:\MinGW\bin;C:\MinGW\usr\bin" && + SET "PATH_MINGW64=C:\msys64\mingw64\bin;C:\msys64\usr\bin" && + COPY C:\msys64\usr\bin\make.exe C:\MinGW\bin\make.exe && + COPY C:\MinGW\bin\gcc.exe C:\MinGW\bin\cc.exe + ) + - IF [%HOST%]==[visual] IF [%PLATFORM%]==[x64] ( + SET ADDITIONALPARAM=/p:LibraryPath="C:\Program Files\Microsoft SDKs\Windows\v7.1\lib\x64;c:\Program Files (x86)\Microsoft Visual Studio 10.0\VC\lib\amd64;C:\Program Files (x86)\Microsoft Visual Studio 10.0\;C:\Program Files (x86)\Microsoft Visual Studio 10.0\lib\amd64;" + ) + + build_script: + - ECHO Building %COMPILER% %PLATFORM% %CONFIGURATION% + - if [%HOST%]==[mingw] ( + ( if [%PLATFORM%]==[x64] ( + SET "PATH=%PATH_MINGW64%;%PATH_ORIGINAL%" + ) else if [%PLATFORM%]==[x86] ( + SET "PATH=%PATH_MINGW32%;%PATH_ORIGINAL%" + ) ) && + make -v && + sh -c "%COMPILER% -v" && + sh -c "CC=%COMPILER% %SCRIPT%" + ) + - if [%HOST%]==[visual] ( + ECHO *** && + ECHO *** Building Visual Studio 2015 %PLATFORM%\%CONFIGURATION% && + ECHO *** && + msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v140 /p:ForceImportBeforeCppTargets=%APPVEYOR_BUILD_FOLDER%\build\VS2010\CompileAsCpp.props /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && + MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && + msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v140 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && + MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && + COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2015_%PLATFORM%_%CONFIGURATION%.exe && + COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe tests\ + ) diff --git a/appveyor_old.yml b/appveyor_old.yml new file mode 100644 index 000000000..124850a2f --- /dev/null +++ b/appveyor_old.yml @@ -0,0 +1,187 @@ +version: 1.0.{build} +environment: + - + branches: + only: + - dev + - master + matrix: + - COMPILER: "gcc" + PLATFORM: "mingw64" + MAKE_PARAMS: '"make test && make lib && make -C tests test-symbols fullbench-dll fullbench-lib"' + - COMPILER: "gcc" + PLATFORM: "mingw32" + MAKE_PARAMS: '"make -C tests test-zstd test-fullbench test-fuzzer test-invalidDictionaries"' + - COMPILER: "gcc" + PLATFORM: "clang" + MAKE_PARAMS: '"make -C tests zstd fullbench fuzzer paramgrill datagen CC=clang MOREFLAGS="--target=x86_64-w64-mingw32 -Werror -Wconversion -Wno-sign-conversion""' + - COMPILER: "visual" + CONFIGURATION: "Debug" + PLATFORM: "x64" + - COMPILER: "visual" + CONFIGURATION: "Debug" + PLATFORM: "Win32" + - COMPILER: "visual" + CONFIGURATION: "Release" + PLATFORM: "x64" + - COMPILER: "visual" + CONFIGURATION: "Release" + PLATFORM: "Win32" + + - + branches: + only: + - ci + matrix: + - COMPILER: "gcc" + PLATFORM: "mingw64" + MAKE_PARAMS: '"make all"' + +install: + - ECHO Installing %COMPILER% %PLATFORM% %CONFIGURATION% + - MKDIR bin + - if [%COMPILER%]==[gcc] SET PATH_ORIGINAL=%PATH% + - if [%COMPILER%]==[gcc] ( + SET "PATH_MINGW32=c:\MinGW\bin;c:\MinGW\usr\bin" && + SET "PATH_MINGW64=c:\msys64\mingw64\bin;c:\msys64\usr\bin" && + COPY C:\msys64\usr\bin\make.exe C:\MinGW\bin\make.exe && + COPY C:\MinGW\bin\gcc.exe C:\MinGW\bin\cc.exe + ) else ( + IF [%PLATFORM%]==[x64] (SET ADDITIONALPARAM=/p:LibraryPath="C:\Program Files\Microsoft SDKs\Windows\v7.1\lib\x64;c:\Program Files (x86)\Microsoft Visual Studio 10.0\VC\lib\amd64;C:\Program Files (x86)\Microsoft Visual Studio 10.0\;C:\Program Files (x86)\Microsoft Visual Studio 10.0\lib\amd64;") + ) + +build_script: + - ECHO Building %COMPILER% %PLATFORM% %CONFIGURATION% + - if [%PLATFORM%]==[mingw32] SET PATH=%PATH_MINGW32%;%PATH_ORIGINAL% + - if [%PLATFORM%]==[mingw64] SET PATH=%PATH_MINGW64%;%PATH_ORIGINAL% + - if [%PLATFORM%]==[clang] SET PATH=%PATH_MINGW64%;%PATH_ORIGINAL% + - if [%COMPILER%]==[gcc] ( + ECHO *** && + ECHO *** Building %PLATFORM% && + ECHO *** && + make -v && + cc -v && + ECHO %MAKE_PARAMS% && + sh -c %MAKE_PARAMS% + ) + - if [%PLATFORM%]==[clang] COPY tests\fuzzer.exe tests\fuzzer_clang.exe + - if [%COMPILER%]==[gcc] if [%PLATFORM%]==[mingw64] ( + COPY programs\zstd.exe bin\zstd.exe && + appveyor PushArtifact bin\zstd.exe + ) + - if [%COMPILER%]==[gcc] if [%PLATFORM%]==[mingw32] ( + COPY programs\zstd.exe bin\zstd32.exe && + appveyor PushArtifact bin\zstd32.exe + ) + - if [%COMPILER%]==[gcc] make clean + - if [%COMPILER%]==[visual] ( + ECHO *** && + ECHO *** Building Visual Studio 2008 %PLATFORM%\%CONFIGURATION% in %APPVEYOR_BUILD_FOLDER% && + ECHO *** && + msbuild "build\VS2008\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v90 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2008\bin\%PLATFORM%\%CONFIGURATION%\*.exe && + MD5sum build/VS2008/bin/%PLATFORM%/%CONFIGURATION%/*.exe && + COPY build\VS2008\bin\%PLATFORM%\%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2008_%PLATFORM%_%CONFIGURATION%.exe && + ECHO *** && + ECHO *** Building Visual Studio 2010 %PLATFORM%\%CONFIGURATION% && + ECHO *** && + msbuild "build\VS2010\zstd.sln" %ADDITIONALPARAM% /m /verbosity:minimal /property:PlatformToolset=v100 /p:ForceImportBeforeCppTargets=%APPVEYOR_BUILD_FOLDER%\build\VS2010\CompileAsCpp.props /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && + MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && + msbuild "build\VS2010\zstd.sln" %ADDITIONALPARAM% /m /verbosity:minimal /property:PlatformToolset=v100 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && + MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && + COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2010_%PLATFORM%_%CONFIGURATION%.exe && + ECHO *** && + ECHO *** Building Visual Studio 2012 %PLATFORM%\%CONFIGURATION% && + ECHO *** && + msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v110 /p:ForceImportBeforeCppTargets=%APPVEYOR_BUILD_FOLDER%\build\VS2010\CompileAsCpp.props /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && + MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && + msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v110 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && + MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && + COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2012_%PLATFORM%_%CONFIGURATION%.exe && + ECHO *** && + ECHO *** Building Visual Studio 2013 %PLATFORM%\%CONFIGURATION% && + ECHO *** && + msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v120 /p:ForceImportBeforeCppTargets=%APPVEYOR_BUILD_FOLDER%\build\VS2010\CompileAsCpp.props /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && + MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && + msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v120 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && + MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && + COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2013_%PLATFORM%_%CONFIGURATION%.exe && + ECHO *** && + ECHO *** Building Visual Studio 2015 %PLATFORM%\%CONFIGURATION% && + ECHO *** && + msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v140 /p:ForceImportBeforeCppTargets=%APPVEYOR_BUILD_FOLDER%\build\VS2010\CompileAsCpp.props /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && + MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && + msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v140 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && + DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && + MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && + COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2015_%PLATFORM%_%CONFIGURATION%.exe && + COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe tests\ + ) + +test_script: + - ECHO Testing %COMPILER% %PLATFORM% %CONFIGURATION% + - SET FUZZERTEST=-T1mn + - if [%COMPILER%]==[gcc] if [%PLATFORM%]==[clang] ( + tests\fuzzer_clang.exe %FUZZERTEST% && + ECHO *** && + ECHO *** Building cmake for %PLATFORM% && + ECHO *** && + mkdir build\cmake\build && + cd build\cmake\build && + cmake -G "Visual Studio 14 2015 Win64" .. && + cd ..\..\.. && + make clean && + ECHO *** && + ECHO *** Building pzstd for %PLATFORM% && + ECHO *** && + make -C contrib\pzstd googletest-mingw64 && + make -C contrib\pzstd pzstd.exe && + make -C contrib\pzstd tests && + make -C contrib\pzstd check && + make -C contrib\pzstd clean + ) + - if [%COMPILER%]==[visual] if [%CONFIGURATION%]==[Release] ( + CD tests && + SET ZSTD=./zstd.exe && + sh -e playTests.sh --test-large-data && + fullbench.exe -i1 && + fullbench.exe -i1 -P0 && + fuzzer_VS2008_%PLATFORM%_Release.exe %FUZZERTEST% && + fuzzer_VS2010_%PLATFORM%_Release.exe %FUZZERTEST% && + fuzzer_VS2012_%PLATFORM%_Release.exe %FUZZERTEST% && + fuzzer_VS2013_%PLATFORM%_Release.exe %FUZZERTEST% && + fuzzer_VS2015_%PLATFORM%_Release.exe %FUZZERTEST% + ) + +artifacts: + - path: bin\zstd.exe + - path: bin\zstd32.exe + +deploy: +- provider: GitHub + auth_token: + secure: LgJo8emYc3sFnlNWkGl4/VYK3nk/8+RagcsqDlAi3xeqNGNutnKjcftjg84uJoT4 + artifact: bin\zstd.exe + force_update: true + on: + branch: autobuild + COMPILER: gcc + PLATFORM: "mingw64" + appveyor_repo_tag: true +- provider: GitHub + auth_token: + secure: LgJo8emYc3sFnlNWkGl4/VYK3nk/8+RagcsqDlAi3xeqNGNutnKjcftjg84uJoT4 + artifact: bin\zstd32.exe + force_update: true + on: + branch: autobuild + COMPILER: gcc + PLATFORM: "mingw32" + appveyor_repo_tag: true diff --git a/tests/Makefile b/tests/Makefile index 8a12a7324..809c90df5 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -65,14 +65,16 @@ FUZZERTEST ?= -T5mn ZSTDRTTEST = --test-large-data DECODECORPUS_TESTTIME ?= -T30 -.PHONY: default all all32 dll clean test test32 test-all namespaceTest versionsTest +.PHONY: default all all32 allnothread dll clean test test32 test-all namespaceTest versionsTest default: fullbench -all: fullbench fuzzer zstreamtest paramgrill datagen zbufftest +all: fullbench fuzzer zstreamtest paramgrill datagen zbufftest decodecorpus all32: fullbench32 fuzzer32 zstreamtest32 zbufftest32 +allnothread: fullbench fuzzer paramgrill datagen zbufftest decodecorpus + dll: fuzzer-dll zstreamtest-dll zbufftest-dll zstd: From 8013c86c7d9985b9d120930e6bca63f9206ef856 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Thu, 16 Mar 2017 14:03:54 +0100 Subject: [PATCH 028/305] Improve resolving ROOT_DIR --- build/cmake/CMakeLists.txt | 1 + build/cmake/contrib/pzstd/CMakeLists.txt | 9 +++------ build/cmake/lib/CMakeLists.txt | 7 ++----- build/cmake/programs/CMakeLists.txt | 9 +++------ build/cmake/tests/CMakeLists.txt | 9 +++------ 5 files changed, 12 insertions(+), 23 deletions(-) diff --git a/build/cmake/CMakeLists.txt b/build/cmake/CMakeLists.txt index 4805cc2c9..d0f6daade 100644 --- a/build/cmake/CMakeLists.txt +++ b/build/cmake/CMakeLists.txt @@ -9,6 +9,7 @@ PROJECT(zstd) CMAKE_MINIMUM_REQUIRED(VERSION 2.8.7) +SET(ZSTD_SOURCE_DIR "${CMAKE_SOURCE_DIR}/../..") OPTION(ZSTD_LEGACY_SUPPORT "LEGACY SUPPORT" OFF) OPTION(ZSTD_MULTITHREAD_SUPPORT "MULTITHREADING SUPPORT" ON) diff --git a/build/cmake/contrib/pzstd/CMakeLists.txt b/build/cmake/contrib/pzstd/CMakeLists.txt index 2a3663f31..f89be8901 100644 --- a/build/cmake/contrib/pzstd/CMakeLists.txt +++ b/build/cmake/contrib/pzstd/CMakeLists.txt @@ -14,13 +14,10 @@ PROJECT(pzstd) SET(CMAKE_INCLUDE_CURRENT_DIR TRUE) -# Define project root directory -SET(ROOT_DIR ../../../..) - # Define programs directory, where sources and header files are located -SET(LIBRARY_DIR ${ROOT_DIR}/lib) -SET(PROGRAMS_DIR ${ROOT_DIR}/programs) -SET(PZSTD_DIR ${ROOT_DIR}/contrib/pzstd) +SET(LIBRARY_DIR ${ZSTD_SOURCE_DIR}/lib) +SET(PROGRAMS_DIR ${ZSTD_SOURCE_DIR}/programs) +SET(PZSTD_DIR ${ZSTD_SOURCE_DIR}/contrib/pzstd) INCLUDE_DIRECTORIES(${PROGRAMS_DIR} ${LIBRARY_DIR} ${LIBRARY_DIR}/common ${PZSTD_DIR}) ADD_EXECUTABLE(pzstd ${PZSTD_DIR}/main.cpp ${PZSTD_DIR}/Options.cpp ${PZSTD_DIR}/Pzstd.cpp ${PZSTD_DIR}/SkippableFrame.cpp) diff --git a/build/cmake/lib/CMakeLists.txt b/build/cmake/lib/CMakeLists.txt index 1950d97cd..ffb6bd4a3 100644 --- a/build/cmake/lib/CMakeLists.txt +++ b/build/cmake/lib/CMakeLists.txt @@ -22,11 +22,8 @@ PROJECT(libzstd) SET(CMAKE_INCLUDE_CURRENT_DIR TRUE) -# Define project root directory -SET(ROOT_DIR ../../..) - # Define library directory, where sources and header files are located -SET(LIBRARY_DIR ${ROOT_DIR}/lib) +SET(LIBRARY_DIR ${ZSTD_SOURCE_DIR}/lib) INCLUDE_DIRECTORIES(${LIBRARY_DIR} ${LIBRARY_DIR}/common) # Read file content @@ -97,7 +94,7 @@ IF (ZSTD_LEGACY_SUPPORT) ENDIF (ZSTD_LEGACY_SUPPORT) IF (MSVC) - SET(MSVC_RESOURCE_DIR ${ROOT_DIR}/build/VS2010/libzstd-dll) + SET(MSVC_RESOURCE_DIR ${ZSTD_SOURCE_DIR}/build/VS2010/libzstd-dll) SET(PlatformDependResources ${MSVC_RESOURCE_DIR}/libzstd-dll.rc) ENDIF (MSVC) diff --git a/build/cmake/programs/CMakeLists.txt b/build/cmake/programs/CMakeLists.txt index c88ee5cc9..38b4c1e7d 100644 --- a/build/cmake/programs/CMakeLists.txt +++ b/build/cmake/programs/CMakeLists.txt @@ -14,12 +14,9 @@ PROJECT(programs) SET(CMAKE_INCLUDE_CURRENT_DIR TRUE) -# Define project root directory -SET(ROOT_DIR ../../..) - # Define programs directory, where sources and header files are located -SET(LIBRARY_DIR ${ROOT_DIR}/lib) -SET(PROGRAMS_DIR ${ROOT_DIR}/programs) +SET(LIBRARY_DIR ${ZSTD_SOURCE_DIR}/lib) +SET(PROGRAMS_DIR ${ZSTD_SOURCE_DIR}/programs) INCLUDE_DIRECTORIES(${PROGRAMS_DIR} ${LIBRARY_DIR} ${LIBRARY_DIR}/common ${LIBRARY_DIR}/compress ${LIBRARY_DIR}/dictBuilder) IF (ZSTD_LEGACY_SUPPORT) @@ -28,7 +25,7 @@ IF (ZSTD_LEGACY_SUPPORT) ENDIF (ZSTD_LEGACY_SUPPORT) IF (MSVC) - SET(MSVC_RESOURCE_DIR ${ROOT_DIR}/build/VS2010/zstd) + SET(MSVC_RESOURCE_DIR ${ZSTD_SOURCE_DIR}/build/VS2010/zstd) SET(PlatformDependResources ${MSVC_RESOURCE_DIR}/zstd.rc) ENDIF (MSVC) diff --git a/build/cmake/tests/CMakeLists.txt b/build/cmake/tests/CMakeLists.txt index 53a699449..cb327e48c 100644 --- a/build/cmake/tests/CMakeLists.txt +++ b/build/cmake/tests/CMakeLists.txt @@ -34,13 +34,10 @@ PROJECT(tests) SET(CMAKE_INCLUDE_CURRENT_DIR TRUE) -# Define project root directory -SET(ROOT_DIR ../../..) - # Define programs directory, where sources and header files are located -SET(LIBRARY_DIR ${ROOT_DIR}/lib) -SET(PROGRAMS_DIR ${ROOT_DIR}/programs) -SET(TESTS_DIR ${ROOT_DIR}/tests) +SET(LIBRARY_DIR ${ZSTD_SOURCE_DIR}/lib) +SET(PROGRAMS_DIR ${ZSTD_SOURCE_DIR}/programs) +SET(TESTS_DIR ${ZSTD_SOURCE_DIR}/tests) INCLUDE_DIRECTORIES(${TESTS_DIR} ${PROGRAMS_DIR} ${LIBRARY_DIR} ${LIBRARY_DIR}/common ${LIBRARY_DIR}/compress ${LIBRARY_DIR}/dictBuilder) ADD_EXECUTABLE(fullbench ${PROGRAMS_DIR}/datagen.c ${TESTS_DIR}/fullbench.c) From 623baf513e9331954ab76c44eabe63881476969a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Thu, 16 Mar 2017 14:06:49 +0100 Subject: [PATCH 029/305] Move GetLibraryVersion function to its own module file --- build/cmake/CMakeModules/GetLibraryVersion.cmake | 9 +++++++++ build/cmake/lib/CMakeLists.txt | 14 ++------------ 2 files changed, 11 insertions(+), 12 deletions(-) create mode 100644 build/cmake/CMakeModules/GetLibraryVersion.cmake diff --git a/build/cmake/CMakeModules/GetLibraryVersion.cmake b/build/cmake/CMakeModules/GetLibraryVersion.cmake new file mode 100644 index 000000000..a3e2fd25f --- /dev/null +++ b/build/cmake/CMakeModules/GetLibraryVersion.cmake @@ -0,0 +1,9 @@ +function(GetLibraryVersion _header _major _minor _release) + # Read file content + FILE(READ ${_header} CONTENT) + + string(REGEX MATCHALL ".*define ZSTD_VERSION_MAJOR+.* ([0-9]+).*define ZSTD_VERSION_MINOR+.* ([0-9]+).*define ZSTD_VERSION_RELEASE+.* ([0-9]+)" VERSION_REGEX "${CONTENT}") + SET(${_major} ${CMAKE_MATCH_1} PARENT_SCOPE) + SET(${_minor} ${CMAKE_MATCH_2} PARENT_SCOPE) + SET(${_release} ${CMAKE_MATCH_3} PARENT_SCOPE) +endfunction() diff --git a/build/cmake/lib/CMakeLists.txt b/build/cmake/lib/CMakeLists.txt index ffb6bd4a3..ed5b36b21 100644 --- a/build/cmake/lib/CMakeLists.txt +++ b/build/cmake/lib/CMakeLists.txt @@ -10,15 +10,8 @@ # - zstd homepage : http://www.zstd.net/ # ################################################################ -# Get library version based on information from input content (use regular exp) -function(GetLibraryVersion _content _outputVar1 _outputVar2 _outputVar3) - string(REGEX MATCHALL ".*define ZSTD_VERSION_MAJOR+.* ([0-9]+).*define ZSTD_VERSION_MINOR+.* ([0-9]+).*define ZSTD_VERSION_RELEASE+.* ([0-9]+)" VERSION_REGEX "${_content}") - SET(${_outputVar1} ${CMAKE_MATCH_1} PARENT_SCOPE) - SET(${_outputVar2} ${CMAKE_MATCH_2} PARENT_SCOPE) - SET(${_outputVar3} ${CMAKE_MATCH_3} PARENT_SCOPE) -endfunction() - PROJECT(libzstd) +INCLUDE(${CMAKE_SOURCE_DIR}/CMakeModules/GetLibraryVersion.cmake) SET(CMAKE_INCLUDE_CURRENT_DIR TRUE) @@ -26,11 +19,8 @@ SET(CMAKE_INCLUDE_CURRENT_DIR TRUE) SET(LIBRARY_DIR ${ZSTD_SOURCE_DIR}/lib) INCLUDE_DIRECTORIES(${LIBRARY_DIR} ${LIBRARY_DIR}/common) -# Read file content -FILE(READ ${LIBRARY_DIR}/zstd.h HEADER_CONTENT) - # Parse version -GetLibraryVersion("${HEADER_CONTENT}" LIBVER_MAJOR LIBVER_MINOR LIBVER_RELEASE) +GetLibraryVersion(${LIBRARY_DIR}/zstd.h LIBVER_MAJOR LIBVER_MINOR LIBVER_RELEASE) MESSAGE("ZSTD VERSION ${LIBVER_MAJOR}.${LIBVER_MINOR}.${LIBVER_RELEASE}") SET(Sources From c03d7b898d87d50bd2d7dc625f59a4ab60863fb5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Thu, 16 Mar 2017 14:08:13 +0100 Subject: [PATCH 030/305] Cleanup lib rules --- build/cmake/lib/CMakeLists.txt | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/build/cmake/lib/CMakeLists.txt b/build/cmake/lib/CMakeLists.txt index ed5b36b21..65b00b9ed 100644 --- a/build/cmake/lib/CMakeLists.txt +++ b/build/cmake/lib/CMakeLists.txt @@ -101,17 +101,15 @@ ENDIF (MSVC) # Define library base name IF (MSVC) SET(LIBRARY_BASE_NAME zstdlib) -ELSE () - SET(LIBRARY_BASE_NAME libzstd) -ENDIF (MSVC) -IF (MSVC) IF (CMAKE_SIZEOF_VOID_P MATCHES "8") SET(LIBRARY_ARCH_SUFFIX "_x64") ELSE () SET(LIBRARY_ARCH_SUFFIX "_x86") ENDIF (CMAKE_SIZEOF_VOID_P MATCHES "8") ELSE () + SET(LIBRARY_BASE_NAME libzstd) + SET(LIBRARY_ARCH_SUFFIX "") ENDIF (MSVC) From e8517a95f328163f8f8653682d1894339bbf055c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Thu, 16 Mar 2017 14:08:35 +0100 Subject: [PATCH 031/305] Add build documentation by gen_html --- build/cmake/contrib/CMakeLists.txt | 1 + build/cmake/contrib/gen_html/CMakeLists.txt | 31 +++++++++++++++++++++ 2 files changed, 32 insertions(+) create mode 100644 build/cmake/contrib/gen_html/CMakeLists.txt diff --git a/build/cmake/contrib/CMakeLists.txt b/build/cmake/contrib/CMakeLists.txt index 68e0881c5..c7d97aa95 100644 --- a/build/cmake/contrib/CMakeLists.txt +++ b/build/cmake/contrib/CMakeLists.txt @@ -13,4 +13,5 @@ PROJECT(contrib) ADD_SUBDIRECTORY(pzstd) +ADD_SUBDIRECTORY(gen_html) diff --git a/build/cmake/contrib/gen_html/CMakeLists.txt b/build/cmake/contrib/gen_html/CMakeLists.txt new file mode 100644 index 000000000..14f52c68d --- /dev/null +++ b/build/cmake/contrib/gen_html/CMakeLists.txt @@ -0,0 +1,31 @@ +# ################################################################ +# * Copyright (c) 2015-present, Yann Collet, Facebook, Inc. +# * All rights reserved. +# * +# * This source code is licensed under the BSD-style license found in the +# * LICENSE file in the root directory of this source tree. An additional grant +# * of patent rights can be found in the PATENTS file in the same directory. +# +# You can contact the author at : +# - zstd homepage : http://www.zstd.net/ +# ################################################################ + +PROJECT(gen_html) +INCLUDE(${CMAKE_SOURCE_DIR}/CMakeModules/GetLibraryVersion.cmake) + +SET(CMAKE_INCLUDE_CURRENT_DIR TRUE) + +# Define programs directory, where sources and header files are located +SET(LIBRARY_DIR ${ZSTD_SOURCE_DIR}/lib) +SET(PROGRAMS_DIR ${ZSTD_SOURCE_DIR}/programs) +SET(GENHTML_DIR ${ZSTD_SOURCE_DIR}/contrib/gen_html) +SET(GENHTML_BINARY ${PROJECT_BINARY_DIR}/gen_html${CMAKE_EXECUTABLE_SUFFIX}) +INCLUDE_DIRECTORIES(${PROGRAMS_DIR} ${LIBRARY_DIR} ${LIBRARY_DIR}/common ${GENHTML_DIR}) + +ADD_EXECUTABLE(gen_html ${GENHTML_DIR}/gen_html.cpp) + +GetLibraryVersion(${LIBRARY_DIR}/zstd.h VMAJOR VMINOR VRELEASE) +SET(LIBVERSION "${VMAJOR}.${VMINOR}.${VRELEASE}") +ADD_CUSTOM_TARGET(zstd_manual.html ALL + ${GENHTML_BINARY} "${LIBVERSION}" "${LIBRARY_DIR}/zstd.h" "${PROJECT_BINARY_DIR}/zstd_manual.html" + DEPENDS gen_html COMMENT "Update zstd manual") From 56ef200fcb71dcbc97ab40e83e959940d742cfdc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Thu, 16 Mar 2017 14:09:39 +0100 Subject: [PATCH 032/305] .gitignore only build directory --- build/.gitignore | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build/.gitignore b/build/.gitignore index f03aac8b3..85bc9287d 100644 --- a/build/.gitignore +++ b/build/.gitignore @@ -17,4 +17,4 @@ VS2013/bin/ VS2015/bin/ # CMake -cmake/ +cmake/build/ From 0184d80e8ee73011f22ded71d5090703e985b386 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Thu, 16 Mar 2017 16:24:33 +0100 Subject: [PATCH 033/305] Refactored AddExtraCompilationFlags - Easier addition of new flags - Removed flags not used by default - Removed implicit PIC flag for all targets --- .../AddExtraCompilationFlags.cmake | 345 +++--------------- 1 file changed, 57 insertions(+), 288 deletions(-) diff --git a/build/cmake/CMakeModules/AddExtraCompilationFlags.cmake b/build/cmake/CMakeModules/AddExtraCompilationFlags.cmake index e480c7ead..db5b48124 100644 --- a/build/cmake/CMakeModules/AddExtraCompilationFlags.cmake +++ b/build/cmake/CMakeModules/AddExtraCompilationFlags.cmake @@ -1,316 +1,85 @@ +include(CheckCXXCompilerFlag) +include(CheckCCompilerFlag) + +function(EnableCompilerFlag _flag _C _CXX) + message("Checking flag ${_flag}") + if (_C) + CHECK_C_COMPILER_FLAG(${_flag} C_FLAG) + if (C_FLAG) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${_flag}" CACHE INTERNAL "C Flags") + endif () + unset(C_FLAG CACHE) + endif () + if (_CXX) + CHECK_CXX_COMPILER_FLAG(${_flag} CXX_FLAG) + if (CXX_FLAG) + set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${_flag}" CACHE INTERNAL "CXX Flags") + endif () + unset(CXX_FLAG CACHE) + endif () +endfunction() + MACRO(ADD_EXTRA_COMPILATION_FLAGS) - include(CheckCXXCompilerFlag) - include(CheckCCompilerFlag) if (CMAKE_COMPILER_IS_GNUCXX OR MINGW) #Not only UNIX but also WIN32 for MinGW - - set(POSITION_INDEPENDENT_CODE_FLAG "-fPIC") - CHECK_C_COMPILER_FLAG(${POSITION_INDEPENDENT_CODE_FLAG} POSITION_INDEPENDENT_CODE_FLAG_ALLOWED) - if (POSITION_INDEPENDENT_CODE_FLAG_ALLOWED) - MESSAGE("Compiler flag ${POSITION_INDEPENDENT_CODE_FLAG} allowed") - set(ACTIVATE_POSITION_INDEPENDENT_CODE_FLAG "ON" CACHE BOOL "activate -fPIC flag") - else () - MESSAGE("Compiler flag ${POSITION_INDEPENDENT_CODE_FLAG} not allowed") - endif (POSITION_INDEPENDENT_CODE_FLAG_ALLOWED) - - set(WARNING_UNDEF "-Wundef") - CHECK_C_COMPILER_FLAG(${WARNING_UNDEF} WARNING_UNDEF_ALLOWED) - if (WARNING_UNDEF_ALLOWED) - MESSAGE("Compiler flag ${WARNING_UNDEF} allowed") - set(ACTIVATE_WARNING_UNDEF "ON" CACHE BOOL "activate -Wundef flag") - else () - MESSAGE("Compiler flag ${WARNING_UNDEF} not allowed") - endif (WARNING_UNDEF_ALLOWED) - - set(WARNING_SHADOW "-Wshadow") - CHECK_C_COMPILER_FLAG(${WARNING_SHADOW} WARNING_SHADOW_ALLOWED) - if (WARNING_SHADOW_ALLOWED) - MESSAGE("Compiler flag ${WARNING_SHADOW} allowed") - set(ACTIVATE_WARNING_SHADOW "ON" CACHE BOOL "activate -Wshadow flag") - else () - MESSAGE("Compiler flag ${WARNING_SHADOW} not allowed") - endif (WARNING_SHADOW_ALLOWED) - - set(WARNING_CAST_ALIGN "-Wcast-align") - CHECK_C_COMPILER_FLAG(${WARNING_CAST_ALIGN} WARNING_CAST_ALIGN_ALLOWED) - if (WARNING_CAST_ALIGN_ALLOWED) - MESSAGE("Compiler flag ${WARNING_CAST_ALIGN} allowed") - set(ACTIVATE_WARNING_CAST_ALIGN "ON" CACHE BOOL "activate -Wcast-align flag") - else () - MESSAGE("Compiler flag ${WARNING_CAST_ALIGN} not allowed") - endif (WARNING_CAST_ALIGN_ALLOWED) - - set(WARNING_CAST_QUAL "-Wcast-qual") - CHECK_C_COMPILER_FLAG(${WARNING_CAST_QUAL} WARNING_CAST_QUAL_ALLOWED) - if (WARNING_CAST_QUAL_ALLOWED) - MESSAGE("Compiler flag ${WARNING_CAST_QUAL} allowed") - set(ACTIVATE_WARNING_CAST_QUAL "ON" CACHE BOOL "activate -Wcast-qual flag") - else () - MESSAGE("Compiler flag ${WARNING_CAST_QUAL} not allowed") - endif (WARNING_CAST_QUAL_ALLOWED) - - set(WARNING_STRICT_PROTOTYPES "-Wstrict-prototypes") - CHECK_C_COMPILER_FLAG(${WARNING_STRICT_PROTOTYPES} WARNING_STRICT_PROTOTYPES_ALLOWED) - if (WARNING_STRICT_PROTOTYPES_ALLOWED) - MESSAGE("Compiler flag ${WARNING_STRICT_PROTOTYPES} allowed") - set(ACTIVATE_WARNING_STRICT_PROTOTYPES "ON" CACHE BOOL "activate -Wstrict-prototypes flag") - else () - MESSAGE("Compiler flag ${WARNING_STRICT_PROTOTYPES} not allowed") - endif (WARNING_STRICT_PROTOTYPES_ALLOWED) - - set(WARNING_ALL "-Wall") - CHECK_C_COMPILER_FLAG(${WARNING_ALL} WARNING_ALL_ALLOWED) - if (WARNING_ALL_ALLOWED) - MESSAGE("Compiler flag ${WARNING_ALL} allowed") - set(ACTIVATE_WARNING_ALL "ON" CACHE BOOL "activate -Wall flag") - else () - MESSAGE("Compiler flag ${WARNING_ALL} not allowed") - endif (WARNING_ALL_ALLOWED) - - set(WARNING_EXTRA "-Wextra") - CHECK_C_COMPILER_FLAG(${WARNING_EXTRA} WARNING_EXTRA_ALLOWED) - if (WARNING_EXTRA_ALLOWED) - MESSAGE("Compiler flag ${WARNING_EXTRA} allowed") - set(ACTIVATE_WARNING_EXTRA "ON" CACHE BOOL "activate -Wextra flag") - else () - MESSAGE("Compiler flag ${WARNING_EXTRA} not allowed") - endif (WARNING_EXTRA_ALLOWED) - - set(WARNING_FLOAT_EQUAL "-Wfloat-equal") - CHECK_C_COMPILER_FLAG(${WARNING_FLOAT_EQUAL} WARNING_FLOAT_EQUAL_ALLOWED) - if (WARNING_FLOAT_EQUAL_ALLOWED) - MESSAGE("Compiler flag ${WARNING_FLOAT_EQUAL} allowed") - set(ACTIVATE_WARNING_FLOAT_EQUAL "OFF" CACHE BOOL "activate -Wfloat-equal flag") - else () - MESSAGE("Compiler flag ${WARNING_FLOAT_EQUAL} not allowed") - endif (WARNING_FLOAT_EQUAL_ALLOWED) - - set(WARNING_SIGN_CONVERSION "-Wsign-conversion") - CHECK_C_COMPILER_FLAG(${WARNING_SIGN_CONVERSION} WARNING_SIGN_CONVERSION_ALLOWED) - if (WARNING_SIGN_CONVERSION_ALLOWED) - MESSAGE("Compiler flag ${WARNING_SIGN_CONVERSION} allowed") - set(ACTIVATE_WARNING_SIGN_CONVERSION "OFF" CACHE BOOL "activate -Wsign-conversion flag") - else () - MESSAGE("Compiler flag ${WARNING_SIGN_CONVERSION} not allowed") - endif (WARNING_SIGN_CONVERSION_ALLOWED) - - if (ACTIVATE_POSITION_INDEPENDENT_CODE_FLAG) - list(APPEND CMAKE_C_FLAGS ${POSITION_INDEPENDENT_CODE_FLAG}) - else () - string(REPLACE ${POSITION_INDEPENDENT_CODE_FLAG} "" CMAKE_C_FLAGS "${POSITION_INDEPENDENT_CODE_FLAG}") - endif (ACTIVATE_POSITION_INDEPENDENT_CODE_FLAG) - - if (ACTIVATE_WARNING_UNDEF) - list(APPEND CMAKE_CXX_FLAGS ${WARNING_UNDEF}) - list(APPEND CMAKE_C_FLAGS ${WARNING_UNDEF}) - else () - string(REPLACE ${WARNING_UNDEF} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${WARNING_UNDEF} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_WARNING_UNDEF) - - if (ACTIVATE_WARNING_SHADOW) - list(APPEND CMAKE_CXX_FLAGS ${WARNING_SHADOW}) - list(APPEND CMAKE_C_FLAGS ${WARNING_SHADOW}) - else () - string(REPLACE ${WARNING_SHADOW} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${WARNING_SHADOW} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_WARNING_SHADOW) - - if (ACTIVATE_WARNING_CAST_QUAL) - list(APPEND CMAKE_CXX_FLAGS ${WARNING_CAST_QUAL}) - list(APPEND CMAKE_C_FLAGS ${WARNING_CAST_QUAL}) - else () - string(REPLACE ${WARNING_CAST_QUAL} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${WARNING_CAST_QUAL} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_WARNING_CAST_QUAL) - - if (ACTIVATE_WARNING_CAST_ALIGN) - list(APPEND CMAKE_CXX_FLAGS ${WARNING_CAST_ALIGN}) - list(APPEND CMAKE_C_FLAGS ${WARNING_CAST_ALIGN}) - else () - string(REPLACE ${WARNING_CAST_ALIGN} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${WARNING_CAST_ALIGN} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_WARNING_CAST_ALIGN) - - if (ACTIVATE_WARNING_STRICT_PROTOTYPES) - list(APPEND CMAKE_C_FLAGS ${WARNING_STRICT_PROTOTYPES}) - else () - string(REPLACE ${WARNING_STRICT_PROTOTYPES} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_WARNING_STRICT_PROTOTYPES) - - if (ACTIVATE_WARNING_ALL) - list(APPEND CMAKE_CXX_FLAGS ${WARNING_ALL}) - list(APPEND CMAKE_C_FLAGS ${WARNING_ALL}) - else () - string(REPLACE ${WARNING_ALL} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${WARNING_ALL} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_WARNING_ALL) - - if (ACTIVATE_WARNING_EXTRA) - list(APPEND CMAKE_CXX_FLAGS ${WARNING_EXTRA}) - list(APPEND CMAKE_C_FLAGS ${WARNING_EXTRA}) - else () - string(REPLACE ${WARNING_EXTRA} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${WARNING_EXTRA} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_WARNING_EXTRA) - - if (ACTIVATE_WARNING_FLOAT_EQUAL) - list(APPEND CMAKE_CXX_FLAGS ${WARNING_FLOAT_EQUAL}) - list(APPEND CMAKE_C_FLAGS ${WARNING_FLOAT_EQUAL}) - else () - string(REPLACE ${WARNING_FLOAT_EQUAL} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${WARNING_FLOAT_EQUAL} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_WARNING_FLOAT_EQUAL) - - if (ACTIVATE_WARNING_SIGN_CONVERSION) - list(APPEND CMAKE_CXX_FLAGS ${WARNING_SIGN_CONVERSION}) - list(APPEND CMAKE_C_FLAGS ${WARNING_SIGN_CONVERSION}) - else () - string(REPLACE ${WARNING_SIGN_CONVERSION} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${WARNING_SIGN_CONVERSION} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_WARNING_SIGN_CONVERSION) - #Set c++11 by default - list(APPEND CMAKE_CXX_FLAGS "-std=c++11") - + EnableCompilerFlag("-std=c++11" false true) #Set c99 by default - list(APPEND CMAKE_C_FLAGS "-std=c99") + EnableCompilerFlag("-std=c99" true false) + EnableCompilerFlag("-Wall" true true) + EnableCompilerFlag("-Wextra" true true) + EnableCompilerFlag("-Wundef" true true) + EnableCompilerFlag("-Wshadow" true true) + EnableCompilerFlag("-Wcast-align" true true) + EnableCompilerFlag("-Wcast-qual" true true) + EnableCompilerFlag("-Wstrict-prototypes" true false) + elseif (MSVC) # Add specific compilation flags for Windows Visual + EnableCompilerFlag("/Wall" true true) - elseif (MSVC) - # Add specific compilation flags for Windows Visual - - set(WARNING_ALL "/Wall") - CHECK_C_COMPILER_FLAG(${WARNING_ALL} WARNING_ALL_ALLOWED) - if (WARNING_ALL_ALLOWED) - MESSAGE("Compiler flag ${WARNING_ALL} allowed") - set(ACTIVATE_WARNING_ALL "OFF" CACHE BOOL "activate /Wall flag") - else () - MESSAGE("Compiler flag ${WARNING_ALL} not allowed") - endif (WARNING_ALL_ALLOWED) - - set(RTC_FLAG "/RTC1") - CHECK_C_COMPILER_FLAG(${RTC_FLAG} RTC_FLAG_ALLOWED) - if (RTC_FLAG_ALLOWED) - MESSAGE("Compiler flag ${RTC_FLAG} allowed") - set(ACTIVATE_RTC_FLAG "ON" CACHE BOOL "activate /RTC1 flag") - else () - MESSAGE("Compiler flag ${RTC_FLAG} not allowed") - endif (RTC_FLAG_ALLOWED) - - set(ZC_FLAG "/Zc:forScope") - CHECK_C_COMPILER_FLAG(${ZC_FLAG} ZC_FLAG_ALLOWED) - if (ZC_FLAG_ALLOWED) - MESSAGE("Compiler flag ${ZC_FLAG} allowed") - set(ACTIVATE_ZC_FLAG "ON" CACHE BOOL "activate /Zc:forScope flag") - else () - MESSAGE("Compiler flag ${ZC_FLAG} not allowed") - endif (ZC_FLAG_ALLOWED) - - set(GD_FLAG "/Gd") - CHECK_C_COMPILER_FLAG(${GD_FLAG} GD_FLAG_ALLOWED) - if (GD_FLAG_ALLOWED) - MESSAGE("Compiler flag ${GD_FLAG} allowed") - set(ACTIVATE_GD_FLAG "ON" CACHE BOOL "activate /Gd flag") - else () - MESSAGE("Compiler flag ${GD_FLAG} not allowed") - endif (GD_FLAG_ALLOWED) - - set(ANALYZE_FLAG "/analyze:stacksize25000") - CHECK_C_COMPILER_FLAG(${ANALYZE_FLAG} ANALYZE_FLAG_ALLOWED) - if (ANALYZE_FLAG_ALLOWED) - MESSAGE("Compiler flag ${ANALYZE_FLAG} allowed") - set(ACTIVATE_ANALYZE_FLAG "ON" CACHE BOOL "activate /ANALYZE flag") - else () - MESSAGE("Compiler flag ${ANALYZE_FLAG} not allowed") - endif (ANALYZE_FLAG_ALLOWED) - - if (ACTIVATE_WARNING_ALL) - list(APPEND CMAKE_CXX_FLAGS ${WARNING_ALL}) - list(APPEND CMAKE_C_FLAGS ${WARNING_ALL}) - else () - string(REPLACE ${WARNING_ALL} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${WARNING_ALL} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_WARNING_ALL) - # Only for DEBUG version - if (ACTIVATE_RTC_FLAG) - list(APPEND CMAKE_CXX_FLAGS_DEBUG ${RTC_FLAG}) - list(APPEND CMAKE_C_FLAGS_DEBUG ${RTC_FLAG}) - else () - string(REPLACE ${RTC_FLAG} "" CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG}") - string(REPLACE ${RTC_FLAG} "" CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG}") - endif (ACTIVATE_RTC_FLAG) - - if (ACTIVATE_ZC_FLAG) - list(APPEND CMAKE_CXX_FLAGS ${ZC_FLAG}) - list(APPEND CMAKE_C_FLAGS ${ZC_FLAG}) - else () - string(REPLACE ${ZC_FLAG} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${ZC_FLAG} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_ZC_FLAG) - - if (ACTIVATE_GD_FLAG) - list(APPEND CMAKE_CXX_FLAGS ${GD_FLAG}) - list(APPEND CMAKE_C_FLAGS ${GD_FLAG}) - else () - string(REPLACE ${GD_FLAG} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${GD_FLAG} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_GD_FLAG) - - if (ACTIVATE_ANALYZE_FLAG) - list(APPEND CMAKE_CXX_FLAGS ${ANALYZE_FLAG}) - list(APPEND CMAKE_C_FLAGS ${ANALYZE_FLAG}) - else () - string(REPLACE ${ANALYZE_FLAG} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${ANALYZE_FLAG} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_ANALYZE_FLAG) - + EnableCompilerFlag("/RTC1" true true) + EnableCompilerFlag("/Zc:forScope" true true) + EnableCompilerFlag("/Gd" true true) + EnableCompilerFlag("/analyze:stacksize25000" true true) + if (MSVC80 OR MSVC90 OR MSVC10 OR MSVC11) # To avoid compiler warning (level 4) C4571, compile with /EHa if you still want # your catch(...) blocks to catch structured exceptions. - list(APPEND CMAKE_CXX_FLAGS "/EHa") + EnableCompilerFlag("/EHa" false true) endif (MSVC80 OR MSVC90 OR MSVC10 OR MSVC11) - set(MULTITHREADED_COMPILATION "/MP") - MESSAGE("Compiler flag ${MULTITHREADED_COMPILATION} allowed") - set(ACTIVATE_MULTITHREADED_COMPILATION "ON" CACHE BOOL "activate /MP flag") - + set(ACTIVATE_MULTITHREADED_COMPILATION "ON" CACHE BOOL "activate multi-threaded compilation (/MP flag)") if (ACTIVATE_MULTITHREADED_COMPILATION) - list(APPEND CMAKE_CXX_FLAGS ${MULTITHREADED_COMPILATION}) - list(APPEND CMAKE_C_FLAGS ${MULTITHREADED_COMPILATION}) - else () - string(REPLACE ${MULTITHREADED_COMPILATION} "" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}") - string(REPLACE ${MULTITHREADED_COMPILATION} "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}") - endif (ACTIVATE_MULTITHREADED_COMPILATION) + EnableCompilerFlag("/MP" true true) + endif () #For exceptions - list(APPEND CMAKE_CXX_FLAGS "/EHsc") - list(APPEND CMAKE_C_FLAGS "/EHsc") + EnableCompilerFlag("/EHsc" true true) # UNICODE SUPPORT - list(APPEND CMAKE_CXX_FLAGS "/D_UNICODE /DUNICODE") - list(APPEND CMAKE_C_FLAGS "/D_UNICODE /DUNICODE") + EnableCompilerFlag("/D_UNICODE" true true) + EnableCompilerFlag("/DUNICODE" true true) endif () # Remove duplicates compilation flags - FOREACH (flag_var CMAKE_C_FLAGS CMAKE_C_FLAGS_DEBUG CMAKE_C_FLAGS_RELEASE - CMAKE_C_FLAGS_MINSIZEREL CMAKE_C_FLAGS_RELWITHDEBINFO - CMAKE_CXX_FLAGS CMAKE_CXX_FLAGS_DEBUG CMAKE_CXX_FLAGS_RELEASE - CMAKE_CXX_FLAGS_MINSIZEREL CMAKE_CXX_FLAGS_RELWITHDEBINFO) - separate_arguments(${flag_var}) - list(REMOVE_DUPLICATES ${flag_var}) - string(REPLACE ";" " " ${flag_var} "${${flag_var}}") - set(${flag_var} "${${flag_var}}" CACHE STRING "common build flags" FORCE) + FOREACH (flag_var CMAKE_C_FLAGS CMAKE_C_FLAGS_DEBUG CMAKE_C_FLAGS_RELEASE + CMAKE_C_FLAGS_MINSIZEREL CMAKE_C_FLAGS_RELWITHDEBINFO + CMAKE_CXX_FLAGS CMAKE_CXX_FLAGS_DEBUG CMAKE_CXX_FLAGS_RELEASE + CMAKE_CXX_FLAGS_MINSIZEREL CMAKE_CXX_FLAGS_RELWITHDEBINFO) + separate_arguments(${flag_var}) + list(REMOVE_DUPLICATES ${flag_var}) + string(REPLACE ";" " " ${flag_var} "${${flag_var}}") + set(${flag_var} "${${flag_var}}" CACHE STRING "common build flags" FORCE) ENDFOREACH (flag_var) if (MSVC) # Replace /MT to /MD flag - # Replace /O2 to /O3 flag + # Replace /O2 to /O3 flag FOREACH (flag_var CMAKE_C_FLAGS CMAKE_C_FLAGS_DEBUG CMAKE_C_FLAGS_RELEASE - CMAKE_C_FLAGS_MINSIZEREL CMAKE_C_FLAGS_RELWITHDEBINFO - CMAKE_CXX_FLAGS CMAKE_CXX_FLAGS_DEBUG CMAKE_CXX_FLAGS_RELEASE - CMAKE_CXX_FLAGS_MINSIZEREL CMAKE_CXX_FLAGS_RELWITHDEBINFO) + CMAKE_C_FLAGS_MINSIZEREL CMAKE_C_FLAGS_RELWITHDEBINFO + CMAKE_CXX_FLAGS CMAKE_CXX_FLAGS_DEBUG CMAKE_CXX_FLAGS_RELEASE + CMAKE_CXX_FLAGS_MINSIZEREL CMAKE_CXX_FLAGS_RELWITHDEBINFO) STRING(REGEX REPLACE "/MT" "/MD" ${flag_var} "${${flag_var}}") - STRING(REGEX REPLACE "/O2" "/Ox" ${flag_var} "${${flag_var}}") + STRING(REGEX REPLACE "/O2" "/Ox" ${flag_var} "${${flag_var}}") ENDFOREACH (flag_var) endif () From a8c0c2af10ad87d67583c95d2e1b36cdaf75b006 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Thu, 16 Mar 2017 22:59:11 +0100 Subject: [PATCH 034/305] Fix cmake install prefix with cmake-3.x --- Makefile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Makefile b/Makefile index 49f29d782..518ea5c2d 100644 --- a/Makefile +++ b/Makefile @@ -275,7 +275,7 @@ cmakebuild: cmake --version $(RM) -r $(BUILDIR)/cmake/build mkdir $(BUILDIR)/cmake/build - cd $(BUILDIR)/cmake/build ; cmake -DPREFIX:STRING=~/install_test_dir $(CMAKE_PARAMS) .. ; $(MAKE) install ; $(MAKE) uninstall + cd $(BUILDIR)/cmake/build ; cmake -DCMAKE_INSTALL_PREFIX:PATH=~/install_test_dir $(CMAKE_PARAMS) .. ; $(MAKE) install ; $(MAKE) uninstall c90build: clean gcc -v From ce11d77e4db6e5166fea22f78b846ad36dbcff05 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Fri, 17 Mar 2017 14:12:28 +0100 Subject: [PATCH 035/305] Use shared library in programs --- build/cmake/contrib/pzstd/CMakeLists.txt | 2 +- build/cmake/programs/CMakeLists.txt | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/build/cmake/contrib/pzstd/CMakeLists.txt b/build/cmake/contrib/pzstd/CMakeLists.txt index f89be8901..f9cffa2a9 100644 --- a/build/cmake/contrib/pzstd/CMakeLists.txt +++ b/build/cmake/contrib/pzstd/CMakeLists.txt @@ -21,7 +21,7 @@ SET(PZSTD_DIR ${ZSTD_SOURCE_DIR}/contrib/pzstd) INCLUDE_DIRECTORIES(${PROGRAMS_DIR} ${LIBRARY_DIR} ${LIBRARY_DIR}/common ${PZSTD_DIR}) ADD_EXECUTABLE(pzstd ${PZSTD_DIR}/main.cpp ${PZSTD_DIR}/Options.cpp ${PZSTD_DIR}/Pzstd.cpp ${PZSTD_DIR}/SkippableFrame.cpp) -TARGET_LINK_LIBRARIES(pzstd libzstd_static pthread) +TARGET_LINK_LIBRARIES(pzstd libzstd_shared pthread) SET_TARGET_PROPERTIES(pzstd PROPERTIES COMPILE_DEFINITIONS "NDEBUG") SET_TARGET_PROPERTIES(pzstd PROPERTIES COMPILE_OPTIONS "-Wno-shadow") diff --git a/build/cmake/programs/CMakeLists.txt b/build/cmake/programs/CMakeLists.txt index 38b4c1e7d..122163a95 100644 --- a/build/cmake/programs/CMakeLists.txt +++ b/build/cmake/programs/CMakeLists.txt @@ -30,17 +30,17 @@ IF (MSVC) ENDIF (MSVC) ADD_EXECUTABLE(zstd ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c ${PROGRAMS_DIR}/bench.c ${PROGRAMS_DIR}/datagen.c ${PROGRAMS_DIR}/dibio.c ${PlatformDependResources}) -TARGET_LINK_LIBRARIES(zstd libzstd_static) +TARGET_LINK_LIBRARIES(zstd libzstd_shared) IF (UNIX) ADD_EXECUTABLE(zstd-frugal ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c) - TARGET_LINK_LIBRARIES(zstd-frugal libzstd_static) + TARGET_LINK_LIBRARIES(zstd-frugal libzstd_shared) SET_TARGET_PROPERTIES(zstd-frugal PROPERTIES COMPILE_DEFINITIONS "ZSTD_NOBENCH;ZSTD_NODICT") ENDIF (UNIX) IF (ZSTD_MULTITHREAD_SUPPORT) ADD_EXECUTABLE(zstdmt ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c ${PROGRAMS_DIR}/bench.c ${PROGRAMS_DIR}/datagen.c ${PROGRAMS_DIR}/dibio.c ${PlatformDependResources}) SET_TARGET_PROPERTIES(zstdmt PROPERTIES COMPILE_DEFINITIONS "ZSTD_MULTITHREAD") - TARGET_LINK_LIBRARIES(zstdmt libzstd_static) + TARGET_LINK_LIBRARIES(zstdmt libzstd_shared) IF (UNIX) TARGET_LINK_LIBRARIES(zstdmt pthread) ENDIF (UNIX) From c2430434df7079e5db418d2b928b7b1b2da48888 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Fri, 17 Mar 2017 14:13:47 +0100 Subject: [PATCH 036/305] Install everything, not only libraries --- build/cmake/contrib/gen_html/CMakeLists.txt | 2 ++ build/cmake/contrib/pzstd/CMakeLists.txt | 1 + build/cmake/programs/CMakeLists.txt | 3 +++ 3 files changed, 6 insertions(+) diff --git a/build/cmake/contrib/gen_html/CMakeLists.txt b/build/cmake/contrib/gen_html/CMakeLists.txt index 14f52c68d..dff8c7a11 100644 --- a/build/cmake/contrib/gen_html/CMakeLists.txt +++ b/build/cmake/contrib/gen_html/CMakeLists.txt @@ -29,3 +29,5 @@ SET(LIBVERSION "${VMAJOR}.${VMINOR}.${VRELEASE}") ADD_CUSTOM_TARGET(zstd_manual.html ALL ${GENHTML_BINARY} "${LIBVERSION}" "${LIBRARY_DIR}/zstd.h" "${PROJECT_BINARY_DIR}/zstd_manual.html" DEPENDS gen_html COMMENT "Update zstd manual") + +INSTALL(FILES "${PROJECT_BINARY_DIR}/zstd_manual.html" DESTINATION "share/doc") diff --git a/build/cmake/contrib/pzstd/CMakeLists.txt b/build/cmake/contrib/pzstd/CMakeLists.txt index f9cffa2a9..f581c6aa4 100644 --- a/build/cmake/contrib/pzstd/CMakeLists.txt +++ b/build/cmake/contrib/pzstd/CMakeLists.txt @@ -25,3 +25,4 @@ TARGET_LINK_LIBRARIES(pzstd libzstd_shared pthread) SET_TARGET_PROPERTIES(pzstd PROPERTIES COMPILE_DEFINITIONS "NDEBUG") SET_TARGET_PROPERTIES(pzstd PROPERTIES COMPILE_OPTIONS "-Wno-shadow") +INSTALL(TARGETS pzstd RUNTIME DESTINATION "bin") diff --git a/build/cmake/programs/CMakeLists.txt b/build/cmake/programs/CMakeLists.txt index 122163a95..3f8c3b0f2 100644 --- a/build/cmake/programs/CMakeLists.txt +++ b/build/cmake/programs/CMakeLists.txt @@ -31,6 +31,8 @@ ENDIF (MSVC) ADD_EXECUTABLE(zstd ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c ${PROGRAMS_DIR}/bench.c ${PROGRAMS_DIR}/datagen.c ${PROGRAMS_DIR}/dibio.c ${PlatformDependResources}) TARGET_LINK_LIBRARIES(zstd libzstd_shared) +INSTALL(TARGETS zstd RUNTIME DESTINATION "bin") + IF (UNIX) ADD_EXECUTABLE(zstd-frugal ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c) TARGET_LINK_LIBRARIES(zstd-frugal libzstd_shared) @@ -44,4 +46,5 @@ IF (ZSTD_MULTITHREAD_SUPPORT) IF (UNIX) TARGET_LINK_LIBRARIES(zstdmt pthread) ENDIF (UNIX) + INSTALL(TARGETS zstdmt RUNTIME DESTINATION "bin") ENDIF (ZSTD_MULTITHREAD_SUPPORT) From 0d3daffa2ffaab6070fa509b408e74fe216ebecd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Fri, 17 Mar 2017 14:14:46 +0100 Subject: [PATCH 037/305] Much easier way of installing libraries with correct names --- build/cmake/lib/CMakeLists.txt | 56 ++++++---------------------------- 1 file changed, 9 insertions(+), 47 deletions(-) diff --git a/build/cmake/lib/CMakeLists.txt b/build/cmake/lib/CMakeLists.txt index 65b00b9ed..874111287 100644 --- a/build/cmake/lib/CMakeLists.txt +++ b/build/cmake/lib/CMakeLists.txt @@ -100,71 +100,33 @@ ENDIF (MSVC) # Define library base name IF (MSVC) - SET(LIBRARY_BASE_NAME zstdlib) IF (CMAKE_SIZEOF_VOID_P MATCHES "8") - SET(LIBRARY_ARCH_SUFFIX "_x64") + SET(LIBRARY_BASE_NAME "zstdlib_x64") ELSE () - SET(LIBRARY_ARCH_SUFFIX "_x86") + SET(LIBRARY_BASE_NAME "zstdlib_x86") ENDIF (CMAKE_SIZEOF_VOID_P MATCHES "8") ELSE () - SET(LIBRARY_BASE_NAME libzstd) - - SET(LIBRARY_ARCH_SUFFIX "") + SET(LIBRARY_BASE_NAME zstd) ENDIF (MSVC) # Define static and shared library names -SET(STATIC_LIBRARY_OUTPUT_NAME ${LIBRARY_BASE_NAME}${LIBRARY_ARCH_SUFFIX} CACHE STRING "Static library output name") -SET(SHARED_LIBRARY_OUTPUT_NAME ${LIBRARY_BASE_NAME}.${LIBVER_MAJOR}.${LIBVER_MINOR}.${LIBVER_RELEASE}${LIBRARY_ARCH_SUFFIX} CACHE STRING "Shared library output name") - SET_TARGET_PROPERTIES( libzstd_static PROPERTIES - PREFIX "" - OUTPUT_NAME ${STATIC_LIBRARY_OUTPUT_NAME}) + OUTPUT_NAME ${LIBRARY_BASE_NAME}) SET_TARGET_PROPERTIES( libzstd_shared PROPERTIES - PREFIX "" - OUTPUT_NAME ${SHARED_LIBRARY_OUTPUT_NAME}) + OUTPUT_NAME ${LIBRARY_BASE_NAME} + SOVERSION ${LIBVER_MAJOR}.${LIBVER_MINOR}.${LIBVER_RELEASE}) IF (UNIX) - IF ("${PREFIX}" STREQUAL "") - SET(PREFIX /usr/local) - ENDIF() - MESSAGE("the variable PREFIX=${PREFIX}") - SET(INSTALL_LIBRARY_DIR ${PREFIX}/lib) - SET(INSTALL_INCLUDE_DIR ${PREFIX}/include) - # install target - INSTALL(FILES ${LIBRARY_DIR}/zstd.h ${LIBRARY_DIR}/deprecated/zbuff.h ${LIBRARY_DIR}/dictBuilder/zdict.h DESTINATION ${INSTALL_INCLUDE_DIR}) - INSTALL(TARGETS libzstd_static DESTINATION ${INSTALL_LIBRARY_DIR}) - INSTALL(TARGETS libzstd_shared LIBRARY DESTINATION ${INSTALL_LIBRARY_DIR}) - - # Create symlinks and setup this files - SET(SHARED_LIBRARY_LINK ${SHARED_LIBRARY_OUTPUT_NAME}${CMAKE_SHARED_LIBRARY_SUFFIX}) - SET(SHARED_LIBRARY_SYMLINK1 ${LIBRARY_BASE_NAME}${CMAKE_SHARED_LIBRARY_SUFFIX}) - SET(SHARED_LIBRARY_SYMLINK2 ${LIBRARY_BASE_NAME}${CMAKE_SHARED_LIBRARY_SUFFIX}.${LIBVER_MAJOR}) - - SET(SHARED_LIBRARY_LINK_PATH ${CMAKE_CURRENT_BINARY_DIR}/${SHARED_LIBRARY_LINK}) - SET(SHARED_LIBRARY_SYMLINK1_PATH ${CMAKE_CURRENT_BINARY_DIR}/${SHARED_LIBRARY_SYMLINK1}) - SET(SHARED_LIBRARY_SYMLINK2_PATH ${CMAKE_CURRENT_BINARY_DIR}/${SHARED_LIBRARY_SYMLINK2}) - - ADD_CUSTOM_COMMAND(TARGET libzstd_shared POST_BUILD - COMMAND ${CMAKE_COMMAND} -E create_symlink ${SHARED_LIBRARY_LINK} ${SHARED_LIBRARY_SYMLINK1} - DEPENDS ${SHARED_LIBRARY_LINK_PATH} - COMMENT "Generating symbolic link ${SHARED_LIBRARY_LINK} -> ${SHARED_LIBRARY_SYMLINK1}") - - ADD_CUSTOM_COMMAND(TARGET libzstd_shared POST_BUILD - COMMAND ${CMAKE_COMMAND} -E create_symlink ${SHARED_LIBRARY_LINK} ${SHARED_LIBRARY_SYMLINK2} - DEPENDS ${SHARED_LIBRARY_LINK_PATH} - COMMENT "Generating symbolic link ${SHARED_LIBRARY_LINK} -> ${SHARED_LIBRARY_SYMLINK2}") - - SET_DIRECTORY_PROPERTIES(PROPERTIES ADDITIONAL_MAKE_CLEAN_FILES "${SHARED_LIBRARY_SYMLINK1};${SHARED_LIBRARY_SYMLINK2}") - - INSTALL(FILES ${SHARED_LIBRARY_SYMLINK1_PATH} DESTINATION ${INSTALL_LIBRARY_DIR}) - INSTALL(FILES ${SHARED_LIBRARY_SYMLINK2_PATH} DESTINATION ${INSTALL_LIBRARY_DIR}) + INSTALL(FILES ${LIBRARY_DIR}/zstd.h ${LIBRARY_DIR}/deprecated/zbuff.h ${LIBRARY_DIR}/dictBuilder/zdict.h DESTINATION "include") + INSTALL(TARGETS libzstd_static ARCHIVE DESTINATION "lib") + INSTALL(TARGETS libzstd_shared LIBRARY DESTINATION "lib") # uninstall target CONFIGURE_FILE( From b98b6fcf41bda6089ca0df821eaf90c4d272642a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Fri, 17 Mar 2017 14:27:01 +0100 Subject: [PATCH 038/305] Check for pthreads library --- build/cmake/contrib/pzstd/CMakeLists.txt | 9 ++++++++- build/cmake/programs/CMakeLists.txt | 12 ++++++++---- 2 files changed, 16 insertions(+), 5 deletions(-) diff --git a/build/cmake/contrib/pzstd/CMakeLists.txt b/build/cmake/contrib/pzstd/CMakeLists.txt index f581c6aa4..b4fbe1689 100644 --- a/build/cmake/contrib/pzstd/CMakeLists.txt +++ b/build/cmake/contrib/pzstd/CMakeLists.txt @@ -21,8 +21,15 @@ SET(PZSTD_DIR ${ZSTD_SOURCE_DIR}/contrib/pzstd) INCLUDE_DIRECTORIES(${PROGRAMS_DIR} ${LIBRARY_DIR} ${LIBRARY_DIR}/common ${PZSTD_DIR}) ADD_EXECUTABLE(pzstd ${PZSTD_DIR}/main.cpp ${PZSTD_DIR}/Options.cpp ${PZSTD_DIR}/Pzstd.cpp ${PZSTD_DIR}/SkippableFrame.cpp) -TARGET_LINK_LIBRARIES(pzstd libzstd_shared pthread) SET_TARGET_PROPERTIES(pzstd PROPERTIES COMPILE_DEFINITIONS "NDEBUG") SET_TARGET_PROPERTIES(pzstd PROPERTIES COMPILE_OPTIONS "-Wno-shadow") +SET(THREADS_PREFER_PTHREAD_FLAG ON) +FIND_PACKAGE(Threads REQUIRED) +IF (CMAKE_USE_PTHREADS_INIT) + TARGET_LINK_LIBRARIES(pzstd libzstd_shared ${CMAKE_THREAD_LIBS_INIT}) +ELSE() + MESSAGE(SEND_ERROR "ZSTD currently does not support thread libraries other than pthreads") +ENDIF() + INSTALL(TARGETS pzstd RUNTIME DESTINATION "bin") diff --git a/build/cmake/programs/CMakeLists.txt b/build/cmake/programs/CMakeLists.txt index 3f8c3b0f2..bc20a265f 100644 --- a/build/cmake/programs/CMakeLists.txt +++ b/build/cmake/programs/CMakeLists.txt @@ -43,8 +43,12 @@ IF (ZSTD_MULTITHREAD_SUPPORT) ADD_EXECUTABLE(zstdmt ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c ${PROGRAMS_DIR}/bench.c ${PROGRAMS_DIR}/datagen.c ${PROGRAMS_DIR}/dibio.c ${PlatformDependResources}) SET_TARGET_PROPERTIES(zstdmt PROPERTIES COMPILE_DEFINITIONS "ZSTD_MULTITHREAD") TARGET_LINK_LIBRARIES(zstdmt libzstd_shared) - IF (UNIX) - TARGET_LINK_LIBRARIES(zstdmt pthread) - ENDIF (UNIX) - INSTALL(TARGETS zstdmt RUNTIME DESTINATION "bin") + + SET(THREADS_PREFER_PTHREAD_FLAG ON) + FIND_PACKAGE(Threads REQUIRED) + IF (CMAKE_USE_PTHREADS_INIT) + TARGET_LINK_LIBRARIES(zstdmt ${CMAKE_THREAD_LIBS_INIT}) + ELSE() + MESSAGE(SEND_ERROR "ZSTD currently does not support thread libraries other than pthreads") + ENDIF() ENDIF (ZSTD_MULTITHREAD_SUPPORT) From 8d562311c1869f5b081b8fe34cb8b142ad7b73c7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Fri, 17 Mar 2017 14:42:44 +0100 Subject: [PATCH 039/305] Test new cmake stuff --- Makefile | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Makefile b/Makefile index 518ea5c2d..985bff587 100644 --- a/Makefile +++ b/Makefile @@ -106,6 +106,7 @@ clean: #------------------------------------------------------------------------------ ifneq (,$(filter $(shell uname),Linux Darwin GNU/kFreeBSD GNU FreeBSD DragonFly NetBSD)) HOST_OS = POSIX +CMAKE_PARAMS = -DZSTD_BUILD_CONTRIB:BOOL=ON .PHONY: install uninstall travis-install clangtest gpptest armtest usan asan uasan install: @@ -263,7 +264,7 @@ endif ifneq (,$(filter MSYS%,$(shell uname))) HOST_OS = MSYS -CMAKE_PARAMS = -G"MSYS Makefiles" +CMAKE_PARAMS = -G"MSYS Makefiles" -DZSTD_MULTITHREAD_SUPPORT:BOOL=OFF endif From f0076adccb5c6750283aba478d83efb52741a475 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Fri, 17 Mar 2017 15:36:37 +0100 Subject: [PATCH 040/305] Add_extra_compilation_flags macro needs to run before adding sources Also adding comments in the top-level CMakeLists.txt --- build/cmake/CMakeLists.txt | 17 +++++++++++++---- 1 file changed, 13 insertions(+), 4 deletions(-) diff --git a/build/cmake/CMakeLists.txt b/build/cmake/CMakeLists.txt index d0f6daade..7fa8bbc79 100644 --- a/build/cmake/CMakeLists.txt +++ b/build/cmake/CMakeLists.txt @@ -11,6 +11,15 @@ PROJECT(zstd) CMAKE_MINIMUM_REQUIRED(VERSION 2.8.7) SET(ZSTD_SOURCE_DIR "${CMAKE_SOURCE_DIR}/../..") +#----------------------------------------------------------------------------- +# Add extra compilation flags +#----------------------------------------------------------------------------- +INCLUDE(CMakeModules/AddExtraCompilationFlags.cmake) +ADD_EXTRA_COMPILATION_FLAGS() + +#----------------------------------------------------------------------------- +# Options +#----------------------------------------------------------------------------- OPTION(ZSTD_LEGACY_SUPPORT "LEGACY SUPPORT" OFF) OPTION(ZSTD_MULTITHREAD_SUPPORT "MULTITHREADING SUPPORT" ON) OPTION(ZSTD_BUILD_CONTRIB "BUILD CONTRIB" OFF) @@ -23,6 +32,9 @@ ELSE (ZSTD_LEGACY_SUPPORT) ADD_DEFINITIONS(-DZSTD_LEGACY_SUPPORT=0) ENDIF (ZSTD_LEGACY_SUPPORT) +#----------------------------------------------------------------------------- +# Add source directories +#----------------------------------------------------------------------------- ADD_SUBDIRECTORY(lib) ADD_SUBDIRECTORY(programs) ADD_SUBDIRECTORY(tests) @@ -31,11 +43,8 @@ IF (ZSTD_BUILD_CONTRIB) ENDIF (ZSTD_BUILD_CONTRIB) #----------------------------------------------------------------------------- -# Add extra compilation flags +# Add clean-all target #----------------------------------------------------------------------------- -INCLUDE(CMakeModules/AddExtraCompilationFlags.cmake) -ADD_EXTRA_COMPILATION_FLAGS() - ADD_CUSTOM_TARGET(clean-all COMMAND ${CMAKE_BUILD_TOOL} clean COMMAND rm -rf ${CMAKE_BINARY_DIR}/ From 16f771dfb152e09801adeccb857ea6e4c666feed Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Mon, 20 Mar 2017 15:47:02 +0100 Subject: [PATCH 041/305] Install forgotten zstdmt tool and zstd.1 man page --- build/cmake/programs/CMakeLists.txt | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/build/cmake/programs/CMakeLists.txt b/build/cmake/programs/CMakeLists.txt index bc20a265f..b3cd1d129 100644 --- a/build/cmake/programs/CMakeLists.txt +++ b/build/cmake/programs/CMakeLists.txt @@ -34,6 +34,8 @@ TARGET_LINK_LIBRARIES(zstd libzstd_shared) INSTALL(TARGETS zstd RUNTIME DESTINATION "bin") IF (UNIX) + INSTALL(FILES ${PROGRAMS_DIR}/zstd.1 DESTINATION "share/man/man1") + ADD_EXECUTABLE(zstd-frugal ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c) TARGET_LINK_LIBRARIES(zstd-frugal libzstd_shared) SET_TARGET_PROPERTIES(zstd-frugal PROPERTIES COMPILE_DEFINITIONS "ZSTD_NOBENCH;ZSTD_NODICT") @@ -51,4 +53,6 @@ IF (ZSTD_MULTITHREAD_SUPPORT) ELSE() MESSAGE(SEND_ERROR "ZSTD currently does not support thread libraries other than pthreads") ENDIF() + + INSTALL(TARGETS zstdmt RUNTIME DESTINATION "bin") ENDIF (ZSTD_MULTITHREAD_SUPPORT) From 01d7ea27e9430248522b848d43fbf38c81413608 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Mon, 20 Mar 2017 15:47:28 +0100 Subject: [PATCH 042/305] Added options for building programs, static library and tests --- build/cmake/CMakeLists.txt | 17 +++++++++++++++-- build/cmake/lib/CMakeLists.txt | 25 +++++++++++++++++-------- 2 files changed, 32 insertions(+), 10 deletions(-) diff --git a/build/cmake/CMakeLists.txt b/build/cmake/CMakeLists.txt index 7fa8bbc79..e338e6cc1 100644 --- a/build/cmake/CMakeLists.txt +++ b/build/cmake/CMakeLists.txt @@ -22,7 +22,9 @@ ADD_EXTRA_COMPILATION_FLAGS() #----------------------------------------------------------------------------- OPTION(ZSTD_LEGACY_SUPPORT "LEGACY SUPPORT" OFF) OPTION(ZSTD_MULTITHREAD_SUPPORT "MULTITHREADING SUPPORT" ON) +OPTION(ZSTD_BUILD_PROGRAMS "BUILD PROGRAMS" ON) OPTION(ZSTD_BUILD_CONTRIB "BUILD CONTRIB" OFF) +OPTION(ZSTD_BUILD_TESTS "BUILD TESTS" OFF) IF (ZSTD_LEGACY_SUPPORT) MESSAGE(STATUS "ZSTD_LEGACY_SUPPORT defined!") @@ -36,8 +38,19 @@ ENDIF (ZSTD_LEGACY_SUPPORT) # Add source directories #----------------------------------------------------------------------------- ADD_SUBDIRECTORY(lib) -ADD_SUBDIRECTORY(programs) -ADD_SUBDIRECTORY(tests) + +IF (ZSTD_BUILD_PROGRAMS) + ADD_SUBDIRECTORY(programs) +ENDIF (ZSTD_BUILD_PROGRAMS) + +IF (ZSTD_BUILD_TESTS) + IF (NOT ZSTD_BUILD_STATIC) + MESSAGE(SEND_ERROR "You need to build static library to build tests") + ENDIF (NOT ZSTD_BUILD_STATIC) + + ADD_SUBDIRECTORY(tests) +ENDIF (ZSTD_BUILD_TESTS) + IF (ZSTD_BUILD_CONTRIB) ADD_SUBDIRECTORY(contrib) ENDIF (ZSTD_BUILD_CONTRIB) diff --git a/build/cmake/lib/CMakeLists.txt b/build/cmake/lib/CMakeLists.txt index 874111287..7a345bf56 100644 --- a/build/cmake/lib/CMakeLists.txt +++ b/build/cmake/lib/CMakeLists.txt @@ -14,6 +14,7 @@ PROJECT(libzstd) INCLUDE(${CMAKE_SOURCE_DIR}/CMakeModules/GetLibraryVersion.cmake) SET(CMAKE_INCLUDE_CURRENT_DIR TRUE) +OPTION(ZSTD_BUILD_STATIC "BUILD STATIC LIBRARIES" OFF) # Define library directory, where sources and header files are located SET(LIBRARY_DIR ${ZSTD_SOURCE_DIR}/lib) @@ -89,13 +90,17 @@ IF (MSVC) ENDIF (MSVC) # Split project to static and shared libraries build -ADD_LIBRARY(libzstd_static STATIC ${Sources} ${Headers}) ADD_LIBRARY(libzstd_shared SHARED ${Sources} ${Headers} ${PlatformDependResources}) +IF (ZSTD_BUILD_STATIC) + ADD_LIBRARY(libzstd_static STATIC ${Sources} ${Headers}) +ENDIF (ZSTD_BUILD_STATIC) # Add specific compile definitions for MSVC project IF (MSVC) - SET_TARGET_PROPERTIES(libzstd_static PROPERTIES COMPILE_DEFINITIONS "ZSTD_HEAPMODE=0;_CRT_SECURE_NO_WARNINGS") SET_TARGET_PROPERTIES(libzstd_shared PROPERTIES COMPILE_DEFINITIONS "ZSTD_DLL_EXPORT=1;ZSTD_HEAPMODE=0;_CONSOLE;_CRT_SECURE_NO_WARNINGS") + IF (ZSTD_BUILD_STATIC) + SET_TARGET_PROPERTIES(libzstd_static PROPERTIES COMPILE_DEFINITIONS "ZSTD_HEAPMODE=0;_CRT_SECURE_NO_WARNINGS") + ENDIF (ZSTD_BUILD_STATIC) ENDIF (MSVC) # Define library base name @@ -111,22 +116,26 @@ ELSE () ENDIF (MSVC) # Define static and shared library names -SET_TARGET_PROPERTIES( - libzstd_static - PROPERTIES - OUTPUT_NAME ${LIBRARY_BASE_NAME}) - SET_TARGET_PROPERTIES( libzstd_shared PROPERTIES OUTPUT_NAME ${LIBRARY_BASE_NAME} SOVERSION ${LIBVER_MAJOR}.${LIBVER_MINOR}.${LIBVER_RELEASE}) +IF (ZSTD_BUILD_STATIC) + SET_TARGET_PROPERTIES( + libzstd_static + PROPERTIES + OUTPUT_NAME ${LIBRARY_BASE_NAME}) +ENDIF (ZSTD_BUILD_STATIC) + IF (UNIX) # install target INSTALL(FILES ${LIBRARY_DIR}/zstd.h ${LIBRARY_DIR}/deprecated/zbuff.h ${LIBRARY_DIR}/dictBuilder/zdict.h DESTINATION "include") - INSTALL(TARGETS libzstd_static ARCHIVE DESTINATION "lib") INSTALL(TARGETS libzstd_shared LIBRARY DESTINATION "lib") + IF (ZSTD_BUILD_STATIC) + INSTALL(TARGETS libzstd_static ARCHIVE DESTINATION "lib") + ENDIF (ZSTD_BUILD_STATIC) # uninstall target CONFIGURE_FILE( From 0d09c778a525530f21eaaa3dbf2c85ca12db8874 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Mon, 20 Mar 2017 15:57:08 +0100 Subject: [PATCH 043/305] Add build of new cmake branches to Makefile --- Makefile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Makefile b/Makefile index 985bff587..81cf51670 100644 --- a/Makefile +++ b/Makefile @@ -106,7 +106,7 @@ clean: #------------------------------------------------------------------------------ ifneq (,$(filter $(shell uname),Linux Darwin GNU/kFreeBSD GNU FreeBSD DragonFly NetBSD)) HOST_OS = POSIX -CMAKE_PARAMS = -DZSTD_BUILD_CONTRIB:BOOL=ON +CMAKE_PARAMS = -DZSTD_BUILD_CONTRIB:BOOL=ON -DZSTD_BUILD_STATIC:BOOL=ON -DZSTD_BUILD_TESTS:BOOL=ON .PHONY: install uninstall travis-install clangtest gpptest armtest usan asan uasan install: @@ -264,7 +264,7 @@ endif ifneq (,$(filter MSYS%,$(shell uname))) HOST_OS = MSYS -CMAKE_PARAMS = -G"MSYS Makefiles" -DZSTD_MULTITHREAD_SUPPORT:BOOL=OFF +CMAKE_PARAMS = -G"MSYS Makefiles" -DZSTD_MULTITHREAD_SUPPORT:BOOL=OFF -DZSTD_BUILD_STATIC:BOOL=ON -DZSTD_BUILD_TESTS:BOOL=ON endif From a2286a333c3d1f40c2da1e7e5ad9fc187bdc693f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Mon, 20 Mar 2017 22:58:02 +0100 Subject: [PATCH 044/305] Install zstdcat and unzstd symlinks by cmake --- build/cmake/programs/CMakeLists.txt | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/build/cmake/programs/CMakeLists.txt b/build/cmake/programs/CMakeLists.txt index b3cd1d129..a481a8e1d 100644 --- a/build/cmake/programs/CMakeLists.txt +++ b/build/cmake/programs/CMakeLists.txt @@ -31,10 +31,19 @@ ENDIF (MSVC) ADD_EXECUTABLE(zstd ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c ${PROGRAMS_DIR}/bench.c ${PROGRAMS_DIR}/datagen.c ${PROGRAMS_DIR}/dibio.c ${PlatformDependResources}) TARGET_LINK_LIBRARIES(zstd libzstd_shared) +ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd zstdcat COMMENT "Creating zstdcat symlink") +ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd unzstd COMMENT "Creating unzstd symlink") INSTALL(TARGETS zstd RUNTIME DESTINATION "bin") +INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/zstdcat DESTINATION "bin") +INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/unzstd DESTINATION "bin") IF (UNIX) - INSTALL(FILES ${PROGRAMS_DIR}/zstd.1 DESTINATION "share/man/man1") + FILE(COPY ${PROGRAMS_DIR}/zstd.1 DESTINATION .) + ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd.1 zstdcat.1 COMMENT "Creating zstdcat.1 symlink") + ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd.1 unzstd.1 COMMENT "Creating unzstd.1 symlink") + INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/zstd.1 DESTINATION "share/man/man1") + INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/zstdcat.1 DESTINATION "share/man/man1") + INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/unzstd.1 DESTINATION "share/man/man1") ADD_EXECUTABLE(zstd-frugal ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c) TARGET_LINK_LIBRARIES(zstd-frugal libzstd_shared) From 5b3a9cbed2c8459b0127a8453aacb9b485d4f7b7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= Date: Tue, 21 Mar 2017 13:00:05 +0100 Subject: [PATCH 045/305] Remove multithreading support by default on non UNIX platforms --- build/cmake/CMakeLists.txt | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/build/cmake/CMakeLists.txt b/build/cmake/CMakeLists.txt index e338e6cc1..9ce52ed21 100644 --- a/build/cmake/CMakeLists.txt +++ b/build/cmake/CMakeLists.txt @@ -21,7 +21,11 @@ ADD_EXTRA_COMPILATION_FLAGS() # Options #----------------------------------------------------------------------------- OPTION(ZSTD_LEGACY_SUPPORT "LEGACY SUPPORT" OFF) -OPTION(ZSTD_MULTITHREAD_SUPPORT "MULTITHREADING SUPPORT" ON) +IF (UNIX) + OPTION(ZSTD_MULTITHREAD_SUPPORT "MULTITHREADING SUPPORT" ON) +ELSE (UNIX) + OPTION(ZSTD_MULTITHREAD_SUPPORT "MULTITHREADING SUPPORT" OFF) +ENDIF (UNIX) OPTION(ZSTD_BUILD_PROGRAMS "BUILD PROGRAMS" ON) OPTION(ZSTD_BUILD_CONTRIB "BUILD CONTRIB" OFF) OPTION(ZSTD_BUILD_TESTS "BUILD TESTS" OFF) From ad92b5544859eaf4017b800803fdc1de7b728b68 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 21 Mar 2017 11:19:48 -0700 Subject: [PATCH 046/305] Fix msvc fuzzer test runtime --- appveyor.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/appveyor.yml b/appveyor.yml index 26ad31002..7ffc2c2b0 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -143,7 +143,7 @@ make -C contrib\pzstd clean ) - if [%HOST%]==[visual] if [%CONFIGURATION%]==[Release] ( - SET FUZZERTEST=-T1mn && + SET "FUZZERTEST=-T1mn" && CD tests && SET ZSTD=./zstd.exe && sh -e playTests.sh --test-large-data && From f3dfcdccd10d69519d6bf3304a0e953d8cfe6780 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 21 Mar 2017 12:18:28 -0700 Subject: [PATCH 047/305] bump version number --- NEWS | 4 ++++ lib/zstd.h | 2 +- 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/NEWS b/NEWS index bbd9e1688..fe0c64624 100644 --- a/NEWS +++ b/NEWS @@ -1,3 +1,7 @@ +v1.1.5 +cli : fix : does not output compressed data on console, by Sean Purcell +build: improved cmake script, by @Majlen + v1.1.4 cli : new : can compress in *.gz format, using --format=gzip command, by Przemyslaw Skibinski cli : new : advanced benchmark command --priority=rt diff --git a/lib/zstd.h b/lib/zstd.h index a3237c77e..4531a84bf 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -56,7 +56,7 @@ extern "C" { /*------ Version ------*/ #define ZSTD_VERSION_MAJOR 1 #define ZSTD_VERSION_MINOR 1 -#define ZSTD_VERSION_RELEASE 4 +#define ZSTD_VERSION_RELEASE 5 #define ZSTD_LIB_VERSION ZSTD_VERSION_MAJOR.ZSTD_VERSION_MINOR.ZSTD_VERSION_RELEASE #define ZSTD_QUOTE(str) #str From eaf69b07f0ceeb8d1e8dabd7b4570e2ff3cb182a Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 21 Mar 2017 13:20:59 -0700 Subject: [PATCH 048/305] Zero pointers after freeing --- lib/compress/zstd_compress.c | 4 ++++ lib/decompress/zstd_decompress.c | 4 ++++ 2 files changed, 8 insertions(+) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 450e5970a..1aab3553f 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -2970,9 +2970,13 @@ size_t ZSTD_freeCStream(ZSTD_CStream* zcs) if (zcs==NULL) return 0; /* support free on NULL */ { ZSTD_customMem const cMem = zcs->customMem; ZSTD_freeCCtx(zcs->cctx); + zcs->cctx = NULL; ZSTD_freeCDict(zcs->cdictLocal); + zcs->cdictLocal = NULL; ZSTD_free(zcs->inBuff, cMem); + zcs->inBuff = NULL; ZSTD_free(zcs->outBuff, cMem); + zcs->outBuff = NULL; ZSTD_free(zcs, cMem); return 0; } diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c index 2aaa4a3df..65e6eda70 100644 --- a/lib/decompress/zstd_decompress.c +++ b/lib/decompress/zstd_decompress.c @@ -2209,9 +2209,13 @@ size_t ZSTD_freeDStream(ZSTD_DStream* zds) if (zds==NULL) return 0; /* support free on null */ { ZSTD_customMem const cMem = zds->customMem; ZSTD_freeDCtx(zds->dctx); + zds->dctx = NULL; ZSTD_freeDDict(zds->ddictLocal); + zds->ddictLocal = NULL; ZSTD_free(zds->inBuff, cMem); + zds->inBuff = NULL; ZSTD_free(zds->outBuff, cMem); + zds->outBuff = NULL; #if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) if (zds->legacyContext) ZSTD_freeLegacyStreamContext(zds->legacyContext, zds->previousLegacyVersion); From f7a78609e79ca3c8a0a621f8038ca0f83efb363f Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 21 Mar 2017 13:44:33 -0700 Subject: [PATCH 049/305] CMake: Fix version parsing and allow Unix flags on Clang --- build/cmake/CMakeModules/AddExtraCompilationFlags.cmake | 2 +- build/cmake/CMakeModules/GetLibraryVersion.cmake | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/build/cmake/CMakeModules/AddExtraCompilationFlags.cmake b/build/cmake/CMakeModules/AddExtraCompilationFlags.cmake index db5b48124..e099a01da 100644 --- a/build/cmake/CMakeModules/AddExtraCompilationFlags.cmake +++ b/build/cmake/CMakeModules/AddExtraCompilationFlags.cmake @@ -20,7 +20,7 @@ function(EnableCompilerFlag _flag _C _CXX) endfunction() MACRO(ADD_EXTRA_COMPILATION_FLAGS) - if (CMAKE_COMPILER_IS_GNUCXX OR MINGW) #Not only UNIX but also WIN32 for MinGW + if (CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang" OR MINGW) #Not only UNIX but also WIN32 for MinGW #Set c++11 by default EnableCompilerFlag("-std=c++11" false true) #Set c99 by default diff --git a/build/cmake/CMakeModules/GetLibraryVersion.cmake b/build/cmake/CMakeModules/GetLibraryVersion.cmake index a3e2fd25f..95d84a89f 100644 --- a/build/cmake/CMakeModules/GetLibraryVersion.cmake +++ b/build/cmake/CMakeModules/GetLibraryVersion.cmake @@ -2,7 +2,7 @@ function(GetLibraryVersion _header _major _minor _release) # Read file content FILE(READ ${_header} CONTENT) - string(REGEX MATCHALL ".*define ZSTD_VERSION_MAJOR+.* ([0-9]+).*define ZSTD_VERSION_MINOR+.* ([0-9]+).*define ZSTD_VERSION_RELEASE+.* ([0-9]+)" VERSION_REGEX "${CONTENT}") + string(REGEX MATCH ".*define ZSTD_VERSION_MAJOR *([0-9]+).*define ZSTD_VERSION_MINOR *([0-9]+).*define ZSTD_VERSION_RELEASE *([0-9]+)" VERSION_REGEX "${CONTENT}") SET(${_major} ${CMAKE_MATCH_1} PARENT_SCOPE) SET(${_minor} ${CMAKE_MATCH_2} PARENT_SCOPE) SET(${_release} ${CMAKE_MATCH_3} PARENT_SCOPE) From 4cfed3c526bfb1eb1989035d751d3e999f5e1c66 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 21 Mar 2017 14:21:00 -0700 Subject: [PATCH 050/305] Attempt to fix FUZZERTEST variable set and remove appveyor_old.yml --- appveyor.yml | 2 +- appveyor_old.yml | 187 ----------------------------------------------- 2 files changed, 1 insertion(+), 188 deletions(-) delete mode 100644 appveyor_old.yml diff --git a/appveyor.yml b/appveyor.yml index 7ffc2c2b0..7239e2e13 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -142,8 +142,8 @@ make -C contrib\pzstd check && make -C contrib\pzstd clean ) + - SET "FUZZERTEST=-T1mn" - if [%HOST%]==[visual] if [%CONFIGURATION%]==[Release] ( - SET "FUZZERTEST=-T1mn" && CD tests && SET ZSTD=./zstd.exe && sh -e playTests.sh --test-large-data && diff --git a/appveyor_old.yml b/appveyor_old.yml deleted file mode 100644 index 124850a2f..000000000 --- a/appveyor_old.yml +++ /dev/null @@ -1,187 +0,0 @@ -version: 1.0.{build} -environment: - - - branches: - only: - - dev - - master - matrix: - - COMPILER: "gcc" - PLATFORM: "mingw64" - MAKE_PARAMS: '"make test && make lib && make -C tests test-symbols fullbench-dll fullbench-lib"' - - COMPILER: "gcc" - PLATFORM: "mingw32" - MAKE_PARAMS: '"make -C tests test-zstd test-fullbench test-fuzzer test-invalidDictionaries"' - - COMPILER: "gcc" - PLATFORM: "clang" - MAKE_PARAMS: '"make -C tests zstd fullbench fuzzer paramgrill datagen CC=clang MOREFLAGS="--target=x86_64-w64-mingw32 -Werror -Wconversion -Wno-sign-conversion""' - - COMPILER: "visual" - CONFIGURATION: "Debug" - PLATFORM: "x64" - - COMPILER: "visual" - CONFIGURATION: "Debug" - PLATFORM: "Win32" - - COMPILER: "visual" - CONFIGURATION: "Release" - PLATFORM: "x64" - - COMPILER: "visual" - CONFIGURATION: "Release" - PLATFORM: "Win32" - - - - branches: - only: - - ci - matrix: - - COMPILER: "gcc" - PLATFORM: "mingw64" - MAKE_PARAMS: '"make all"' - -install: - - ECHO Installing %COMPILER% %PLATFORM% %CONFIGURATION% - - MKDIR bin - - if [%COMPILER%]==[gcc] SET PATH_ORIGINAL=%PATH% - - if [%COMPILER%]==[gcc] ( - SET "PATH_MINGW32=c:\MinGW\bin;c:\MinGW\usr\bin" && - SET "PATH_MINGW64=c:\msys64\mingw64\bin;c:\msys64\usr\bin" && - COPY C:\msys64\usr\bin\make.exe C:\MinGW\bin\make.exe && - COPY C:\MinGW\bin\gcc.exe C:\MinGW\bin\cc.exe - ) else ( - IF [%PLATFORM%]==[x64] (SET ADDITIONALPARAM=/p:LibraryPath="C:\Program Files\Microsoft SDKs\Windows\v7.1\lib\x64;c:\Program Files (x86)\Microsoft Visual Studio 10.0\VC\lib\amd64;C:\Program Files (x86)\Microsoft Visual Studio 10.0\;C:\Program Files (x86)\Microsoft Visual Studio 10.0\lib\amd64;") - ) - -build_script: - - ECHO Building %COMPILER% %PLATFORM% %CONFIGURATION% - - if [%PLATFORM%]==[mingw32] SET PATH=%PATH_MINGW32%;%PATH_ORIGINAL% - - if [%PLATFORM%]==[mingw64] SET PATH=%PATH_MINGW64%;%PATH_ORIGINAL% - - if [%PLATFORM%]==[clang] SET PATH=%PATH_MINGW64%;%PATH_ORIGINAL% - - if [%COMPILER%]==[gcc] ( - ECHO *** && - ECHO *** Building %PLATFORM% && - ECHO *** && - make -v && - cc -v && - ECHO %MAKE_PARAMS% && - sh -c %MAKE_PARAMS% - ) - - if [%PLATFORM%]==[clang] COPY tests\fuzzer.exe tests\fuzzer_clang.exe - - if [%COMPILER%]==[gcc] if [%PLATFORM%]==[mingw64] ( - COPY programs\zstd.exe bin\zstd.exe && - appveyor PushArtifact bin\zstd.exe - ) - - if [%COMPILER%]==[gcc] if [%PLATFORM%]==[mingw32] ( - COPY programs\zstd.exe bin\zstd32.exe && - appveyor PushArtifact bin\zstd32.exe - ) - - if [%COMPILER%]==[gcc] make clean - - if [%COMPILER%]==[visual] ( - ECHO *** && - ECHO *** Building Visual Studio 2008 %PLATFORM%\%CONFIGURATION% in %APPVEYOR_BUILD_FOLDER% && - ECHO *** && - msbuild "build\VS2008\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v90 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && - DIR build\VS2008\bin\%PLATFORM%\%CONFIGURATION%\*.exe && - MD5sum build/VS2008/bin/%PLATFORM%/%CONFIGURATION%/*.exe && - COPY build\VS2008\bin\%PLATFORM%\%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2008_%PLATFORM%_%CONFIGURATION%.exe && - ECHO *** && - ECHO *** Building Visual Studio 2010 %PLATFORM%\%CONFIGURATION% && - ECHO *** && - msbuild "build\VS2010\zstd.sln" %ADDITIONALPARAM% /m /verbosity:minimal /property:PlatformToolset=v100 /p:ForceImportBeforeCppTargets=%APPVEYOR_BUILD_FOLDER%\build\VS2010\CompileAsCpp.props /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && - DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && - MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && - msbuild "build\VS2010\zstd.sln" %ADDITIONALPARAM% /m /verbosity:minimal /property:PlatformToolset=v100 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && - DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && - MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && - COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2010_%PLATFORM%_%CONFIGURATION%.exe && - ECHO *** && - ECHO *** Building Visual Studio 2012 %PLATFORM%\%CONFIGURATION% && - ECHO *** && - msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v110 /p:ForceImportBeforeCppTargets=%APPVEYOR_BUILD_FOLDER%\build\VS2010\CompileAsCpp.props /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && - DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && - MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && - msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v110 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && - DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && - MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && - COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2012_%PLATFORM%_%CONFIGURATION%.exe && - ECHO *** && - ECHO *** Building Visual Studio 2013 %PLATFORM%\%CONFIGURATION% && - ECHO *** && - msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v120 /p:ForceImportBeforeCppTargets=%APPVEYOR_BUILD_FOLDER%\build\VS2010\CompileAsCpp.props /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && - DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && - MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && - msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v120 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && - DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && - MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && - COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2013_%PLATFORM%_%CONFIGURATION%.exe && - ECHO *** && - ECHO *** Building Visual Studio 2015 %PLATFORM%\%CONFIGURATION% && - ECHO *** && - msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v140 /p:ForceImportBeforeCppTargets=%APPVEYOR_BUILD_FOLDER%\build\VS2010\CompileAsCpp.props /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && - DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && - MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && - msbuild "build\VS2010\zstd.sln" /m /verbosity:minimal /property:PlatformToolset=v140 /t:Clean,Build /p:Platform=%PLATFORM% /p:Configuration=%CONFIGURATION% /logger:"C:\Program Files\AppVeyor\BuildAgent\Appveyor.MSBuildLogger.dll" && - DIR build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe && - MD5sum build/VS2010/bin/%PLATFORM%_%CONFIGURATION%/*.exe && - COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\fuzzer.exe tests\fuzzer_VS2015_%PLATFORM%_%CONFIGURATION%.exe && - COPY build\VS2010\bin\%PLATFORM%_%CONFIGURATION%\*.exe tests\ - ) - -test_script: - - ECHO Testing %COMPILER% %PLATFORM% %CONFIGURATION% - - SET FUZZERTEST=-T1mn - - if [%COMPILER%]==[gcc] if [%PLATFORM%]==[clang] ( - tests\fuzzer_clang.exe %FUZZERTEST% && - ECHO *** && - ECHO *** Building cmake for %PLATFORM% && - ECHO *** && - mkdir build\cmake\build && - cd build\cmake\build && - cmake -G "Visual Studio 14 2015 Win64" .. && - cd ..\..\.. && - make clean && - ECHO *** && - ECHO *** Building pzstd for %PLATFORM% && - ECHO *** && - make -C contrib\pzstd googletest-mingw64 && - make -C contrib\pzstd pzstd.exe && - make -C contrib\pzstd tests && - make -C contrib\pzstd check && - make -C contrib\pzstd clean - ) - - if [%COMPILER%]==[visual] if [%CONFIGURATION%]==[Release] ( - CD tests && - SET ZSTD=./zstd.exe && - sh -e playTests.sh --test-large-data && - fullbench.exe -i1 && - fullbench.exe -i1 -P0 && - fuzzer_VS2008_%PLATFORM%_Release.exe %FUZZERTEST% && - fuzzer_VS2010_%PLATFORM%_Release.exe %FUZZERTEST% && - fuzzer_VS2012_%PLATFORM%_Release.exe %FUZZERTEST% && - fuzzer_VS2013_%PLATFORM%_Release.exe %FUZZERTEST% && - fuzzer_VS2015_%PLATFORM%_Release.exe %FUZZERTEST% - ) - -artifacts: - - path: bin\zstd.exe - - path: bin\zstd32.exe - -deploy: -- provider: GitHub - auth_token: - secure: LgJo8emYc3sFnlNWkGl4/VYK3nk/8+RagcsqDlAi3xeqNGNutnKjcftjg84uJoT4 - artifact: bin\zstd.exe - force_update: true - on: - branch: autobuild - COMPILER: gcc - PLATFORM: "mingw64" - appveyor_repo_tag: true -- provider: GitHub - auth_token: - secure: LgJo8emYc3sFnlNWkGl4/VYK3nk/8+RagcsqDlAi3xeqNGNutnKjcftjg84uJoT4 - artifact: bin\zstd32.exe - force_update: true - on: - branch: autobuild - COMPILER: gcc - PLATFORM: "mingw32" - appveyor_repo_tag: true From ea14b737955b2620cbc02955126cc1cba8409d8f Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 21 Mar 2017 15:03:23 -0700 Subject: [PATCH 051/305] Educational decoder: Clarify IO_rewind_bits --- doc/educational_decoder/zstd_decompress.c | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/doc/educational_decoder/zstd_decompress.c b/doc/educational_decoder/zstd_decompress.c index ae4eaa81c..5a94fc4aa 100644 --- a/doc/educational_decoder/zstd_decompress.c +++ b/doc/educational_decoder/zstd_decompress.c @@ -1598,11 +1598,16 @@ static inline void IO_rewind_bits(istream_t *const in, int num) { ERROR("Attempting to rewind stream by a negative number of bits"); } + /* move the offset back by `num` bits */ const int new_offset = in->bit_offset - num; - const i64 bytes = (new_offset - 7) / 8; + /* determine the number of whole bytes we have to rewind, rounding up to an + * integer number (e.g. if `new_offset == -5`, `bytes == 1`) */ + const i64 bytes = -(new_offset - 7) / 8; - in->ptr += bytes; - in->len -= bytes; + in->ptr -= bytes; + in->len += bytes; + /* make sure the resulting `bit_offset` is positive, as mod in C does not + * convert numbers from negative to positive (e.g. -22 % 8 == -6) */ in->bit_offset = ((new_offset % 8) + 8) % 8; } From 9f048cb74ac463baf293b30af278620af3e45373 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 21 Mar 2017 17:01:51 -0700 Subject: [PATCH 052/305] Educational decoder: Some more minor clarity fixes --- doc/educational_decoder/zstd_decompress.c | 61 +++++++++++------------ 1 file changed, 30 insertions(+), 31 deletions(-) diff --git a/doc/educational_decoder/zstd_decompress.c b/doc/educational_decoder/zstd_decompress.c index 5a94fc4aa..346cd296a 100644 --- a/doc/educational_decoder/zstd_decompress.c +++ b/doc/educational_decoder/zstd_decompress.c @@ -87,9 +87,9 @@ typedef struct { /// non-byte aligned /// Reads `num` bits from a bitstream, and updates the internal offset -static inline u64 IO_read_bits(istream_t *const in, const int num); +static inline u64 IO_read_bits(istream_t *const in, const int num_bits); /// Rewinds the stream by `num` bits -static inline void IO_rewind_bits(istream_t *const in, const int num); +static inline void IO_rewind_bits(istream_t *const in, const int num_bits); /// If the remaining bits in a byte will be unused, advance to the end of the /// byte static inline void IO_align_stream(istream_t *const in); @@ -124,7 +124,7 @@ static inline istream_t IO_make_sub_istream(istream_t *const in, size_t len); /*** BITSTREAM OPERATIONS *************/ /// Read `num` bits (up to 64) from `src + offset`, where `offset` is in bits -static inline u64 read_bits_LE(const u8 *src, const int num, +static inline u64 read_bits_LE(const u8 *src, const int num_bits, const size_t offset); /// Read bits from the end of a HUF or FSE bitstream. `offset` is in bits, so @@ -136,9 +136,8 @@ static inline u64 STREAM_read_bits(const u8 *src, const int bits, /*** END BITSTREAM OPERATIONS *********/ /*** BIT COUNTING OPERATIONS **********/ -/// Returns `x`, where `2^x` is the largest power of 2 less than or equal to -/// `num`, or `-1` if `num == 0`. -static inline int log2inf(const u64 num); +/// Returns the index of the highest set bit in `num`, or `-1` if `num == 0` +static inline int highest_set_bit(const u64 num); /*** END BIT COUNTING OPERATIONS ******/ /*** HUFFMAN PRIMITIVES ***************/ @@ -1157,7 +1156,7 @@ static void decompress_sequences(frame_context_t *const ctx, istream_t *in, // "After writing the last bit containing information, the compressor writes // a single 1-bit and then fills the byte with 0-7 0 bits of padding." - const int padding = 8 - log2inf(src[len - 1]); + const int padding = 8 - highest_set_bit(src[len - 1]); i64 offset = len * 8 - padding; // "The bitstream starts with initial state values, each using the required @@ -1571,20 +1570,20 @@ static void free_dictionary(dictionary_t *const dict) { /******* IO STREAM OPERATIONS *************************************************/ #define UNALIGNED() ERROR("Attempting to operate on a non-byte aligned stream") /// Reads `num` bits from a bitstream, and updates the internal offset -static inline u64 IO_read_bits(istream_t *const in, const int num) { - if (num > 64 || num <= 0) { +static inline u64 IO_read_bits(istream_t *const in, const int num_bits) { + if (num_bits > 64 || num_bits <= 0) { ERROR("Attempt to read an invalid number of bits"); } - const size_t bytes = (num + in->bit_offset + 7) / 8; - const size_t full_bytes = (num + in->bit_offset) / 8; + const size_t bytes = (num_bits + in->bit_offset + 7) / 8; + const size_t full_bytes = (num_bits + in->bit_offset) / 8; if (bytes > in->len) { INP_SIZE(); } - const u64 result = read_bits_LE(in->ptr, num, in->bit_offset); + const u64 result = read_bits_LE(in->ptr, num_bits, in->bit_offset); - in->bit_offset = (num + in->bit_offset) % 8; + in->bit_offset = (num_bits + in->bit_offset) % 8; in->ptr += full_bytes; in->len -= full_bytes; @@ -1593,21 +1592,21 @@ static inline u64 IO_read_bits(istream_t *const in, const int num) { /// If a non-zero number of bits have been read from the current byte, advance /// the offset to the next byte -static inline void IO_rewind_bits(istream_t *const in, int num) { - if (num < 0) { +static inline void IO_rewind_bits(istream_t *const in, int num_bits) { + if (num_bits < 0) { ERROR("Attempting to rewind stream by a negative number of bits"); } - /* move the offset back by `num` bits */ - const int new_offset = in->bit_offset - num; - /* determine the number of whole bytes we have to rewind, rounding up to an - * integer number (e.g. if `new_offset == -5`, `bytes == 1`) */ + // move the offset back by `num_bits` bits + const int new_offset = in->bit_offset - num_bits; + // determine the number of whole bytes we have to rewind, rounding up to an + // integer number (e.g. if `new_offset == -5`, `bytes == 1`) const i64 bytes = -(new_offset - 7) / 8; in->ptr -= bytes; in->len += bytes; - /* make sure the resulting `bit_offset` is positive, as mod in C does not - * convert numbers from negative to positive (e.g. -22 % 8 == -6) */ + // make sure the resulting `bit_offset` is positive, as mod in C does not + // convert numbers from negative to positive (e.g. -22 % 8 == -6) in->bit_offset = ((new_offset % 8) + 8) % 8; } @@ -1712,9 +1711,9 @@ static inline istream_t IO_make_sub_istream(istream_t *const in, size_t len) { /******* BITSTREAM OPERATIONS *************************************************/ /// Read `num` bits (up to 64) from `src + offset`, where `offset` is in bits -static inline u64 read_bits_LE(const u8 *src, const int num, +static inline u64 read_bits_LE(const u8 *src, const int num_bits, const size_t offset) { - if (num > 64) { + if (num_bits > 64) { ERROR("Attempt to read an invalid number of bits"); } @@ -1724,7 +1723,7 @@ static inline u64 read_bits_LE(const u8 *src, const int num, u64 res = 0; int shift = 0; - int left = num; + int left = num_bits; while (left > 0) { u64 mask = left >= 8 ? 0xff : (((u64)1 << left) - 1); // Dead the next byte, shift it to account for the offset, and then mask @@ -1766,7 +1765,7 @@ static inline u64 STREAM_read_bits(const u8 *const src, const int bits, /******* BIT COUNTING OPERATIONS **********************************************/ /// Returns `x`, where `2^x` is the largest power of 2 less than or equal to /// `num`, or `-1` if `num == 0`. -static inline int log2inf(const u64 num) { +static inline int highest_set_bit(const u64 num) { for (int i = 63; i >= 0; i--) { if (((u64)1 << i) <= num) { return i; @@ -1818,7 +1817,7 @@ static size_t HUF_decompress_1stream(const HUF_dtable *const dtable, // final-bit-flag. Consequently, a last byte of 0 is not possible. And the // final-bit-flag itself is not part of the useful bitstream. Hence, the // last byte contains between 0 and 7 useful bits." - const int padding = 8 - log2inf(src[len - 1]); + const int padding = 8 - highest_set_bit(src[len - 1]); i64 offset = len * 8 - padding; u16 state; @@ -1965,7 +1964,7 @@ static void HUF_init_dtable_usingweights(HUF_dtable *const table, } // Find the first power of 2 larger than the sum - const int max_bits = log2inf(weight_sum) + 1; + const int max_bits = highest_set_bit(weight_sum) + 1; const u64 left_over = ((u64)1 << max_bits) - weight_sum; // If the left over isn't a power of 2, the weights are invalid if (left_over & (left_over - 1)) { @@ -1974,7 +1973,7 @@ static void HUF_init_dtable_usingweights(HUF_dtable *const table, // left_over is used to find the last weight as it's not transmitted // by inverting 2^(weight - 1) we can determine the value of last_weight - const int last_weight = log2inf(left_over) + 1; + const int last_weight = highest_set_bit(left_over) + 1; for (int i = 0; i < num_symbs; i++) { // "Number_of_Bits = Number_of_Bits ? Max_Number_of_Bits + 1 - Weight : 0" @@ -2068,7 +2067,7 @@ static size_t FSE_decompress_interleaved2(const FSE_dtable *const dtable, // final-bit-flag. Consequently, a last byte of 0 is not possible. And the // final-bit-flag itself is not part of the useful bitstream. Hence, the // last byte contains between 0 and 7 useful bits." - const int padding = 8 - log2inf(src[len - 1]); + const int padding = 8 - highest_set_bit(src[len - 1]); i64 offset = len * 8 - padding; u16 state1, state2; @@ -2189,7 +2188,7 @@ static void FSE_init_dtable(FSE_dtable *const dtable, u16 next_state_desc = state_desc[symbol]++; // Fills in the table appropriately, next_state_desc increases by symbol // over time, decreasing number of bits - dtable->num_bits[i] = (u8)(accuracy_log - log2inf(next_state_desc)); + dtable->num_bits[i] = (u8)(accuracy_log - highest_set_bit(next_state_desc)); // Baseline increases until the bit threshold is passed, at which point // it resets to 0 dtable->new_state_base[i] = @@ -2240,7 +2239,7 @@ static void FSE_decode_header(FSE_dtable *const dtable, istream_t *const in, int symb = 0; while (remaining > 0 && symb < FSE_MAX_SYMBS) { // Log of the number of possible values we could read - int bits = log2inf(remaining + 1) + 1; + int bits = highest_set_bit(remaining + 1) + 1; u16 val = IO_read_bits(in, bits); From 35fc66f2571bc125e5d23708fdac259fdb40ac31 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 21 Mar 2017 17:52:42 -0700 Subject: [PATCH 053/305] updated .gitignore for new cmake script --- .gitignore | 11 ++++++----- build/cmake/.gitignore | 3 ++- 2 files changed, 8 insertions(+), 6 deletions(-) diff --git a/.gitignore b/.gitignore index e02119883..5fe9afd41 100644 --- a/.gitignore +++ b/.gitignore @@ -24,6 +24,12 @@ zstdmt tmp* dictionary* +# Build artefacts +projects/ +bin/ +.buckd/ +buck-out/ + # Other files .directory _codelite/ @@ -34,8 +40,3 @@ _zstdbench/ .DS_Store googletest/ *.d - -# Directories -bin/ -.buckd/ -buck-out/ diff --git a/build/cmake/.gitignore b/build/cmake/.gitignore index 98f29c79d..ad4283f99 100644 --- a/build/cmake/.gitignore +++ b/build/cmake/.gitignore @@ -1,6 +1,7 @@ -# cmake producted +# cmake artefacts CMakeCache.txt CMakeFiles Makefile cmake_install.cmake cmake_uninstall.cmake +*.1 From 741e3e831489266c8174cb157ca59b1f9bafd5bc Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Wed, 22 Mar 2017 12:33:08 -0700 Subject: [PATCH 054/305] Add $(FLAGS) when testing for zlib/lzma support. * If zlib/lzma isn't in the usual spot, it won't be used, even if `$CFLAGS` and `$LDFLAGS` add the location it is in. * Update the test code snippets to not trigger any warnings. --- programs/Makefile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index bb91e069a..5c3fd19e7 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -85,7 +85,7 @@ endif # zlib detection NO_ZLIB_MSG := ==> no zlib, building zstd without .gz support VOID = /dev/null -HAVE_ZLIB := $(shell printf '\#include \nint main(){}' | $(CC) -o have_zlib -x c - -lz 2> $(VOID) && rm have_zlib$(EXT) && echo 1 || echo 0) +HAVE_ZLIB := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_zlib -x c - -lz 2> $(VOID) && rm have_zlib$(EXT) && echo 1 || echo 0) ifeq ($(HAVE_ZLIB), 1) ZLIB_MSG := ==> building zstd with .gz compression support ZLIBCPP = -DZSTD_GZCOMPRESS -DZSTD_GZDECOMPRESS @@ -95,7 +95,7 @@ ZLIB_MSG := $(NO_ZLIB_MSG) endif # lzma detection NO_LZMA_MSG := ==> no liblzma, building zstd without .xz/.lzma support -HAVE_LZMA := $(shell printf '\#include \nint main(){}' | $(CC) -o have_lzma -x c - -llzma 2> $(VOID) && rm have_lzma$(EXT) && echo 1 || echo 0) +HAVE_LZMA := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_lzma -x c - -llzma 2> $(VOID) && rm have_lzma$(EXT) && echo 1 || echo 0) ifeq ($(HAVE_LZMA), 1) LZMA_MSG := ==> building zstd with .xz/.lzma compression support LZMACPP = -DZSTD_LZMACOMPRESS -DZSTD_LZMADECOMPRESS From dfc75f74da75d83c7fa57ab9e8cb187a1b04972c Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Wed, 22 Mar 2017 13:24:52 -0700 Subject: [PATCH 055/305] Add $(EXT) to binary to fix zlib detection for some MinGW versions --- programs/Makefile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index 5c3fd19e7..b8e976b21 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -85,7 +85,7 @@ endif # zlib detection NO_ZLIB_MSG := ==> no zlib, building zstd without .gz support VOID = /dev/null -HAVE_ZLIB := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_zlib -x c - -lz 2> $(VOID) && rm have_zlib$(EXT) && echo 1 || echo 0) +HAVE_ZLIB := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_zlib$(EXT) -x c - -lz 2> $(VOID) && rm have_zlib$(EXT) && echo 1 || echo 0) ifeq ($(HAVE_ZLIB), 1) ZLIB_MSG := ==> building zstd with .gz compression support ZLIBCPP = -DZSTD_GZCOMPRESS -DZSTD_GZDECOMPRESS @@ -95,7 +95,7 @@ ZLIB_MSG := $(NO_ZLIB_MSG) endif # lzma detection NO_LZMA_MSG := ==> no liblzma, building zstd without .xz/.lzma support -HAVE_LZMA := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_lzma -x c - -llzma 2> $(VOID) && rm have_lzma$(EXT) && echo 1 || echo 0) +HAVE_LZMA := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_lzma$(EXT) -x c - -llzma 2> $(VOID) && rm have_lzma$(EXT) && echo 1 || echo 0) ifeq ($(HAVE_LZMA), 1) LZMA_MSG := ==> building zstd with .xz/.lzma compression support LZMACPP = -DZSTD_LZMACOMPRESS -DZSTD_LZMADECOMPRESS From 042ba122ae753dd770db44d8730a6a2dc3bf98df Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 23 Mar 2017 11:13:52 -0700 Subject: [PATCH 056/305] Change g_displayLevel to int and fix DISPLAYUPDATE flush --- lib/dictBuilder/cover.c | 2 -- lib/dictBuilder/zdict.c | 2 +- programs/bench.c | 4 +-- programs/dibio.c | 4 +-- programs/fileio.c | 4 +-- programs/zstdcli.c | 48 +++++++++++++++---------------- tests/fuzzer.c | 2 +- tests/zbufftest.c | 2 +- tests/zstreamtest.c | 2 +- zlibWrapper/examples/zwrapbench.c | 4 +-- 10 files changed, 36 insertions(+), 38 deletions(-) diff --git a/lib/dictBuilder/cover.c b/lib/dictBuilder/cover.c index 3a7b9f39f..1db42f95b 100644 --- a/lib/dictBuilder/cover.c +++ b/lib/dictBuilder/cover.c @@ -59,8 +59,6 @@ static int g_displayLevel = 2; if ((clock() - g_time > refreshRate) || (displayLevel >= 4)) { \ g_time = clock(); \ DISPLAY(__VA_ARGS__); \ - if (displayLevel >= 4) \ - fflush(stdout); \ } \ } #define DISPLAYUPDATE(l, ...) LOCALDISPLAYUPDATE(g_displayLevel, l, __VA_ARGS__) diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index 0757dbbbb..c84195791 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -480,7 +480,7 @@ static size_t ZDICT_trainBuffer(dictItem* dictList, U32 dictListSize, # define DISPLAYUPDATE(l, ...) if (notificationLevel>=l) { \ if (ZDICT_clockSpan(displayClock) > refreshRate) \ { displayClock = clock(); DISPLAY(__VA_ARGS__); \ - if (notificationLevel>=4) fflush(stdout); } } + if (notificationLevel>=4) fflush(stderr); } } /* init */ DISPLAYLEVEL(2, "\r%70s\r", ""); /* clean display line */ diff --git a/programs/bench.c b/programs/bench.c index 2dd1cfb0f..2643dd45a 100644 --- a/programs/bench.c +++ b/programs/bench.c @@ -70,12 +70,12 @@ static U32 g_compressibilityDefault = 50; ***************************************/ #define DISPLAY(...) fprintf(stderr, __VA_ARGS__) #define DISPLAYLEVEL(l, ...) if (g_displayLevel>=l) { DISPLAY(__VA_ARGS__); } -static U32 g_displayLevel = 2; /* 0 : no display; 1: errors; 2 : + result + interaction + warnings; 3 : + progression; 4 : + information */ +static int g_displayLevel = 2; /* 0 : no display; 1: errors; 2 : + result + interaction + warnings; 3 : + progression; 4 : + information */ #define DISPLAYUPDATE(l, ...) if (g_displayLevel>=l) { \ if ((clock() - g_time > refreshRate) || (g_displayLevel>=4)) \ { g_time = clock(); DISPLAY(__VA_ARGS__); \ - if (g_displayLevel>=4) fflush(stdout); } } + if (g_displayLevel>=4) fflush(stderr); } } static const clock_t refreshRate = CLOCKS_PER_SEC * 15 / 100; static clock_t g_time = 0; diff --git a/programs/dibio.c b/programs/dibio.c index 5ef202c8a..b7ed280ea 100644 --- a/programs/dibio.c +++ b/programs/dibio.c @@ -53,12 +53,12 @@ static const size_t maxMemory = (sizeof(size_t) == 4) ? (2 GB - 64 MB) : ((size_ ***************************************/ #define DISPLAY(...) fprintf(stderr, __VA_ARGS__) #define DISPLAYLEVEL(l, ...) if (g_displayLevel>=l) { DISPLAY(__VA_ARGS__); } -static unsigned g_displayLevel = 0; /* 0 : no display; 1: errors; 2: default; 4: full information */ +static int g_displayLevel = 0; /* 0 : no display; 1: errors; 2: default; 4: full information */ #define DISPLAYUPDATE(l, ...) if (g_displayLevel>=l) { \ if ((DIB_clockSpan(g_time) > refreshRate) || (g_displayLevel>=4)) \ { g_time = clock(); DISPLAY(__VA_ARGS__); \ - if (g_displayLevel>=4) fflush(stdout); } } + if (g_displayLevel>=4) fflush(stderr); } } static const clock_t refreshRate = CLOCKS_PER_SEC * 2 / 10; static clock_t g_time = 0; diff --git a/programs/fileio.c b/programs/fileio.c index e6481f1fa..572e9c9ac 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -81,13 +81,13 @@ ***************************************/ #define DISPLAY(...) fprintf(stderr, __VA_ARGS__) #define DISPLAYLEVEL(l, ...) { if (g_displayLevel>=l) { DISPLAY(__VA_ARGS__); } } -static U32 g_displayLevel = 2; /* 0 : no display; 1: errors; 2 : + result + interaction + warnings; 3 : + progression; 4 : + information */ +static int g_displayLevel = 2; /* 0 : no display; 1: errors; 2 : + result + interaction + warnings; 3 : + progression; 4 : + information */ void FIO_setNotificationLevel(unsigned level) { g_displayLevel=level; } #define DISPLAYUPDATE(l, ...) { if (g_displayLevel>=l) { \ if ((clock() - g_time > refreshRate) || (g_displayLevel>=4)) \ { g_time = clock(); DISPLAY(__VA_ARGS__); \ - if (g_displayLevel>=4) fflush(stdout); } } } + if (g_displayLevel>=4) fflush(stderr); } } } static const clock_t refreshRate = CLOCKS_PER_SEC * 15 / 100; static clock_t g_time = 0; diff --git a/programs/zstdcli.c b/programs/zstdcli.c index cd53d4c36..29bc76377 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -74,10 +74,10 @@ static U32 g_overlapLog = OVERLAP_LOG_DEFAULT; /*-************************************ * Display Macros **************************************/ -#define DISPLAY(...) fprintf(displayOut, __VA_ARGS__) -#define DISPLAYLEVEL(l, ...) if (displayLevel>=l) { DISPLAY(__VA_ARGS__); } -static FILE* displayOut; -static unsigned displayLevel = DEFAULT_DISPLAY_LEVEL; /* 0 : no display, 1: errors, 2 : + result + interaction + warnings, 3 : + progression, 4 : + information */ +#define DISPLAY(...) fprintf(g_displayOut, __VA_ARGS__) +#define DISPLAYLEVEL(l, ...) { if (g_displayLevel>=l) { DISPLAY(__VA_ARGS__); } } +static int g_displayLevel = DEFAULT_DISPLAY_LEVEL; /* 0 : no display, 1: errors, 2 : + result + interaction + warnings, 3 : + progression, 4 : + information */ +static FILE* g_displayOut; /*-************************************ @@ -167,7 +167,7 @@ static int usage_advanced(const char* programName) static int badusage(const char* programName) { DISPLAYLEVEL(1, "Incorrect parameters\n"); - if (displayLevel >= 2) usage(programName); + if (g_displayLevel >= 2) usage(programName); return 1; } @@ -316,7 +316,7 @@ int main(int argCount, const char* argv[]) (void)memLimit; /* not used when ZSTD_NODECOMPRESS set */ if (filenameTable==NULL) { DISPLAY("zstd: %s \n", strerror(errno)); exit(1); } filenameTable[0] = stdinmark; - displayOut = stderr; + g_displayOut = stderr; /* Pick out program name from path. Don't rely on stdlib because of conflicting behavior */ { size_t pos; for (pos = (int)strlen(programName); pos > 0; pos--) { if (programName[pos] == '/') { pos++; break; } } @@ -325,10 +325,10 @@ int main(int argCount, const char* argv[]) /* preset behaviors */ if (!strcmp(programName, ZSTD_UNZSTD)) operation=zom_decompress; - if (!strcmp(programName, ZSTD_CAT)) { operation=zom_decompress; forceStdout=1; FIO_overwriteMode(); outFileName=stdoutmark; displayLevel=1; } + if (!strcmp(programName, ZSTD_CAT)) { operation=zom_decompress; forceStdout=1; FIO_overwriteMode(); outFileName=stdoutmark; g_displayLevel=1; } if (!strcmp(programName, ZSTD_GZ)) { suffix = GZ_EXTENSION; FIO_setCompressionType(FIO_gzipCompression); FIO_setRemoveSrcFile(1); } /* behave like gzip */ if (!strcmp(programName, ZSTD_GUNZIP)) { operation=zom_decompress; FIO_setRemoveSrcFile(1); } /* behave like gunzip */ - if (!strcmp(programName, ZSTD_GZCAT)) { operation=zom_decompress; forceStdout=1; FIO_overwriteMode(); outFileName=stdoutmark; displayLevel=1; } /* behave like gzcat */ + if (!strcmp(programName, ZSTD_GZCAT)) { operation=zom_decompress; forceStdout=1; FIO_overwriteMode(); outFileName=stdoutmark; g_displayLevel=1; } /* behave like gzcat */ if (!strcmp(programName, ZSTD_LZMA)) { suffix = LZMA_EXTENSION; FIO_setCompressionType(FIO_lzmaCompression); FIO_setRemoveSrcFile(1); } /* behave like lzma */ if (!strcmp(programName, ZSTD_XZ)) { suffix = XZ_EXTENSION; FIO_setCompressionType(FIO_xzCompression); FIO_setRemoveSrcFile(1); } /* behave like xz */ memset(&compressionParams, 0, sizeof(compressionParams)); @@ -344,7 +344,7 @@ int main(int argCount, const char* argv[]) if (!filenameIdx) { filenameIdx=1, filenameTable[0]=stdinmark; outFileName=stdoutmark; - displayLevel-=(displayLevel==2); + g_displayLevel-=(g_displayLevel==2); continue; } } @@ -358,11 +358,11 @@ int main(int argCount, const char* argv[]) if (!strcmp(argument, "--decompress")) { operation=zom_decompress; continue; } if (!strcmp(argument, "--uncompress")) { operation=zom_decompress; continue; } if (!strcmp(argument, "--force")) { FIO_overwriteMode(); continue; } - if (!strcmp(argument, "--version")) { displayOut=stdout; DISPLAY(WELCOME_MESSAGE); CLEAN_RETURN(0); } - if (!strcmp(argument, "--help")) { displayOut=stdout; CLEAN_RETURN(usage_advanced(programName)); } - if (!strcmp(argument, "--verbose")) { displayLevel++; continue; } - if (!strcmp(argument, "--quiet")) { displayLevel--; continue; } - if (!strcmp(argument, "--stdout")) { forceStdout=1; outFileName=stdoutmark; displayLevel-=(displayLevel==2); continue; } + if (!strcmp(argument, "--version")) { g_displayOut=stdout; DISPLAY(WELCOME_MESSAGE); CLEAN_RETURN(0); } + if (!strcmp(argument, "--help")) { g_displayOut=stdout; CLEAN_RETURN(usage_advanced(programName)); } + if (!strcmp(argument, "--verbose")) { g_displayLevel++; continue; } + if (!strcmp(argument, "--quiet")) { g_displayLevel--; continue; } + if (!strcmp(argument, "--stdout")) { forceStdout=1; outFileName=stdoutmark; g_displayLevel-=(g_displayLevel==2); continue; } if (!strcmp(argument, "--ultra")) { ultra=1; continue; } if (!strcmp(argument, "--check")) { FIO_setChecksumFlag(2); continue; } if (!strcmp(argument, "--no-check")) { FIO_setChecksumFlag(0); continue; } @@ -424,9 +424,9 @@ int main(int argCount, const char* argv[]) switch(argument[0]) { /* Display help */ - case 'V': displayOut=stdout; DISPLAY(WELCOME_MESSAGE); CLEAN_RETURN(0); /* Version Only */ + case 'V': g_displayOut=stdout; DISPLAY(WELCOME_MESSAGE); CLEAN_RETURN(0); /* Version Only */ case 'H': - case 'h': displayOut=stdout; CLEAN_RETURN(usage_advanced(programName)); + case 'h': g_displayOut=stdout; CLEAN_RETURN(usage_advanced(programName)); /* Compress */ case 'z': operation=zom_compress; argument++; break; @@ -448,10 +448,10 @@ int main(int argCount, const char* argv[]) case 'f': FIO_overwriteMode(); forceStdout=1; argument++; break; /* Verbose mode */ - case 'v': displayLevel++; argument++; break; + case 'v': g_displayLevel++; argument++; break; /* Quiet mode */ - case 'q': displayLevel--; argument++; break; + case 'q': g_displayLevel--; argument++; break; /* keep source file (default); for gzip/xz compatibility */ case 'k': FIO_setRemoveSrcFile(0); argument++; break; @@ -597,7 +597,7 @@ int main(int argCount, const char* argv[]) /* Check if benchmark is selected */ if (operation==zom_bench) { #ifndef ZSTD_NOBENCH - BMK_setNotificationLevel(displayLevel); + BMK_setNotificationLevel(g_displayLevel); BMK_setBlockSize(blockSize); BMK_setNbThreads(nbThreads); BMK_setNbSeconds(bench_nbSeconds); @@ -613,7 +613,7 @@ int main(int argCount, const char* argv[]) if (cover) { coverParams.nbThreads = nbThreads; coverParams.compressionLevel = dictCLevel; - coverParams.notificationLevel = displayLevel; + coverParams.notificationLevel = g_displayLevel; coverParams.dictID = dictID; DiB_trainFromFiles(outFileName, maxDictSize, filenameTable, filenameIdx, NULL, &coverParams, cover - 1); } else { @@ -621,7 +621,7 @@ int main(int argCount, const char* argv[]) memset(&dictParams, 0, sizeof(dictParams)); dictParams.compressionLevel = dictCLevel; dictParams.selectivityLevel = dictSelect; - dictParams.notificationLevel = displayLevel; + dictParams.notificationLevel = g_displayLevel; dictParams.dictID = dictID; DiB_trainFromFiles(outFileName, maxDictSize, filenameTable, filenameIdx, &dictParams, NULL, 0); } @@ -654,11 +654,11 @@ int main(int argCount, const char* argv[]) #endif /* No status message in pipe mode (stdin - stdout) or multi-files mode */ - if (!strcmp(filenameTable[0], stdinmark) && outFileName && !strcmp(outFileName,stdoutmark) && (displayLevel==2)) displayLevel=1; - if ((filenameIdx>1) & (displayLevel==2)) displayLevel=1; + if (!strcmp(filenameTable[0], stdinmark) && outFileName && !strcmp(outFileName,stdoutmark) && (g_displayLevel==2)) g_displayLevel=1; + if ((filenameIdx>1) & (g_displayLevel==2)) g_displayLevel=1; /* IO Stream/File */ - FIO_setNotificationLevel(displayLevel); + FIO_setNotificationLevel(g_displayLevel); if (operation==zom_compress) { #ifndef ZSTD_NOCOMPRESS FIO_setNbThreads(nbThreads); diff --git a/tests/fuzzer.c b/tests/fuzzer.c index 1d8be17af..111ca8243 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -57,7 +57,7 @@ static U32 g_displayLevel = 2; #define DISPLAYUPDATE(l, ...) if (g_displayLevel>=l) { \ if ((FUZ_clockSpan(g_displayClock) > g_refreshRate) || (g_displayLevel>=4)) \ { g_displayClock = clock(); DISPLAY(__VA_ARGS__); \ - if (g_displayLevel>=4) fflush(stdout); } } + if (g_displayLevel>=4) fflush(stderr); } } static const clock_t g_refreshRate = CLOCKS_PER_SEC / 6; static clock_t g_displayClock = 0; diff --git a/tests/zbufftest.c b/tests/zbufftest.c index 14b739233..601aa808d 100644 --- a/tests/zbufftest.c +++ b/tests/zbufftest.c @@ -60,7 +60,7 @@ static U32 g_displayLevel = 2; #define DISPLAYUPDATE(l, ...) if (g_displayLevel>=l) { \ if ((FUZ_GetClockSpan(g_displayClock) > g_refreshRate) || (g_displayLevel>=4)) \ { g_displayClock = clock(); DISPLAY(__VA_ARGS__); \ - if (g_displayLevel>=4) fflush(stdout); } } + if (g_displayLevel>=4) fflush(stderr); } } static const clock_t g_refreshRate = CLOCKS_PER_SEC * 15 / 100; static clock_t g_displayClock = 0; diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index a03ee9b1f..370859ca9 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -59,7 +59,7 @@ static U32 g_displayLevel = 2; #define DISPLAYUPDATE(l, ...) if (g_displayLevel>=l) { \ if ((FUZ_GetClockSpan(g_displayClock) > g_refreshRate) || (g_displayLevel>=4)) \ { g_displayClock = clock(); DISPLAY(__VA_ARGS__); \ - if (g_displayLevel>=4) fflush(stdout); } } + if (g_displayLevel>=4) fflush(stderr); } } static const clock_t g_refreshRate = CLOCKS_PER_SEC / 6; static clock_t g_displayClock = 0; diff --git a/zlibWrapper/examples/zwrapbench.c b/zlibWrapper/examples/zwrapbench.c index 23c3ca4da..951c315b8 100644 --- a/zlibWrapper/examples/zwrapbench.c +++ b/zlibWrapper/examples/zwrapbench.c @@ -73,13 +73,13 @@ static U32 g_compressibilityDefault = 50; #define DEFAULT_DISPLAY_LEVEL 2 #define DISPLAY(...) fprintf(displayOut, __VA_ARGS__) #define DISPLAYLEVEL(l, ...) if (g_displayLevel>=l) { DISPLAY(__VA_ARGS__); } -static U32 g_displayLevel = DEFAULT_DISPLAY_LEVEL; /* 0 : no display; 1: errors; 2 : + result + interaction + warnings; 3 : + progression; 4 : + information */ +static int g_displayLevel = DEFAULT_DISPLAY_LEVEL; /* 0 : no display; 1: errors; 2 : + result + interaction + warnings; 3 : + progression; 4 : + information */ static FILE* displayOut; #define DISPLAYUPDATE(l, ...) if (g_displayLevel>=l) { \ if ((clock() - g_time > refreshRate) || (g_displayLevel>=4)) \ { g_time = clock(); DISPLAY(__VA_ARGS__); \ - if (g_displayLevel>=4) fflush(stdout); } } + if (g_displayLevel>=4) fflush(displayOut); } } static const clock_t refreshRate = CLOCKS_PER_SEC * 15 / 100; static clock_t g_time = 0; From 680e4e0953bf7c895165e41098a37c3279870c1f Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 23 Mar 2017 11:52:09 -0700 Subject: [PATCH 057/305] Ignore symbolic links unless --force specified --- programs/util.h | 41 ++++++++++++++++++++++++------- programs/zstd.1 | 4 +-- programs/zstd.1.md | 2 +- programs/zstdcli.c | 21 +++++++++++++--- tests/playTests.sh | 13 ++++++++++ zlibWrapper/examples/zwrapbench.c | 2 +- 6 files changed, 66 insertions(+), 17 deletions(-) diff --git a/programs/util.h b/programs/util.h index 59e19d027..e0b705117 100644 --- a/programs/util.h +++ b/programs/util.h @@ -1,6 +1,6 @@ /** * util.h - utility functions - * + * * Copyright (c) 2016-present, Przemyslaw Skibinski, Yann Collet, Facebook, Inc. * All rights reserved. * @@ -228,6 +228,20 @@ UTIL_STATIC U32 UTIL_isDirectory(const char* infilename) return 0; } +UTIL_STATIC U32 UTIL_isLink(const char* infilename) +{ +#if defined(_WIN32) + /* no symlinks on windows */ + (void)infilename; +#else + int r; + stat_t statbuf; + r = lstat(infilename, &statbuf); + if (!r && S_ISLNK(statbuf.st_mode)) return 1; +#endif + return 0; +} + UTIL_STATIC U64 UTIL_getFileSize(const char* infilename) { @@ -271,11 +285,10 @@ UTIL_STATIC void *UTIL_realloc(void *ptr, size_t size) return NULL; } - #ifdef _WIN32 # define UTIL_HAS_CREATEFILELIST -UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_t* pos, char** bufEnd) +UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_t* pos, char** bufEnd, int followLinks) { char* path; int dirLength, fnameLength, pathLength, nbFiles = 0; @@ -311,7 +324,7 @@ UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_ if (strcmp (cFile.cFileName, "..") == 0 || strcmp (cFile.cFileName, ".") == 0) continue; - nbFiles += UTIL_prepareFileList(path, bufStart, pos, bufEnd); /* Recursively call "UTIL_prepareFileList" with the new path. */ + nbFiles += UTIL_prepareFileList(path, bufStart, pos, bufEnd, followLinks); /* Recursively call "UTIL_prepareFileList" with the new path. */ if (*bufStart == NULL) { free(path); FindClose(hFile); return 0; } } else if ((cFile.dwFileAttributes & FILE_ATTRIBUTE_NORMAL) || (cFile.dwFileAttributes & FILE_ATTRIBUTE_ARCHIVE) || (cFile.dwFileAttributes & FILE_ATTRIBUTE_COMPRESSED)) { @@ -339,7 +352,11 @@ UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_ # include /* opendir, readdir */ # include /* strerror, memcpy */ -UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_t* pos, char** bufEnd) +static int g_displayLevel; +#define UTIL_DISPLAY(...) fprintf(stderr, __VA_ARGS__) +#define UTIL_DISPLAYLEVEL(l, ...) { if (g_displayLevel>=l) { UTIL_DISPLAY(__VA_ARGS__); } } + +UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_t* pos, char** bufEnd, int followLinks) { DIR *dir; struct dirent *entry; @@ -360,13 +377,19 @@ UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_ path = (char*) malloc(dirLength + fnameLength + 2); if (!path) { closedir(dir); return 0; } memcpy(path, dirName, dirLength); + path[dirLength] = '/'; memcpy(path+dirLength+1, entry->d_name, fnameLength); pathLength = dirLength+1+fnameLength; path[pathLength] = 0; + if (!followLinks && UTIL_isLink(path)) { + UTIL_DISPLAYLEVEL(2, "Warning : %s is a symbolic link, ignoring\n", path); + continue; + } + if (UTIL_isDirectory(path)) { - nbFiles += UTIL_prepareFileList(path, bufStart, pos, bufEnd); /* Recursively call "UTIL_prepareFileList" with the new path. */ + nbFiles += UTIL_prepareFileList(path, bufStart, pos, bufEnd, followLinks); /* Recursively call "UTIL_prepareFileList" with the new path. */ if (*bufStart == NULL) { free(path); closedir(dir); return 0; } } else { if (*bufStart + *pos + pathLength >= *bufEnd) { @@ -396,7 +419,7 @@ UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_ #else -UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_t* pos, char** bufEnd) +UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_t* pos, char** bufEnd, int followLinks) { (void)bufStart; (void)bufEnd; (void)pos; fprintf(stderr, "Directory %s ignored (compiled without _WIN32 or _POSIX_C_SOURCE)\n", dirName); @@ -411,7 +434,7 @@ UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_ * After finishing usage of the list the structures should be freed with UTIL_freeFileList(params: return value, allocatedBuffer) * In case of error UTIL_createFileList returns NULL and UTIL_freeFileList should not be called. */ -UTIL_STATIC const char** UTIL_createFileList(const char **inputNames, unsigned inputNamesNb, char** allocatedBuffer, unsigned* allocatedNamesNb) +UTIL_STATIC const char** UTIL_createFileList(const char **inputNames, unsigned inputNamesNb, char** allocatedBuffer, unsigned* allocatedNamesNb, int followLinks) { size_t pos; unsigned i, nbFiles; @@ -436,7 +459,7 @@ UTIL_STATIC const char** UTIL_createFileList(const char **inputNames, unsigned i nbFiles++; } } else { - nbFiles += UTIL_prepareFileList(inputNames[i], &buf, &pos, &bufend); + nbFiles += UTIL_prepareFileList(inputNames[i], &buf, &pos, &bufend, followLinks); if (buf == NULL) return NULL; } } diff --git a/programs/zstd.1 b/programs/zstd.1 index f79b3ce16..02423358d 100644 --- a/programs/zstd.1 +++ b/programs/zstd.1 @@ -1,5 +1,5 @@ . -.TH "ZSTD" "1" "March 2017" "zstd 1.1.4" "User Commands" +.TH "ZSTD" "1" "March 2017" "zstd 1.1.5" "User Commands" . .SH "NAME" \fBzstd\fR \- zstd, unzstd, zstdcat \- Compress or decompress \.zst files @@ -110,7 +110,7 @@ save result into \fBfile\fR (only possible with a single INPUT\-FILE) . .TP \fB\-f\fR, \fB\-\-force\fR -overwrite output without prompting +overwrite output without prompting, and (de)compress symbolic links . .TP \fB\-c\fR, \fB\-\-stdout\fR diff --git a/programs/zstd.1.md b/programs/zstd.1.md index bb5b22664..c9ff33276 100644 --- a/programs/zstd.1.md +++ b/programs/zstd.1.md @@ -108,7 +108,7 @@ the last one takes effect. * `-o file`: save result into `file` (only possible with a single INPUT-FILE) * `-f`, `--force`: - overwrite output without prompting + overwrite output without prompting, and (de)compress symbolic links * `-c`, `--stdout`: force write to standard output, even if it is the console * `--[no-]sparse`: diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 29bc76377..38b8394a0 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -99,7 +99,7 @@ static int usage(const char* programName) #endif DISPLAY( " -D file: use `file` as Dictionary \n"); DISPLAY( " -o file: result stored into `file` (only if 1 input file) \n"); - DISPLAY( " -f : overwrite output without prompting \n"); + DISPLAY( " -f : overwrite output without prompting and (de)compress links \n"); DISPLAY( "--rm : remove source file(s) after successful de/compression \n"); DISPLAY( " -k : preserve source file(s) (default) \n"); DISPLAY( " -h/-H : display help/long help and exit\n"); @@ -270,6 +270,7 @@ int main(int argCount, const char* argv[]) { int argNb, forceStdout=0, + followLinks=0, main_pause=0, nextEntryIsDictionary=0, operationResult=0, @@ -357,7 +358,7 @@ int main(int argCount, const char* argv[]) if (!strcmp(argument, "--compress")) { operation=zom_compress; continue; } if (!strcmp(argument, "--decompress")) { operation=zom_decompress; continue; } if (!strcmp(argument, "--uncompress")) { operation=zom_decompress; continue; } - if (!strcmp(argument, "--force")) { FIO_overwriteMode(); continue; } + if (!strcmp(argument, "--force")) { FIO_overwriteMode(); forceStdout=1; followLinks=1; continue; } if (!strcmp(argument, "--version")) { g_displayOut=stdout; DISPLAY(WELCOME_MESSAGE); CLEAN_RETURN(0); } if (!strcmp(argument, "--help")) { g_displayOut=stdout; CLEAN_RETURN(usage_advanced(programName)); } if (!strcmp(argument, "--verbose")) { g_displayLevel++; continue; } @@ -445,7 +446,7 @@ int main(int argCount, const char* argv[]) case 'D': nextEntryIsDictionary = 1; lastCommand = 1; argument++; break; /* Overwrite */ - case 'f': FIO_overwriteMode(); forceStdout=1; argument++; break; + case 'f': FIO_overwriteMode(); forceStdout=1; followLinks=1; argument++; break; /* Verbose mode */ case 'v': g_displayLevel++; argument++; break; @@ -581,9 +582,21 @@ int main(int argCount, const char* argv[]) DISPLAYLEVEL(4, "PLATFORM_POSIX_VERSION defined: %ldL\n", (long) PLATFORM_POSIX_VERSION); #endif + + if (!followLinks) { + unsigned u; + for (u=0, fileNamesNb=0; u /dev/full" $ECHO foo | $ZSTD > /dev/full && die "write error not detected!" $ECHO "$ECHO foo | $ZSTD | $ZSTD -d > /dev/full" $ECHO foo | $ZSTD | $ZSTD -d > /dev/full && die "write error not detected!" + +$ECHO "\n**** symbolic link test **** " + +rm -f hello.tmp world.tmp hello.tmp.zst world.tmp.zst +$ECHO "hello world" > hello.tmp +ln -s hello.tmp world.tmp +$ZSTD world.tmp hello.tmp +ls hello.tmp.zst || die "regular file should have been compressed!" +ls world.tmp.zst && die "symbolic link should not have been compressed!" +$ZSTD world.tmp hello.tmp -f +ls world.tmp.zst || die "symbol link should have been compressed with --force" +rm -f hello.tmp world.tmp hello.tmp.zst world.tmp.zst + fi diff --git a/zlibWrapper/examples/zwrapbench.c b/zlibWrapper/examples/zwrapbench.c index 951c315b8..1c3391e90 100644 --- a/zlibWrapper/examples/zwrapbench.c +++ b/zlibWrapper/examples/zwrapbench.c @@ -982,7 +982,7 @@ int main(int argCount, char** argv) #ifdef UTIL_HAS_CREATEFILELIST if (recursive) { - fileNamesTable = UTIL_createFileList(filenameTable, filenameIdx, &fileNamesBuf, &fileNamesNb); + fileNamesTable = UTIL_createFileList(filenameTable, filenameIdx, &fileNamesBuf, &fileNamesNb, 1); if (fileNamesTable) { unsigned u; for (u=0; u Date: Thu, 23 Mar 2017 12:09:35 -0700 Subject: [PATCH 058/305] Fix displayLevel redefinition issues --- programs/util.h | 8 ++++---- programs/zstdcli.c | 1 + 2 files changed, 5 insertions(+), 4 deletions(-) diff --git a/programs/util.h b/programs/util.h index e0b705117..9d23cbb8e 100644 --- a/programs/util.h +++ b/programs/util.h @@ -285,6 +285,10 @@ UTIL_STATIC void *UTIL_realloc(void *ptr, size_t size) return NULL; } +static int g_utilDisplayLevel; +#define UTIL_DISPLAY(...) fprintf(stderr, __VA_ARGS__) +#define UTIL_DISPLAYLEVEL(l, ...) { if (g_utilDisplayLevel>=l) { UTIL_DISPLAY(__VA_ARGS__); } } + #ifdef _WIN32 # define UTIL_HAS_CREATEFILELIST @@ -352,10 +356,6 @@ UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_ # include /* opendir, readdir */ # include /* strerror, memcpy */ -static int g_displayLevel; -#define UTIL_DISPLAY(...) fprintf(stderr, __VA_ARGS__) -#define UTIL_DISPLAYLEVEL(l, ...) { if (g_displayLevel>=l) { UTIL_DISPLAY(__VA_ARGS__); } } - UTIL_STATIC int UTIL_prepareFileList(const char *dirName, char** bufStart, size_t* pos, char** bufEnd, int followLinks) { DIR *dir; diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 38b8394a0..4d7fbb357 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -583,6 +583,7 @@ int main(int argCount, const char* argv[]) #endif + g_utilDisplayLevel = g_displayLevel; if (!followLinks) { unsigned u; for (u=0, fileNamesNb=0; u Date: Thu, 23 Mar 2017 12:41:51 -0700 Subject: [PATCH 059/305] Add symbolic link checking to pzstd --- contrib/pzstd/Options.cpp | 22 ++++++++++++++++++++-- 1 file changed, 20 insertions(+), 2 deletions(-) diff --git a/contrib/pzstd/Options.cpp b/contrib/pzstd/Options.cpp index a0d969393..1f53f2bff 100644 --- a/contrib/pzstd/Options.cpp +++ b/contrib/pzstd/Options.cpp @@ -91,7 +91,7 @@ void usage() { std::fprintf(stderr, " -# : # compression level (1-%d, default:%d)\n", kMaxNonUltraCompressionLevel, kDefaultCompressionLevel); std::fprintf(stderr, " -d, --decompress : decompression\n"); std::fprintf(stderr, " -o file : result stored into `file` (only if 1 input file)\n"); - std::fprintf(stderr, " -f, --force : overwrite output without prompting\n"); + std::fprintf(stderr, " -f, --force : overwrite output without prompting, (de)compress links\n"); std::fprintf(stderr, " --rm : remove source file(s) after successful (de)compression\n"); std::fprintf(stderr, " -k, --keep : preserve source file(s) (default)\n"); std::fprintf(stderr, " -h, --help : display help and exit\n"); @@ -121,6 +121,7 @@ Options::Status Options::parse(int argc, const char **argv) { bool recursive = false; bool ultra = false; bool forceStdout = false; + bool followLinks = false; // Local copy of input files, which are pointers into argv. std::vector localInputFiles; for (int i = 1; i < argc; ++i) { @@ -255,6 +256,7 @@ Options::Status Options::parse(int argc, const char **argv) { case 'f': overwrite = true; forceStdout = true; + followLinks = true; break; case 't': test = true; @@ -328,13 +330,29 @@ Options::Status Options::parse(int argc, const char **argv) { } } + g_utilDisplayLevel = verbosity; + // Remove local input files that are symbolic links + if (!followLinks) { + std::remove_if(localInputFiles.begin(), localInputFiles.end(), + [&](const char *path) { + bool isLink = UTIL_isLink(path); + if (isLink && verbosity >= 2) { + std::fprintf( + stderr, + "Warning : %s is symbolic link, ignoring\n", + path); + } + return isLink; + }); + } + // Translate input files/directories into files to (de)compress if (recursive) { char *scratchBuffer = nullptr; unsigned numFiles = 0; const char **files = UTIL_createFileList(localInputFiles.data(), localInputFiles.size(), - &scratchBuffer, &numFiles); + &scratchBuffer, &numFiles, followLinks); if (files == nullptr) { std::fprintf(stderr, "Error traversing directories\n"); return Status::Failure; From 7f67f8dce6acb18d2f8da90435040659ba4cd040 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 23 Mar 2017 14:33:38 -0700 Subject: [PATCH 060/305] Educational decoder polish updates --- doc/educational_decoder/zstd_decompress.c | 181 ++++++++++++---------- 1 file changed, 95 insertions(+), 86 deletions(-) diff --git a/doc/educational_decoder/zstd_decompress.c b/doc/educational_decoder/zstd_decompress.c index 346cd296a..7c8d8114d 100644 --- a/doc/educational_decoder/zstd_decompress.c +++ b/doc/educational_decoder/zstd_decompress.c @@ -27,16 +27,19 @@ size_t ZSTD_decompress_with_dict(void *const dst, const size_t dst_len, /// Get the decompressed size of an input stream so memory can be allocated in /// advance +/// Returns -1 if the size can't be determined size_t ZSTD_get_decompressed_size(const void *const src, const size_t src_len); /******* UTILITY MACROS AND TYPES *********************************************/ -// Max block size decompressed size is 128 KB and literal blocks must be smaller -// than that +// Max block size decompressed size is 128 KB and literal blocks can't be +// larger than their block #define MAX_LITERALS_SIZE ((size_t)128 * 1024) #define MAX(a, b) ((a) > (b) ? (a) : (b)) #define MIN(a, b) ((a) < (b) ? (a) : (b)) +/// This decoder calls exit(1) when it encounters an error, however a production +/// library should propagate error codes #define ERROR(s) \ do { \ fprintf(stderr, "Error: %s\n", s); \ @@ -67,20 +70,22 @@ typedef int64_t i64; /// decompression functions. /*** IO STREAM OPERATIONS *************/ -/// These structs are the interface for IO, and do bounds checking on all -/// operations. They should be used opaquely to ensure safety. -/// Output is always done byte-by-byte +/// ostream_t/istream_t are used to wrap the pointers/length data passed into +/// ZSTD_decompress, so that all IO operations are safely bounds checked +/// They are written/read forward, and reads are treated as little-endian +/// They should be used opaquely to ensure safety typedef struct { u8 *ptr; size_t len; } ostream_t; -/// Input often reads a few bits at a time, so maintain an internal offset typedef struct { const u8 *ptr; - int bit_offset; size_t len; + + // Input often reads a few bits at a time, so maintain an internal offset + int bit_offset; } istream_t; /// The following two functions are the only ones that allow the istream to be @@ -88,7 +93,7 @@ typedef struct { /// Reads `num` bits from a bitstream, and updates the internal offset static inline u64 IO_read_bits(istream_t *const in, const int num_bits); -/// Rewinds the stream by `num` bits +/// Backs-up the stream by `num` bits so they can be read again static inline void IO_rewind_bits(istream_t *const in, const int num_bits); /// If the remaining bits in a byte will be unused, advance to the end of the /// byte @@ -101,29 +106,30 @@ static inline void IO_write_byte(ostream_t *const out, u8 symb); /// be byte aligned. static inline size_t IO_istream_len(const istream_t *const in); -/// Returns a pointer where `len` bytes can be read, and advances the internal -/// state. The stream must be byte aligned. +/// Advances the stream by `len` bytes, and returns a pointer to the chunk that +/// was skipped. The stream must be byte aligned. static inline const u8 *IO_read_bytes(istream_t *const in, size_t len); -/// Returns a pointer where `len` bytes can be written, and advances the internal -/// state. The stream must be byte aligned. +/// Advances the stream by `len` bytes, and returns a pointer to the chunk that +/// was skipped so it can be written to. static inline u8 *IO_write_bytes(ostream_t *const out, size_t len); /// Advance the inner state by `len` bytes. The stream must be byte aligned. static inline void IO_advance_input(istream_t *const in, size_t len); -/// Returns an `ostream_t` constructed from the given pointer and length +/// Returns an `ostream_t` constructed from the given pointer and length. static inline ostream_t IO_make_ostream(u8 *out, size_t len); -/// Returns an `istream_t` constructed from the given pointer and length +/// Returns an `istream_t` constructed from the given pointer and length. static inline istream_t IO_make_istream(const u8 *in, size_t len); -/// Returns an `istream_t` with the same base as `in`, and length `len` -/// Then, advance `in` to account for the consumed bytes -/// `in` must be byte aligned +/// Returns an `istream_t` with the same base as `in`, and length `len`. +/// Then, advance `in` to account for the consumed bytes. +/// `in` must be byte aligned. static inline istream_t IO_make_sub_istream(istream_t *const in, size_t len); /*** END IO STREAM OPERATIONS *********/ /*** BITSTREAM OPERATIONS *************/ -/// Read `num` bits (up to 64) from `src + offset`, where `offset` is in bits +/// Read `num` bits (up to 64) from `src + offset`, where `offset` is in bits, +/// and return them interpreted as a little-endian unsigned integer. static inline u64 read_bits_LE(const u8 *src, const int num_bits, const size_t offset); @@ -383,8 +389,8 @@ size_t ZSTD_decompress_with_dict(void *const dst, const size_t dst_len, parse_dictionary(&parsed_dict, (const u8 *)dict, dict_len); } - istream_t in = {(const u8 *)src, 0, src_len}; - ostream_t out = {(u8 *)dst, dst_len}; + istream_t in = IO_make_istream(src, src_len); + ostream_t out = IO_make_ostream(dst, dst_len); // "A content compressed by Zstandard is transformed into a Zstandard frame. // Multiple frames can be appended into a single file or stream. A frame is @@ -632,6 +638,7 @@ static void frame_context_apply_dict(frame_context_t *const ctx, FSE_copy_dtable(&ctx->of_dtable, &dict->of_dtable); FSE_copy_dtable(&ctx->ml_dtable, &dict->ml_dtable); + // Copy the repeated offsets memcpy(ctx->previous_offsets, dict->previous_offsets, sizeof(ctx->previous_offsets)); } @@ -667,7 +674,7 @@ static void decompress_data(frame_context_t *const ctx, ostream_t *const out, // number of bytes to read and copy." const u8 *const read_ptr = IO_read_bytes(in, block_len); u8 *const write_ptr = IO_write_bytes(out, block_len); - // + // Copy the raw data into the output memcpy(write_ptr, read_ptr, block_len); @@ -681,7 +688,7 @@ static void decompress_data(frame_context_t *const ctx, ostream_t *const out, const u8 *const read_ptr = IO_read_bytes(in, 1); u8 *const write_ptr = IO_write_bytes(out, block_len); - // Copy `block_len` copies of `streams->src[0]` to the output + // Copy `block_len` copies of `read_ptr[0]` to the output memset(write_ptr, read_ptr[0], block_len); ctx->current_total_output += block_len; @@ -750,7 +757,7 @@ static size_t decode_literals_compressed(frame_context_t *const ctx, u8 **const literals, const int block_type, const int size_format); -static void decode_huf_table(istream_t *const in, HUF_dtable *const dtable); +static void decode_huf_table(HUF_dtable *const dtable, istream_t *const in); static void fse_decode_hufweights(ostream_t *weights, istream_t *const in, int *const num_symbs); @@ -893,12 +900,12 @@ static size_t decode_literals_compressed(frame_context_t *const ctx, istream_t huf_stream = IO_make_sub_istream(in, compressed_size); if (block_type == 2) { - // Decode provided Huffman table + // Decode the provided Huffman table // "This section is only present when Literals_Block_Type type is // Compressed_Literals_Block (2)." HUF_free_dtable(&ctx->literals_dtable); - decode_huf_table(&huf_stream, &ctx->literals_dtable); + decode_huf_table(&ctx->literals_dtable, &huf_stream); } else { // If the previous Huffman table is being repeated, ensure it exists if (!ctx->literals_dtable.symbols) { @@ -921,13 +928,13 @@ static size_t decode_literals_compressed(frame_context_t *const ctx, } // Decode the Huffman table description -static void decode_huf_table(istream_t *const in, HUF_dtable *const dtable) { - const u8 header = IO_read_bits(in, 8); - +static void decode_huf_table(HUF_dtable *const dtable, istream_t *const in) { // "All literal values from zero (included) to last present one (excluded) // are represented by Weight with values from 0 to Max_Number_of_Bits." // "This is a single byte value (0-255), which describes how to decode the list of weights." + const u8 header = IO_read_bits(in, 8); + u8 weights[HUF_MAX_SYMBS]; memset(weights, 0, sizeof(weights)); @@ -996,7 +1003,7 @@ typedef struct { u16 ll_state; u16 of_state; u16 ml_state; -} sequence_state_t; +} sequence_states_t; /// Different modes to signal to decode_seq_tables what to do typedef enum { @@ -1051,10 +1058,10 @@ static void decompress_sequences(frame_context_t *const ctx, istream_t *const in, sequence_command_t *const sequences, const size_t num_sequences); -static sequence_command_t decode_sequence(sequence_state_t *const state, +static sequence_command_t decode_sequence(sequence_states_t *const state, const u8 *const src, i64 *const offset); -static void decode_seq_table(istream_t *const in, FSE_dtable *const table, +static void decode_seq_table(FSE_dtable *const table, istream_t *const in, const seq_part_t type, const seq_mode_t mode); static size_t decode_sequences(frame_context_t *const ctx, istream_t *in, @@ -1130,34 +1137,33 @@ static void decompress_sequences(frame_context_t *const ctx, istream_t *in, // Offsets // Match Lengths" // Update the tables we have stored in the context - decode_seq_table(in, &ctx->ll_dtable, seq_literal_length, + decode_seq_table(&ctx->ll_dtable, in, seq_literal_length, (compression_modes >> 6) & 3); - decode_seq_table(in, &ctx->of_dtable, seq_offset, + decode_seq_table(&ctx->of_dtable, in, seq_offset, (compression_modes >> 4) & 3); - decode_seq_table(in, &ctx->ml_dtable, seq_match_length, + decode_seq_table(&ctx->ml_dtable, in, seq_match_length, (compression_modes >> 2) & 3); - // Check to make sure none of the tables are uninitialized - if (!ctx->ll_dtable.symbols || !ctx->of_dtable.symbols || - !ctx->ml_dtable.symbols) { - CORRUPTION(); + + sequence_states_t states; + + // Initialize the decoding tables + { + states.ll_table = ctx->ll_dtable; + states.of_table = ctx->of_dtable; + states.ml_table = ctx->ml_dtable; } - sequence_state_t state; - // Copy the context's tables into the local state - memcpy(&state.ll_table, &ctx->ll_dtable, sizeof(FSE_dtable)); - memcpy(&state.of_table, &ctx->of_dtable, sizeof(FSE_dtable)); - memcpy(&state.ml_table, &ctx->ml_dtable, sizeof(FSE_dtable)); - - size_t len = IO_istream_len(in); + const size_t len = IO_istream_len(in); const u8 *const src = IO_read_bytes(in, len); // "After writing the last bit containing information, the compressor writes // a single 1-bit and then fills the byte with 0-7 0 bits of padding." const int padding = 8 - highest_set_bit(src[len - 1]); - i64 offset = len * 8 - padding; + // The offset starts at the end because FSE streams are read backwards + i64 bit_offset = len * 8 - padding; // "The bitstream starts with initial state values, each using the required // number of bits in their respective accuracy, decoded previously from @@ -1165,24 +1171,22 @@ static void decompress_sequences(frame_context_t *const ctx, istream_t *in, // // It starts by Literals_Length_State, followed by Offset_State, and finally // Match_Length_State." - FSE_init_state(&state.ll_table, &state.ll_state, src, &offset); - FSE_init_state(&state.of_table, &state.of_state, src, &offset); - FSE_init_state(&state.ml_table, &state.ml_state, src, &offset); + FSE_init_state(&states.ll_table, &states.ll_state, src, &bit_offset); + FSE_init_state(&states.of_table, &states.of_state, src, &bit_offset); + FSE_init_state(&states.ml_table, &states.ml_state, src, &bit_offset); for (size_t i = 0; i < num_sequences; i++) { // Decode sequences one by one - sequences[i] = decode_sequence(&state, src, &offset); + sequences[i] = decode_sequence(&states, src, &bit_offset); } - if (offset != 0) { + if (bit_offset != 0) { CORRUPTION(); } - - // Don't free tables so they can be used in the next block } // Decode a single sequence and update the state -static sequence_command_t decode_sequence(sequence_state_t *const state, +static sequence_command_t decode_sequence(sequence_states_t *const states, const u8 *const src, i64 *const offset) { // "Each symbol is a code in its own context, which specifies Baseline and @@ -1190,9 +1194,9 @@ static sequence_command_t decode_sequence(sequence_state_t *const state, // additional bits in the same bitstream." // Decode symbols, but don't update states - const u8 of_code = FSE_peek_symbol(&state->of_table, state->of_state); - const u8 ll_code = FSE_peek_symbol(&state->ll_table, state->ll_state); - const u8 ml_code = FSE_peek_symbol(&state->ml_table, state->ml_state); + const u8 of_code = FSE_peek_symbol(&states->of_table, states->of_state); + const u8 ll_code = FSE_peek_symbol(&states->ll_table, states->ll_state); + const u8 ml_code = FSE_peek_symbol(&states->ml_table, states->ml_state); // Offset doesn't need a max value as it's not decoded using a table if (ll_code > SEQ_MAX_CODES[seq_literal_length] || @@ -1220,17 +1224,18 @@ static sequence_command_t decode_sequence(sequence_state_t *const state, // then Offset_State." // If the stream is complete don't read bits to update state if (*offset != 0) { - FSE_update_state(&state->ll_table, &state->ll_state, src, offset); - FSE_update_state(&state->ml_table, &state->ml_state, src, offset); - FSE_update_state(&state->of_table, &state->of_state, src, offset); + FSE_update_state(&states->ll_table, &states->ll_state, src, offset); + FSE_update_state(&states->ml_table, &states->ml_state, src, offset); + FSE_update_state(&states->of_table, &states->of_state, src, offset); } return seq; } /// Given a sequence part and table mode, decode the FSE distribution -static void decode_seq_table(istream_t *const in, FSE_dtable *const table, - const seq_part_t type, const seq_mode_t mode) { +/// Errors if the mode is `seq_repeat` without a pre-existing table in `table` +static void decode_seq_table(FSE_dtable *const table, istream_t *const in, + const seq_part_t type, const seq_mode_t mode) { // Constant arrays indexed by seq_part_t const i16 *const default_distributions[] = {SEQ_LITERAL_LENGTH_DEFAULT_DIST, SEQ_OFFSET_DEFAULT_DIST, @@ -1271,12 +1276,17 @@ static void decode_seq_table(istream_t *const in, FSE_dtable *const table, // "Repeat_Mode : re-use distribution table from previous compressed // block." // Nothing to do here, table will be unchanged + if (!table->symbols) { + // This mode is invalid if we don't already have a table + CORRUPTION(); + } break; default: // Impossible, as mode is from 0-3 IMPOSSIBLE(); break; } + } /******* END SEQUENCE DECODING ************************************************/ @@ -1295,6 +1305,8 @@ static void execute_sequences(frame_context_t *const ctx, ostream_t *const out, const sequence_command_t seq = sequences[i]; { + // If the sequence asks for more literals than are left, the + // sequence must be corrupted if (seq.literal_length > IO_istream_len(&litstream)) { CORRUPTION(); } @@ -1335,7 +1347,8 @@ static void execute_sequences(frame_context_t *const ctx, ostream_t *const out, // as per the exception listed above offset = idx < 3 ? offset_hist[idx] : offset_hist[0] - 1; - // If idx == 1 we don't need to modify offset_hist[2] + // If idx == 1 we don't need to modify offset_hist[2], since + // we're using the second-most recent code if (idx > 1) { offset_hist[2] = offset_hist[1]; } @@ -1343,6 +1356,8 @@ static void execute_sequences(frame_context_t *const ctx, ostream_t *const out, offset_hist[0] = offset; } } else { + // When it's not a repeat offset: + // "if (Offset_Value > 3) offset = Offset_Value - 3;" offset = seq.offset - 3; // Shift back history @@ -1390,11 +1405,11 @@ static void execute_sequences(frame_context_t *const ctx, ostream_t *const out, total_output += seq.match_length; } + // Copy any leftover literals { size_t len = IO_istream_len(&litstream); u8 *const write_ptr = IO_write_bytes(out, len); const u8 *const read_ptr = IO_read_bytes(&litstream, len); - // Copy any leftover literals memcpy(write_ptr, read_ptr, len); total_output += len; @@ -1516,10 +1531,10 @@ static void parse_dictionary(dictionary_t *const dict, const u8 *src, // recent offsets (instead of using {1,4,8}), stored in order, 4-bytes // little-endian each, for a total of 12 bytes. Each recent offset must have // a value < dictionary size." - decode_huf_table(&in, &dict->literals_dtable); - decode_seq_table(&in, &dict->of_dtable, seq_offset, seq_fse); - decode_seq_table(&in, &dict->ml_dtable, seq_match_length, seq_fse); - decode_seq_table(&in, &dict->ll_dtable, seq_literal_length, seq_fse); + decode_huf_table(&dict->literals_dtable, &in); + decode_seq_table(&dict->of_dtable, &in, seq_offset, seq_fse); + decode_seq_table(&dict->ml_dtable, &in, seq_match_length, seq_fse); + decode_seq_table(&dict->ll_dtable, &in, seq_literal_length, seq_fse); // Read in the previous offset history dict->previous_offsets[0] = IO_read_bits(&in, 32); @@ -1687,25 +1702,18 @@ static inline ostream_t IO_make_ostream(u8 *out, size_t len) { /// Returns an `istream_t` constructed from the given pointer and length static inline istream_t IO_make_istream(const u8 *in, size_t len) { - return (istream_t) { in, 0, len }; + return (istream_t) { in, len, 0 }; } /// Returns an `istream_t` with the same base as `in`, and length `len` /// Then, advance `in` to account for the consumed bytes /// `in` must be byte aligned static inline istream_t IO_make_sub_istream(istream_t *const in, size_t len) { - if (len > in->len) { - INP_SIZE(); - } - if (in->bit_offset != 0) { - UNALIGNED(); - } - const istream_t sub = { in->ptr, in->bit_offset, len }; + // Consume `len` bytes of the parent stream + const u8 *const ptr = IO_read_bytes(in, len); - in->ptr += len; - in->len -= len; - - return sub; + // Make a substream using the pointer to those `len` bytes + return IO_make_istream(ptr, len); } /******* END IO STREAM OPERATIONS *********************************************/ @@ -1726,7 +1734,7 @@ static inline u64 read_bits_LE(const u8 *src, const int num_bits, int left = num_bits; while (left > 0) { u64 mask = left >= 8 ? 0xff : (((u64)1 << left) - 1); - // Dead the next byte, shift it to account for the offset, and then mask + // Read the next byte, shift it to account for the offset, and then mask // out the top part if we don't need all the bits res += (((u64)*src++ >> bit_offset) & mask) << shift; shift += 8 - bit_offset; @@ -1819,15 +1827,16 @@ static size_t HUF_decompress_1stream(const HUF_dtable *const dtable, // last byte contains between 0 and 7 useful bits." const int padding = 8 - highest_set_bit(src[len - 1]); - i64 offset = len * 8 - padding; + // Offset starts at the end because HUF streams are read backwards + i64 bit_offset = len * 8 - padding; u16 state; - HUF_init_state(dtable, &state, src, &offset); + HUF_init_state(dtable, &state, src, &bit_offset); size_t symbols_written = 0; - while (offset > -dtable->max_bits) { + while (bit_offset > -dtable->max_bits) { // Iterate over the stream, decoding one symbol at a time - IO_write_byte(out, HUF_decode_symbol(dtable, &state, src, &offset)); + IO_write_byte(out, HUF_decode_symbol(dtable, &state, src, &bit_offset)); symbols_written++; } // "The process continues up to reading the required number of symbols per @@ -1840,7 +1849,7 @@ static size_t HUF_decompress_1stream(const HUF_dtable *const dtable, // before the start of `src` // Therefore `offset`, the edge to start reading new bits at, should be // dtable->max_bits before the start of the stream - if (offset != -dtable->max_bits) { + if (bit_offset != -dtable->max_bits) { CORRUPTION(); } From bea78e8fc26bd9d46bc2d98a3d9099586c4ad29d Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 22 Mar 2017 18:09:11 -0700 Subject: [PATCH 061/305] limited CDict acceptation criteria to be the same as DDict --- lib/compress/zstd_compress.c | 55 ++++++++++++++++++------------------ 1 file changed, 28 insertions(+), 27 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 1aab3553f..eecc76950 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -2567,18 +2567,15 @@ static size_t ZSTD_checkDictNCount(short* normalizedCounter, unsigned dictMaxSym /* Dictionary format : - Magic == ZSTD_DICT_MAGIC (4 bytes) - HUF_writeCTable(256) - FSE_writeNCount(off) - FSE_writeNCount(ml) - FSE_writeNCount(ll) - RepOffsets - Dictionary content -*/ -/*! ZSTD_loadDictEntropyStats() : - @return : size read from dictionary - note : magic number supposed already checked */ -static size_t ZSTD_loadDictEntropyStats(ZSTD_CCtx* cctx, const void* dict, size_t dictSize) + * See : + * https://github.com/facebook/zstd/blob/master/doc/zstd_compression_format.md#dictionary-format + */ +/*! ZSTD_loadDictionary() : + * @return : size read from dictionary + * note : magic number supposed already checked + * dictSize supposed > 8 + */ +static size_t ZSTD_loadDictionary(ZSTD_CCtx* cctx, const void* dict, size_t dictSize) { const BYTE* dictPtr = (const BYTE*)dict; const BYTE* const dictEnd = dictPtr + dictSize; @@ -2586,7 +2583,11 @@ static size_t ZSTD_loadDictEntropyStats(ZSTD_CCtx* cctx, const void* dict, size_ unsigned offcodeMaxValue = MaxOff; BYTE scratchBuffer[1<hufTable, 255, dict, dictSize); + dictPtr += 4; /* skip magic number */ + cctx->dictID = cctx->params.fParams.noDictIDFlag ? 0 : MEM_readLE32(dictPtr); + dictPtr += 4; + + { size_t const hufHeaderSize = HUF_readCTable(cctx->hufTable, 255, dictPtr, dictEnd-dictPtr); if (HUF_isError(hufHeaderSize)) return ERROR(dictionary_corrupted); dictPtr += hufHeaderSize; } @@ -2628,15 +2629,20 @@ static size_t ZSTD_loadDictEntropyStats(ZSTD_CCtx* cctx, const void* dict, size_ cctx->rep[2] = MEM_readLE32(dictPtr+8); if (cctx->rep[2] == 0 || cctx->rep[2] >= dictSize) return ERROR(dictionary_corrupted); dictPtr += 12; - { U32 offcodeMax = MaxOff; - if ((size_t)(dictEnd - dictPtr) <= ((U32)-1) - 128 KB) { - U32 const maxOffset = (U32)(dictEnd - dictPtr) + 128 KB; /* The maximum offset that must be supported */ - /* Calculate minimum offset code required to represent maxOffset */ - offcodeMax = ZSTD_highbit32(maxOffset); + { size_t const dictContentSize = (size_t)(dictEnd - dictPtr); + U32 offcodeMax = MaxOff; + if (dictContentSize <= ((U32)-1) - 128 KB) { + U32 const maxOffset = (U32)dictContentSize + 128 KB; /* The maximum offset that must be supported */ + offcodeMax = ZSTD_highbit32(maxOffset); /* Calculate minimum offset code required to represent maxOffset */ } - /* Every possible supported offset <= dictContentSize + 128 KB must be representable */ + /* All offset values <= dictContentSize + 128 KB must be representable */ CHECK_F (ZSTD_checkDictNCount(offcodeNCount, offcodeMaxValue, MIN(offcodeMax, MaxOff))); - } + /* All repCodes must be <= dictContentSize and != 0*/ + { U32 u; + for (u=0; u<3; u++) { + if (cctx->rep[u] == 0) return ERROR(dictionary_corrupted); + if (cctx->rep[u] > dictContentSize) return ERROR(dictionary_corrupted); + } } } cctx->flagStaticTables = 1; cctx->flagStaticHufTable = HUF_repeat_valid; @@ -2652,14 +2658,9 @@ static size_t ZSTD_compress_insertDictionary(ZSTD_CCtx* zc, const void* dict, si /* dict as pure content */ if ((MEM_readLE32(dict) != ZSTD_DICT_MAGIC) || (zc->forceRawDict)) return ZSTD_loadDictionaryContent(zc, dict, dictSize); - zc->dictID = zc->params.fParams.noDictIDFlag ? 0 : MEM_readLE32((const char*)dict+4); - /* known magic number : dict is parsed for entropy stats and content */ - { size_t const loadError = ZSTD_loadDictEntropyStats(zc, (const char*)dict+8 /* skip dictHeader */, dictSize-8); - size_t const eSize = loadError + 8; - if (ZSTD_isError(loadError)) return loadError; - return ZSTD_loadDictionaryContent(zc, (const char*)dict+eSize, dictSize-eSize); - } + /* dict as zstd dictionary */ + return ZSTD_loadDictionary(zc, dict, dictSize); } /*! ZSTD_compressBegin_internal() : From f332ece468f2368001c6e593c8f2dade56cadd8e Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 23 Mar 2017 16:24:02 -0700 Subject: [PATCH 062/305] dictBuilder fails to create dictionary on certain input Properly expressed with an error code (see zstd_errors.h) and a cli return code != 0 --- lib/common/error_private.c | 1 + lib/common/zstd_errors.h | 1 + lib/dictBuilder/zdict.c | 8 +++++--- programs/zstdcli.c | 4 ++-- tests/playTests.sh | 5 +++++ 5 files changed, 14 insertions(+), 5 deletions(-) diff --git a/lib/common/error_private.c b/lib/common/error_private.c index a0fa1724a..44ae20104 100644 --- a/lib/common/error_private.c +++ b/lib/common/error_private.c @@ -37,6 +37,7 @@ const char* ERR_getErrorString(ERR_enum code) case PREFIX(maxSymbolValue_tooSmall): return "Specified maxSymbolValue is too small"; case PREFIX(dictionary_corrupted): return "Dictionary is corrupted"; case PREFIX(dictionary_wrong): return "Dictionary mismatch"; + case PREFIX(dictionaryCreation_failed): return "Cannot create Dictionary from provided samples"; case PREFIX(maxCode): default: return notErrorCode; } diff --git a/lib/common/zstd_errors.h b/lib/common/zstd_errors.h index 949dbd0ff..3d579d969 100644 --- a/lib/common/zstd_errors.h +++ b/lib/common/zstd_errors.h @@ -57,6 +57,7 @@ typedef enum { ZSTD_error_maxSymbolValue_tooSmall, ZSTD_error_dictionary_corrupted, ZSTD_error_dictionary_wrong, + ZSTD_error_dictionaryCreation_failed, ZSTD_error_maxCode } ZSTD_ErrorCode; diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index c84195791..842167db6 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -62,8 +62,9 @@ #define MINRATIO 4 static const int g_compressionLevel_default = 6; static const U32 g_selectivity_default = 9; -static const size_t g_provision_entropySize = 200; +static const size_t g_provision_entropySize = 192; static const size_t g_min_fast_dictContent = 192; +static const size_t g_dictContentSize_min = 32; /*-************************************* @@ -929,8 +930,8 @@ size_t ZDICT_trainFromBuffer_unsafe( /* checks */ if (!dictList) return ERROR(memory_allocation); - if (maxDictSize <= g_provision_entropySize + g_min_fast_dictContent) { free(dictList); return ERROR(dstSize_tooSmall); } - if (samplesBuffSize < ZDICT_MIN_SAMPLES_SIZE) { free(dictList); return 0; } /* not enough source to create dictionary */ + if (maxDictSize <= g_provision_entropySize + g_min_fast_dictContent) { free(dictList); return ERROR(dstSize_tooSmall); } /* requested dictionary size is too small */ + if (samplesBuffSize < ZDICT_MIN_SAMPLES_SIZE) { free(dictList); return ERROR(dictionaryCreation_failed); } /* not enough source to create dictionary */ /* init */ ZDICT_initDictItem(dictList); @@ -963,6 +964,7 @@ size_t ZDICT_trainFromBuffer_unsafe( /* create dictionary */ { U32 dictContentSize = ZDICT_dictSize(dictList); + if (dictContentSize < g_dictContentSize_min) { free(dictList); return ERROR(dictionaryCreation_failed); } /* dictionary content too small */ if (dictContentSize < targetDictSize/3) { DISPLAYLEVEL(2, "! warning : selected content significantly smaller than requested (%u < %u) \n", dictContentSize, (U32)maxDictSize); if (minRep > MINRATIO) { diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 4d7fbb357..281301bd1 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -629,7 +629,7 @@ int main(int argCount, const char* argv[]) coverParams.compressionLevel = dictCLevel; coverParams.notificationLevel = g_displayLevel; coverParams.dictID = dictID; - DiB_trainFromFiles(outFileName, maxDictSize, filenameTable, filenameIdx, NULL, &coverParams, cover - 1); + operationResult = DiB_trainFromFiles(outFileName, maxDictSize, filenameTable, filenameIdx, NULL, &coverParams, cover - 1); } else { ZDICT_params_t dictParams; memset(&dictParams, 0, sizeof(dictParams)); @@ -637,7 +637,7 @@ int main(int argCount, const char* argv[]) dictParams.selectivityLevel = dictSelect; dictParams.notificationLevel = g_displayLevel; dictParams.dictID = dictID; - DiB_trainFromFiles(outFileName, maxDictSize, filenameTable, filenameIdx, &dictParams, NULL, 0); + operationResult = DiB_trainFromFiles(outFileName, maxDictSize, filenameTable, filenameIdx, &dictParams, NULL, 0); } #endif goto _end; diff --git a/tests/playTests.sh b/tests/playTests.sh index e98e0f44e..897a90155 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -281,6 +281,11 @@ case "$UNAME" in *) $MD5SUM -c tmph1 ;; esac rm -rf dirTestDict +$ECHO "- dictionary builder on bogus input" +$ECHO "Hello World" > tmp +$ZSTD --train -q tmp && die "Dictionary training should fail : not enough input source" +./datagen -P0 -g10M > tmp +$ZSTD --train -q tmp && die "Dictionary training should fail : source is pure noise" rm tmp* From 23776ce29022a68170e4cbea042f5616a4093507 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 23 Mar 2017 17:59:50 -0700 Subject: [PATCH 063/305] fixed ERROR_GENERIC on dstSize_tooSmall required by users which depends on this error code to size dest buffer --- lib/compress/fse_compress.c | 2 +- lib/compress/zstd_compress.c | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/lib/compress/fse_compress.c b/lib/compress/fse_compress.c index 6708fb9d7..e520e802b 100644 --- a/lib/compress/fse_compress.c +++ b/lib/compress/fse_compress.c @@ -291,7 +291,7 @@ static size_t FSE_writeNCount_generic (void* header, size_t headerBufferSize, size_t FSE_writeNCount (void* buffer, size_t bufferSize, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog) { - if (tableLog > FSE_MAX_TABLELOG) return ERROR(GENERIC); /* Unsupported */ + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); /* Unsupported */ if (tableLog < FSE_MIN_TABLELOG) return ERROR(GENERIC); /* Unsupported */ if (bufferSize < FSE_NCountWriteBound(maxSymbolValue, tableLog)) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 1aab3553f..b37425dad 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -653,7 +653,7 @@ MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc, if (count[llCodeTable[nbSeq-1]]>1) { count[llCodeTable[nbSeq-1]]--; nbSeq_1--; } FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ - if (FSE_isError(NCountSize)) return ERROR(GENERIC); + if (FSE_isError(NCountSize)) return NCountSize; op += NCountSize; } FSE_buildCTable_wksp(CTable_LitLength, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); LLtype = set_compressed; @@ -677,7 +677,7 @@ MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc, if (count[ofCodeTable[nbSeq-1]]>1) { count[ofCodeTable[nbSeq-1]]--; nbSeq_1--; } FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ - if (FSE_isError(NCountSize)) return ERROR(GENERIC); + if (FSE_isError(NCountSize)) return NCountSize; op += NCountSize; } FSE_buildCTable_wksp(CTable_OffsetBits, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); Offtype = set_compressed; @@ -701,7 +701,7 @@ MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc, if (count[mlCodeTable[nbSeq-1]]>1) { count[mlCodeTable[nbSeq-1]]--; nbSeq_1--; } FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ - if (FSE_isError(NCountSize)) return ERROR(GENERIC); + if (FSE_isError(NCountSize)) return NCountSize; op += NCountSize; } FSE_buildCTable_wksp(CTable_MatchLength, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); MLtype = set_compressed; From 16a0b10781a62dfbd8ed137d435b5022f389f955 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 24 Mar 2017 12:46:46 -0700 Subject: [PATCH 064/305] fixed ZSTD_loadZstdDictionary() forgot to add the dictionary content (tests were not failing, just compressing less). Also : added size protections when adding dict content since hc/bt table filling would fail if size < 8 --- lib/compress/zstd_compress.c | 43 ++++++++++++++++++++---------------- 1 file changed, 24 insertions(+), 19 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index eecc76950..c1ef526b4 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -2504,7 +2504,9 @@ size_t ZSTD_compressBlock(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 0, 0); } - +/*! ZSTD_loadDictionaryContent() : + * @return : 0, or an error code + */ static size_t ZSTD_loadDictionaryContent(ZSTD_CCtx* zc, const void* src, size_t srcSize) { const BYTE* const ip = (const BYTE*) src; @@ -2534,13 +2536,15 @@ static size_t ZSTD_loadDictionaryContent(ZSTD_CCtx* zc, const void* src, size_t case ZSTD_greedy: case ZSTD_lazy: case ZSTD_lazy2: - ZSTD_insertAndFindFirstIndex (zc, iend-HASH_READ_SIZE, zc->params.cParams.searchLength); + if (srcSize >= HASH_READ_SIZE) + ZSTD_insertAndFindFirstIndex(zc, iend-HASH_READ_SIZE, zc->params.cParams.searchLength); break; case ZSTD_btlazy2: case ZSTD_btopt: case ZSTD_btopt2: - ZSTD_updateTree(zc, iend-HASH_READ_SIZE, iend, 1 << zc->params.cParams.searchLog, zc->params.cParams.searchLength); + if (srcSize >= HASH_READ_SIZE) + ZSTD_updateTree(zc, iend-HASH_READ_SIZE, iend, 1 << zc->params.cParams.searchLog, zc->params.cParams.searchLength); break; default: @@ -2570,12 +2574,12 @@ static size_t ZSTD_checkDictNCount(short* normalizedCounter, unsigned dictMaxSym * See : * https://github.com/facebook/zstd/blob/master/doc/zstd_compression_format.md#dictionary-format */ -/*! ZSTD_loadDictionary() : - * @return : size read from dictionary - * note : magic number supposed already checked - * dictSize supposed > 8 +/*! ZSTD_loadZstdDictionary() : + * @return : 0, or an error code + * assumptions : magic number supposed already checked + * dictSize supposed > 8 */ -static size_t ZSTD_loadDictionary(ZSTD_CCtx* cctx, const void* dict, size_t dictSize) +static size_t ZSTD_loadZstdDictionary(ZSTD_CCtx* cctx, const void* dict, size_t dictSize) { const BYTE* dictPtr = (const BYTE*)dict; const BYTE* const dictEnd = dictPtr + dictSize; @@ -2624,9 +2628,9 @@ static size_t ZSTD_loadDictionary(ZSTD_CCtx* cctx, const void* dict, size_t dict } if (dictPtr+12 > dictEnd) return ERROR(dictionary_corrupted); - cctx->rep[0] = MEM_readLE32(dictPtr+0); if (cctx->rep[0] == 0 || cctx->rep[0] >= dictSize) return ERROR(dictionary_corrupted); - cctx->rep[1] = MEM_readLE32(dictPtr+4); if (cctx->rep[1] == 0 || cctx->rep[1] >= dictSize) return ERROR(dictionary_corrupted); - cctx->rep[2] = MEM_readLE32(dictPtr+8); if (cctx->rep[2] == 0 || cctx->rep[2] >= dictSize) return ERROR(dictionary_corrupted); + cctx->rep[0] = MEM_readLE32(dictPtr+0); + cctx->rep[1] = MEM_readLE32(dictPtr+4); + cctx->rep[2] = MEM_readLE32(dictPtr+8); dictPtr += 12; { size_t const dictContentSize = (size_t)(dictEnd - dictPtr); @@ -2642,25 +2646,26 @@ static size_t ZSTD_loadDictionary(ZSTD_CCtx* cctx, const void* dict, size_t dict for (u=0; u<3; u++) { if (cctx->rep[u] == 0) return ERROR(dictionary_corrupted); if (cctx->rep[u] > dictContentSize) return ERROR(dictionary_corrupted); - } } } + } } - cctx->flagStaticTables = 1; - cctx->flagStaticHufTable = HUF_repeat_valid; - return dictPtr - (const BYTE*)dict; + cctx->flagStaticTables = 1; + cctx->flagStaticHufTable = HUF_repeat_valid; + return ZSTD_loadDictionaryContent(cctx, dictPtr, dictContentSize); + } } /** ZSTD_compress_insertDictionary() : * @return : 0, or an error code */ -static size_t ZSTD_compress_insertDictionary(ZSTD_CCtx* zc, const void* dict, size_t dictSize) +static size_t ZSTD_compress_insertDictionary(ZSTD_CCtx* cctx, const void* dict, size_t dictSize) { if ((dict==NULL) || (dictSize<=8)) return 0; /* dict as pure content */ - if ((MEM_readLE32(dict) != ZSTD_DICT_MAGIC) || (zc->forceRawDict)) - return ZSTD_loadDictionaryContent(zc, dict, dictSize); + if ((MEM_readLE32(dict) != ZSTD_DICT_MAGIC) || (cctx->forceRawDict)) + return ZSTD_loadDictionaryContent(cctx, dict, dictSize); /* dict as zstd dictionary */ - return ZSTD_loadDictionary(zc, dict, dictSize); + return ZSTD_loadZstdDictionary(cctx, dict, dictSize); } /*! ZSTD_compressBegin_internal() : From 8d2c63980e698f64df7ff16a8f5c8f03f36c0cb8 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Fri, 24 Mar 2017 11:45:02 -0700 Subject: [PATCH 065/305] Statically build and link zlib on appveyor --- appveyor.yml | 20 ++++++++++++++------ 1 file changed, 14 insertions(+), 6 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index 7239e2e13..a53bce708 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -11,15 +11,18 @@ PLATFORM: "x64" SCRIPT: "make allarch && make -C tests test-symbols fullbench-dll fullbench-lib" ARTIFACT: "true" + BUILD: "true" - COMPILER: "gcc" HOST: "mingw" PLATFORM: "x86" SCRIPT: "make lib && make zstd && make -C tests allnothread" ARTIFACT: "true" + BUILD: "true" - COMPILER: "clang" HOST: "mingw" PLATFORM: "x64" SCRIPT: "MOREFLAGS='--target=x86_64-w64-mingw32 -Werror -Wconversion -Wno-sign-conversion' make allarch" + BUILD: "true" - COMPILER: "gcc" HOST: "mingw" @@ -69,10 +72,19 @@ SET "PATH=%PATH_MINGW64%;%PATH_ORIGINAL%" ) else if [%PLATFORM%]==[x86] ( SET "PATH=%PATH_MINGW32%;%PATH_ORIGINAL%" - ) ) && + ) ) + ) + - if [%HOST%]==[mingw] if [%BUILD%]==[true] ( make -v && sh -c "%COMPILER% -v" && - sh -c "CC=%COMPILER% %SCRIPT%" && + ECHO Building zlib to static link && + SET "CC=%COMPILER%" && + sh -c "cd .. && git clone --depth 1 --branch v1.2.11 https://github.com/madler/zlib" && + sh -c "cd ../zlib && make -f win32/Makefile.gcc libz.a" + ECHO Building zstd && + SET "CPPFLAGS=-I../../zlib" && + SET "LDFLAGS=../../zlib/libz.a" && + sh -c "%SCRIPT%" && ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] COPY programs\zstd.exe zstd_%PLATFORM%.exe && appveyor PushArtifact zstd_%PLATFORM%.exe ) ) - if [%HOST%]==[visual] ( @@ -158,10 +170,6 @@ - version: 1.0.{build} - branches: - except: - - dev - - master environment: matrix: - COMPILER: "gcc" From 9da3b215ec5d088da3c7a40d1779b076cdd2a26b Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 24 Mar 2017 15:02:09 -0700 Subject: [PATCH 066/305] Ensure all limits derived from same constants Now uses ZDICT_DICTSIZE_MIN and ZDICT_CONTENTSIZE_MIN from zdict.h. Also : reduced values to 256 and 128 respectively --- lib/dictBuilder/zdict.c | 19 ++++++++----------- lib/dictBuilder/zdict.h | 12 ++++++------ 2 files changed, 14 insertions(+), 17 deletions(-) diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index 842167db6..ed53197ae 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -11,8 +11,9 @@ /*-************************************** * Tuning parameters ****************************************/ +#define MINRATIO 4 /* minimum nb of apparition to be selected in dictionary */ #define ZDICT_MAX_SAMPLES_SIZE (2000U << 20) -#define ZDICT_MIN_SAMPLES_SIZE 512 +#define ZDICT_MIN_SAMPLES_SIZE (ZDICT_CONTENTSIZE_MIN * MINRATIO) /*-************************************** @@ -59,12 +60,8 @@ #define NOISELENGTH 32 -#define MINRATIO 4 static const int g_compressionLevel_default = 6; static const U32 g_selectivity_default = 9; -static const size_t g_provision_entropySize = 192; -static const size_t g_min_fast_dictContent = 192; -static const size_t g_dictContentSize_min = 32; /*-************************************* @@ -930,7 +927,7 @@ size_t ZDICT_trainFromBuffer_unsafe( /* checks */ if (!dictList) return ERROR(memory_allocation); - if (maxDictSize <= g_provision_entropySize + g_min_fast_dictContent) { free(dictList); return ERROR(dstSize_tooSmall); } /* requested dictionary size is too small */ + if (maxDictSize < ZDICT_DICTSIZE_MIN) { free(dictList); return ERROR(dstSize_tooSmall); } /* requested dictionary size is too small */ if (samplesBuffSize < ZDICT_MIN_SAMPLES_SIZE) { free(dictList); return ERROR(dictionaryCreation_failed); } /* not enough source to create dictionary */ /* init */ @@ -964,15 +961,15 @@ size_t ZDICT_trainFromBuffer_unsafe( /* create dictionary */ { U32 dictContentSize = ZDICT_dictSize(dictList); - if (dictContentSize < g_dictContentSize_min) { free(dictList); return ERROR(dictionaryCreation_failed); } /* dictionary content too small */ - if (dictContentSize < targetDictSize/3) { + if (dictContentSize < ZDICT_CONTENTSIZE_MIN) { free(dictList); return ERROR(dictionaryCreation_failed); } /* dictionary content too small */ + if (dictContentSize < targetDictSize/4) { DISPLAYLEVEL(2, "! warning : selected content significantly smaller than requested (%u < %u) \n", dictContentSize, (U32)maxDictSize); + if (samplesBuffSize < 10 * targetDictSize) + DISPLAYLEVEL(2, "! consider increasing the number of samples (total size : %u MB)\n", (U32)(samplesBuffSize>>20)); if (minRep > MINRATIO) { DISPLAYLEVEL(2, "! consider increasing selectivity to produce larger dictionary (-s%u) \n", selectivity+1); DISPLAYLEVEL(2, "! note : larger dictionaries are not necessarily better, test its efficiency on samples \n"); } - if (samplesBuffSize < 10 * targetDictSize) - DISPLAYLEVEL(2, "! consider increasing the number of samples (total size : %u MB)\n", (U32)(samplesBuffSize>>20)); } if ((dictContentSize > targetDictSize*3) && (nbSamples > 2*MINRATIO) && (selectivity>1)) { @@ -980,7 +977,7 @@ size_t ZDICT_trainFromBuffer_unsafe( while ((nbSamples >> proposedSelectivity) <= MINRATIO) { proposedSelectivity--; } DISPLAYLEVEL(2, "! note : calculated dictionary significantly larger than requested (%u > %u) \n", dictContentSize, (U32)maxDictSize); DISPLAYLEVEL(2, "! consider increasing dictionary size, or produce denser dictionary (-s%u) \n", proposedSelectivity); - DISPLAYLEVEL(2, "! always test dictionary efficiency on samples \n"); + DISPLAYLEVEL(2, "! always test dictionary efficiency on real samples \n"); } /* limit dictionary size */ diff --git a/lib/dictBuilder/zdict.h b/lib/dictBuilder/zdict.h index 4ead4474f..669b78d08 100644 --- a/lib/dictBuilder/zdict.h +++ b/lib/dictBuilder/zdict.h @@ -147,18 +147,18 @@ ZDICTLIB_API size_t COVER_optimizeTrainFromBuffer(void* dictBuffer, size_t dictB Samples must be stored concatenated in a flat buffer `samplesBuffer`, supplied with an array of sizes `samplesSizes`, providing the size of each sample in order. - dictContentSize must be > ZDICT_CONTENTSIZE_MIN bytes. - maxDictSize must be >= dictContentSize, and must be > ZDICT_DICTSIZE_MIN bytes. + dictContentSize must be >= ZDICT_CONTENTSIZE_MIN bytes. + maxDictSize must be >= dictContentSize, and must be >= ZDICT_DICTSIZE_MIN bytes. @return : size of dictionary stored into `dictBuffer` (<= `dictBufferCapacity`), or an error code, which can be tested by ZDICT_isError(). note : ZDICT_finalizeDictionary() will push notifications into stderr if instructed to, using notificationLevel>0. - note 2 : dictBuffer and customDictContent can overlap + note 2 : dictBuffer and dictContent can overlap */ -#define ZDICT_CONTENTSIZE_MIN 256 -#define ZDICT_DICTSIZE_MIN 512 +#define ZDICT_CONTENTSIZE_MIN 128 +#define ZDICT_DICTSIZE_MIN 256 ZDICTLIB_API size_t ZDICT_finalizeDictionary(void* dictBuffer, size_t dictBufferCapacity, - const void* customDictContent, size_t dictContentSize, + const void* dictContent, size_t dictContentSize, const void* samplesBuffer, const size_t* samplesSizes, unsigned nbSamples, ZDICT_params_t parameters); From 96aa3019b25bfa6c016e935e8f4e7a4944b766d8 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 24 Mar 2017 16:04:29 -0700 Subject: [PATCH 067/305] changed advanced commands --maxdict= and --dictID= now works with the `=` variant, which is the recommended one. Old variant `--dictID #` still works, for compatibility with existing scripts. Long term objective is to remove the old variant.. --- lib/dictBuilder/zdict.c | 17 ++++++++--------- programs/zstd.1 | 4 ++-- programs/zstd.1.md | 4 ++-- programs/zstdcli.c | 14 ++++++++------ 4 files changed, 20 insertions(+), 19 deletions(-) diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index ed53197ae..0824b94d1 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -361,11 +361,11 @@ static dictItem ZDICT_analyzePos( } -/*! ZDICT_checkMerge - check if dictItem can be merged, do it if possible - @return : id of destination elt, 0 if not merged -*/ -static U32 ZDICT_checkMerge(dictItem* table, dictItem elt, U32 eltNbToSkip) +/*! ZDICT_tryMerge() : + * check if dictItem can be merged, do it if possible + * @return : id of destination elt, 0 if not merged + */ +static U32 ZDICT_tryMerge(dictItem* table, dictItem elt, U32 eltNbToSkip) { const U32 tableSize = table->pos; const U32 eltEnd = elt.pos + elt.length; @@ -426,11 +426,11 @@ static void ZDICT_removeDictItem(dictItem* table, U32 id) static void ZDICT_insertDictItem(dictItem* table, U32 maxSize, dictItem elt) { /* merge if possible */ - U32 mergeId = ZDICT_checkMerge(table, elt, 0); + U32 mergeId = ZDICT_tryMerge(table, elt, 0); if (mergeId) { U32 newMerge = 1; while (newMerge) { - newMerge = ZDICT_checkMerge(table, table[mergeId], mergeId); + newMerge = ZDICT_tryMerge(table, table[mergeId], mergeId); /* merge existing elt */ if (newMerge) ZDICT_removeDictItem(table, mergeId); mergeId = newMerge; } @@ -810,7 +810,6 @@ static size_t ZDICT_analyzeEntropy(void* dstBuffer, size_t maxDstSize, MEM_writeLE32(dstPtr+4, repStartValue[1]); MEM_writeLE32(dstPtr+8, repStartValue[2]); #endif - //dstPtr += 12; eSize += 12; _cleanup: @@ -829,7 +828,7 @@ size_t ZDICT_finalizeDictionary(void* dictBuffer, size_t dictBufferCapacity, ZDICT_params_t params) { size_t hSize; -#define HBUFFSIZE 256 +#define HBUFFSIZE 256 /* should prove large enough for all entropy headers */ BYTE header[HBUFFSIZE]; int const compressionLevel = (params.compressionLevel <= 0) ? g_compressionLevel_default : params.compressionLevel; U32 const notificationLevel = params.notificationLevel; diff --git a/programs/zstd.1 b/programs/zstd.1 index 02423358d..8b418a0ee 100644 --- a/programs/zstd.1 +++ b/programs/zstd.1 @@ -168,11 +168,11 @@ use FILEs as training set to create a dictionary\. The training set should conta dictionary saved into \fBfile\fR (default: dictionary) . .TP -\fB\-\-maxdict #\fR +\fB\-\-maxdict=#\fR limit dictionary to specified size (default : (112640) . .TP -\fB\-\-dictID #\fR +\fB\-\-dictID=#\fR A dictionary ID is a locally unique ID that a decoder can use to verify it is using the right dictionary\. By default, zstd will create a 4\-bytes random number ID\. It\'s possible to give a precise number instead\. Short numbers have an advantage : an ID < 256 will only need 1 byte in the compressed frame header, and an ID < 65536 will only need 2 bytes\. This compares favorably to 4 bytes default\. However, it\'s up to the dictionary manager to not assign twice the same ID to 2 different dictionaries\. . .TP diff --git a/programs/zstd.1.md b/programs/zstd.1.md index c9ff33276..d1161a522 100644 --- a/programs/zstd.1.md +++ b/programs/zstd.1.md @@ -157,9 +157,9 @@ Typical gains range from 10% (at 64KB) to x5 better (at <1KB). (for example, 10 MB for a 100 KB dictionary). * `-o file`: dictionary saved into `file` (default: dictionary) -* `--maxdict #`: +* `--maxdict=#`: limit dictionary to specified size (default : (112640) -* `--dictID #`: +* `--dictID=#`: A dictionary ID is a locally unique ID that a decoder can use to verify it is using the right dictionary. By default, zstd will create a 4-bytes random number ID. diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 281301bd1..a35485085 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -148,9 +148,9 @@ static int usage_advanced(const char* programName) DISPLAY( "--cover=k=#,d=# : use the cover algorithm with parameters k and d \n"); DISPLAY( "--optimize-cover[=steps=#,k=#,d=#] : optimize cover parameters with optional parameters\n"); DISPLAY( " -o file : `file` is dictionary name (default: %s) \n", g_defaultDictName); - DISPLAY( "--maxdict ## : limit dictionary to specified size (default : %u) \n", g_defaultMaxDictSize); + DISPLAY( "--maxdict=# : limit dictionary to specified size (default : %u) \n", g_defaultMaxDictSize); DISPLAY( " -s# : dictionary selectivity level (default: %u)\n", g_defaultSelectivityLevel); - DISPLAY( "--dictID ## : force dictionary ID to specified value (default: random)\n"); + DISPLAY( "--dictID=# : force dictionary ID to specified value (default: random)\n"); #endif #ifndef ZSTD_NOBENCH DISPLAY( "\n"); @@ -371,8 +371,8 @@ int main(int argCount, const char* argv[]) if (!strcmp(argument, "--no-sparse")) { FIO_setSparseWrite(0); continue; } if (!strcmp(argument, "--test")) { operation=zom_test; continue; } if (!strcmp(argument, "--train")) { operation=zom_train; outFileName=g_defaultDictName; continue; } - if (!strcmp(argument, "--maxdict")) { nextArgumentIsMaxDict=1; lastCommand=1; continue; } - if (!strcmp(argument, "--dictID")) { nextArgumentIsDictID=1; lastCommand=1; continue; } + if (!strcmp(argument, "--maxdict")) { nextArgumentIsMaxDict=1; lastCommand=1; continue; } /* kept available for compatibility with old syntax ; will be removed one day */ + if (!strcmp(argument, "--dictID")) { nextArgumentIsDictID=1; lastCommand=1; continue; } /* kept available for compatibility with old syntax ; will be removed one day */ if (!strcmp(argument, "--no-dictID")) { FIO_setDictIDFlag(0); continue; } if (!strcmp(argument, "--keep")) { FIO_setRemoveSrcFile(0); continue; } if (!strcmp(argument, "--rm")) { FIO_setRemoveSrcFile(1); continue; } @@ -404,6 +404,8 @@ int main(int argCount, const char* argv[]) if (longCommandWArg(&argument, "--memory=")) { memLimit = readU32FromChar(&argument); continue; } if (longCommandWArg(&argument, "--memlimit-decompress=")) { memLimit = readU32FromChar(&argument); continue; } if (longCommandWArg(&argument, "--block-size=")) { blockSize = readU32FromChar(&argument); continue; } + if (longCommandWArg(&argument, "--maxdict=")) { maxDictSize = readU32FromChar(&argument); continue; } + if (longCommandWArg(&argument, "--dictID=")) { dictID = readU32FromChar(&argument); continue; } if (longCommandWArg(&argument, "--zstd=")) { if (!parseCompressionParameters(argument, &compressionParams)) CLEAN_RETURN(badusage(programName)); continue; } /* fall-through, will trigger bad_usage() later on */ } @@ -533,14 +535,14 @@ int main(int argCount, const char* argv[]) continue; } /* if (argument[0]=='-') */ - if (nextArgumentIsMaxDict) { + if (nextArgumentIsMaxDict) { /* kept available for compatibility with old syntax ; will be removed one day */ nextArgumentIsMaxDict = 0; lastCommand = 0; maxDictSize = readU32FromChar(&argument); continue; } - if (nextArgumentIsDictID) { + if (nextArgumentIsDictID) { /* kept available for compatibility with old syntax ; will be removed one day */ nextArgumentIsDictID = 0; lastCommand = 0; dictID = readU32FromChar(&argument); From ef30af843fb6120cb89424ffbe90e184dfd017b2 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Fri, 24 Mar 2017 17:06:09 -0700 Subject: [PATCH 068/305] Ignore extension in command name matching --- programs/zstdcli.c | 38 ++++++++++++++++++++++++++------------ 1 file changed, 26 insertions(+), 12 deletions(-) diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 281301bd1..cd7ea193c 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -179,6 +179,23 @@ static void waitEnter(void) (void)unused; } +static const char* lastNameFromPath(const char* path) +{ + const char* name = path; + if (strrchr(name, '/')) name = strrchr(name, '/') + 1; + if (strrchr(name, '\\')) name = strrchr(name, '\\') + 1; /* windows */ + return name; +} + +/*! exeNameMatch() : + @return : a non-zero value if exeName matches test, excluding the extension + */ +static int exeNameMatch(const char* exeName, const char* test) +{ + return !strncmp(exeName, test, strlen(test)) && + (exeName[strlen(test)] == '\0' || exeName[strlen(test)] == '.'); +} + /*! readU32FromChar() : @return : unsigned integer value read from input in `char` format allows and interprets K, KB, KiB, M, MB and MiB suffix. @@ -318,20 +335,17 @@ int main(int argCount, const char* argv[]) if (filenameTable==NULL) { DISPLAY("zstd: %s \n", strerror(errno)); exit(1); } filenameTable[0] = stdinmark; g_displayOut = stderr; - /* Pick out program name from path. Don't rely on stdlib because of conflicting behavior */ - { size_t pos; - for (pos = (int)strlen(programName); pos > 0; pos--) { if (programName[pos] == '/') { pos++; break; } } - programName += pos; - } + + programName = lastNameFromPath(programName); /* preset behaviors */ - if (!strcmp(programName, ZSTD_UNZSTD)) operation=zom_decompress; - if (!strcmp(programName, ZSTD_CAT)) { operation=zom_decompress; forceStdout=1; FIO_overwriteMode(); outFileName=stdoutmark; g_displayLevel=1; } - if (!strcmp(programName, ZSTD_GZ)) { suffix = GZ_EXTENSION; FIO_setCompressionType(FIO_gzipCompression); FIO_setRemoveSrcFile(1); } /* behave like gzip */ - if (!strcmp(programName, ZSTD_GUNZIP)) { operation=zom_decompress; FIO_setRemoveSrcFile(1); } /* behave like gunzip */ - if (!strcmp(programName, ZSTD_GZCAT)) { operation=zom_decompress; forceStdout=1; FIO_overwriteMode(); outFileName=stdoutmark; g_displayLevel=1; } /* behave like gzcat */ - if (!strcmp(programName, ZSTD_LZMA)) { suffix = LZMA_EXTENSION; FIO_setCompressionType(FIO_lzmaCompression); FIO_setRemoveSrcFile(1); } /* behave like lzma */ - if (!strcmp(programName, ZSTD_XZ)) { suffix = XZ_EXTENSION; FIO_setCompressionType(FIO_xzCompression); FIO_setRemoveSrcFile(1); } /* behave like xz */ + if (exeNameMatch(programName, ZSTD_UNZSTD)) operation=zom_decompress; + if (exeNameMatch(programName, ZSTD_CAT)) { operation=zom_decompress; forceStdout=1; FIO_overwriteMode(); outFileName=stdoutmark; g_displayLevel=1; } + if (exeNameMatch(programName, ZSTD_GZ)) { suffix = GZ_EXTENSION; FIO_setCompressionType(FIO_gzipCompression); FIO_setRemoveSrcFile(1); } /* behave like gzip */ + if (exeNameMatch(programName, ZSTD_GUNZIP)) { operation=zom_decompress; FIO_setRemoveSrcFile(1); } /* behave like gunzip */ + if (exeNameMatch(programName, ZSTD_GZCAT)) { operation=zom_decompress; forceStdout=1; FIO_overwriteMode(); outFileName=stdoutmark; g_displayLevel=1; } /* behave like gzcat */ + if (exeNameMatch(programName, ZSTD_LZMA)) { suffix = LZMA_EXTENSION; FIO_setCompressionType(FIO_lzmaCompression); FIO_setRemoveSrcFile(1); } /* behave like lzma */ + if (exeNameMatch(programName, ZSTD_XZ)) { suffix = XZ_EXTENSION; FIO_setCompressionType(FIO_xzCompression); FIO_setRemoveSrcFile(1); } /* behave like xz */ memset(&compressionParams, 0, sizeof(compressionParams)); /* command switches */ From d41f707e884f54bd2662b759b2bb96c55f9a12f0 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 24 Mar 2017 17:56:45 -0700 Subject: [PATCH 069/305] minor improvement : remove duplicates with 1 char prefix difference --- lib/dictBuilder/zdict.c | 38 ++++++++++++++++++++++++++++++++------ 1 file changed, 32 insertions(+), 6 deletions(-) diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index ed53197ae..1b3c74f16 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -361,14 +361,28 @@ static dictItem ZDICT_analyzePos( } +static int isIncluded(const void* in, const void* container, size_t length) +{ + const char* const ip = (const char*) in; + const char* const into = (const char*) container; + size_t u; + + for (u=0; upos; const U32 eltEnd = elt.pos + elt.length; + const char* const buf = buffer; /* tail overlap */ U32 u; for (u=1; u= elt.pos) && (table[u].pos < elt.pos)) { /* overlap, existing < new */ /* append */ int addedLength = (int)eltEnd - (table[u].pos + table[u].length); @@ -405,7 +420,18 @@ static U32 ZDICT_checkMerge(dictItem* table, dictItem elt, U32 eltNbToSkip) table[u] = table[u-1], u--; table[u] = elt; return u; - } } + } + + if (MEM_read64(buf + table[u].pos) == MEM_read64(buf + elt.pos + 1)) { + if (isIncluded(buf + table[u].pos, buf + elt.pos + 1, table[u].length)) { + size_t const addedLength = MAX( (int)elt.length - (int)table[u].length , 1 ); + table[u].pos = elt.pos; + table[u].savings += elt.savings * addedLength / elt.length; + table[u].length = MIN(elt.length, table[u].length + 1); + return u; + } + } + } return 0; } @@ -423,14 +449,14 @@ static void ZDICT_removeDictItem(dictItem* table, U32 id) } -static void ZDICT_insertDictItem(dictItem* table, U32 maxSize, dictItem elt) +static void ZDICT_insertDictItem(dictItem* table, U32 maxSize, dictItem elt, const void* buffer) { /* merge if possible */ - U32 mergeId = ZDICT_checkMerge(table, elt, 0); + U32 mergeId = ZDICT_checkMerge(table, elt, 0, buffer); if (mergeId) { U32 newMerge = 1; while (newMerge) { - newMerge = ZDICT_checkMerge(table, table[mergeId], mergeId); + newMerge = ZDICT_checkMerge(table, table[mergeId], mergeId, buffer); if (newMerge) ZDICT_removeDictItem(table, mergeId); mergeId = newMerge; } @@ -519,7 +545,7 @@ static size_t ZDICT_trainBuffer(dictItem* dictList, U32 dictListSize, if (doneMarks[cursor]) { cursor++; continue; } solution = ZDICT_analyzePos(doneMarks, suffix, reverseSuffix[cursor], buffer, minRatio, notificationLevel); if (solution.length==0) { cursor++; continue; } - ZDICT_insertDictItem(dictList, dictListSize, solution); + ZDICT_insertDictItem(dictList, dictListSize, solution, buffer); cursor += solution.length; DISPLAYUPDATE(2, "\r%4.2f %% \r", (double)cursor / bufferSize * 100); } } From 4c41d37fcca57130127ca84f89e8cec401d718d0 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 24 Mar 2017 18:36:56 -0700 Subject: [PATCH 070/305] changed test for new syntax --dictID= and --maxdict= --- lib/dictBuilder/zdict.c | 2 +- tests/playTests.sh | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index 0824b94d1..6bc42526a 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -427,7 +427,7 @@ static void ZDICT_insertDictItem(dictItem* table, U32 maxSize, dictItem elt) { /* merge if possible */ U32 mergeId = ZDICT_tryMerge(table, elt, 0); - if (mergeId) { + if (mergeId) { /* recursive : re-merge the newly merged elt */ U32 newMerge = 1; while (newMerge) { newMerge = ZDICT_tryMerge(table, table[mergeId], mergeId); /* merge existing elt */ diff --git a/tests/playTests.sh b/tests/playTests.sh index 897a90155..5abbb14e3 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -253,12 +253,12 @@ $ECHO "- Create second (different) dictionary " $ZSTD --train *.c ../programs/*.c ../programs/*.h -o tmpDictC $ZSTD -d tmp.zst -D tmpDictC -fo result && die "wrong dictionary not detected!" $ECHO "- Create dictionary with short dictID" -$ZSTD --train *.c ../programs/*.c --dictID 1 -o tmpDict1 +$ZSTD --train *.c ../programs/*.c --dictID=1 -o tmpDict1 cmp tmpDict tmpDict1 && die "dictionaries should have different ID !" $ECHO "- Create dictionary with wrong dictID parameter order (must fail)" $ZSTD --train *.c ../programs/*.c --dictID -o 1 tmpDict1 && die "wrong order : --dictID must be followed by argument " $ECHO "- Create dictionary with size limit" -$ZSTD --train *.c ../programs/*.c -o tmpDict2 --maxdict 4K -v +$ZSTD --train *.c ../programs/*.c -o tmpDict2 --maxdict=4K -v $ECHO "- Create dictionary with wrong parameter order (must fail)" $ZSTD --train *.c ../programs/*.c -o tmpDict2 --maxdict -v 4K && die "wrong order : --maxdict must be followed by argument " $ECHO "- Compress without dictID" @@ -303,10 +303,10 @@ $ECHO "- Create second (different) dictionary" $ZSTD --train --cover=k=56,d=8 *.c ../programs/*.c ../programs/*.h -o tmpDictC $ZSTD -d tmp.zst -D tmpDictC -fo result && die "wrong dictionary not detected!" $ECHO "- Create dictionary with short dictID" -$ZSTD --train --cover=k=46,d=8 *.c ../programs/*.c --dictID 1 -o tmpDict1 +$ZSTD --train --cover=k=46,d=8 *.c ../programs/*.c --dictID=1 -o tmpDict1 cmp tmpDict tmpDict1 && die "dictionaries should have different ID !" $ECHO "- Create dictionary with size limit" -$ZSTD --train --optimize-cover=steps=8 *.c ../programs/*.c -o tmpDict2 --maxdict 4K +$ZSTD --train --optimize-cover=steps=8 *.c ../programs/*.c -o tmpDict2 --maxdict=4K rm tmp* From ecee9f2ef8ec41ad5fa5bf8f23f430d094722e52 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Sun, 26 Mar 2017 00:59:14 -0700 Subject: [PATCH 071/305] fixed conversion warnings --- lib/dictBuilder/zdict.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index 1b3c74f16..955d98aa6 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -382,7 +382,7 @@ static U32 ZDICT_checkMerge(dictItem* table, dictItem elt, U32 eltNbToSkip, cons { const U32 tableSize = table->pos; const U32 eltEnd = elt.pos + elt.length; - const char* const buf = buffer; + const char* const buf = (const char*) buffer; /* tail overlap */ U32 u; for (u=1; u Date: Sun, 26 Mar 2017 02:50:00 -0700 Subject: [PATCH 072/305] fixed dictBuilder issue dictionary loading would fail during entropy analysis --- lib/dictBuilder/zdict.c | 51 +++++++++++++++++++---------------------- 1 file changed, 24 insertions(+), 27 deletions(-) diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index 955d98aa6..0cd9cc88e 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -393,7 +393,7 @@ static U32 ZDICT_checkMerge(dictItem* table, dictItem elt, U32 eltNbToSkip, cons table[u].length += addedLength; table[u].pos = elt.pos; table[u].savings += elt.savings * addedLength / elt.length; /* rough approx */ - table[u].savings += elt.length / 8; /* rough approx bonus */ + // table[u].savings += elt.length / 8; /* rough approx bonus */ elt = table[u]; /* sort : improve rank */ while ((u>1) && (table[u-1].savings < elt.savings)) @@ -422,14 +422,13 @@ static U32 ZDICT_checkMerge(dictItem* table, dictItem elt, U32 eltNbToSkip, cons return u; } - if (MEM_read64(buf + table[u].pos) == MEM_read64(buf + elt.pos + 1)) { - if (isIncluded(buf + table[u].pos, buf + elt.pos + 1, table[u].length)) { - size_t const addedLength = MAX( (int)elt.length - (int)table[u].length , 1 ); - table[u].pos = elt.pos; - table[u].savings += (U32)(elt.savings * addedLength / elt.length); - table[u].length = MIN(elt.length, table[u].length + 1); - return u; - } + if ( (MEM_read64(buf + table[u].pos) == MEM_read64(buf + elt.pos + 1)) + && (isIncluded(buf + table[u].pos, buf + elt.pos + 1, table[u].length)) ) { + size_t const addedLength = MAX( (int)elt.length - (int)table[u].length , 1 ); + table[u].pos = elt.pos; + table[u].savings += (U32)(elt.savings * addedLength / elt.length); + table[u].length = MIN(elt.length, table[u].length + 1); + return u; } } @@ -707,19 +706,19 @@ static size_t ZDICT_analyzeEntropy(void* dstBuffer, size_t maxDstSize, goto _cleanup; } if (offcodeMax>OFFCODE_MAX) { eSize = ERROR(dictionary_wrong); goto _cleanup; } /* too large dictionary */ - for (u=0; u<256; u++) countLit[u]=1; /* any character must be described */ - for (u=0; u<=offcodeMax; u++) offcodeCount[u]=1; - for (u=0; u<=MaxML; u++) matchLengthCount[u]=1; - for (u=0; u<=MaxLL; u++) litLengthCount[u]=1; + for (u=0; u<256; u++) countLit[u] = 1; /* any character must be described */ + for (u=0; u<=offcodeMax; u++) offcodeCount[u] = 1; + for (u=0; u<=MaxML; u++) matchLengthCount[u] = 1; + for (u=0; u<=MaxLL; u++) litLengthCount[u] = 1; memset(repOffset, 0, sizeof(repOffset)); repOffset[1] = repOffset[4] = repOffset[8] = 1; memset(bestRepOffset, 0, sizeof(bestRepOffset)); - if (compressionLevel==0) compressionLevel=g_compressionLevel_default; + if (compressionLevel==0) compressionLevel = g_compressionLevel_default; params = ZSTD_getParams(compressionLevel, averageSampleSize, dictBufferSize); { size_t const beginResult = ZSTD_compressBegin_advanced(esr.ref, dictBuffer, dictBufferSize, params, 0); - if (ZSTD_isError(beginResult)) { + if (ZSTD_isError(beginResult)) { + DISPLAYLEVEL(1, "error : ZSTD_compressBegin_advanced() failed : %s \n", ZSTD_getErrorName(beginResult)); eSize = ERROR(GENERIC); - DISPLAYLEVEL(1, "error : ZSTD_compressBegin_advanced failed \n"); goto _cleanup; } } @@ -901,20 +900,11 @@ size_t ZDICT_addEntropyTablesFromBuffer_advanced(void* dictBuffer, size_t dictCo const void* samplesBuffer, const size_t* samplesSizes, unsigned nbSamples, ZDICT_params_t params) { - size_t hSize; int const compressionLevel = (params.compressionLevel <= 0) ? g_compressionLevel_default : params.compressionLevel; U32 const notificationLevel = params.notificationLevel; + size_t hSize = 8; - /* dictionary header */ - MEM_writeLE32(dictBuffer, ZSTD_DICT_MAGIC); - { U64 const randomID = XXH64((char*)dictBuffer + dictBufferCapacity - dictContentSize, dictContentSize, 0); - U32 const compliantID = (randomID % ((1U<<31)-32768)) + 32768; - U32 const dictID = params.dictID ? params.dictID : compliantID; - MEM_writeLE32((char*)dictBuffer+4, dictID); - } - hSize = 8; - - /* entropy tables */ + /* calculate entropy tables */ DISPLAYLEVEL(2, "\r%70s\r", ""); /* clean display line */ DISPLAYLEVEL(2, "statistics ... \n"); { size_t const eSize = ZDICT_analyzeEntropy((char*)dictBuffer+hSize, dictBufferCapacity-hSize, @@ -926,6 +916,13 @@ size_t ZDICT_addEntropyTablesFromBuffer_advanced(void* dictBuffer, size_t dictCo hSize += eSize; } + /* add dictionary header (after entropy tables) */ + MEM_writeLE32(dictBuffer, ZSTD_DICT_MAGIC); + { U64 const randomID = XXH64((char*)dictBuffer + dictBufferCapacity - dictContentSize, dictContentSize, 0); + U32 const compliantID = (randomID % ((1U<<31)-32768)) + 32768; + U32 const dictID = params.dictID ? params.dictID : compliantID; + MEM_writeLE32((char*)dictBuffer+4, dictID); + } if (hSize + dictContentSize < dictBufferCapacity) memmove((char*)dictBuffer + hSize, (char*)dictBuffer + dictBufferCapacity - dictContentSize, dictContentSize); From 582760818f2573106dee003d779cafd4f46a7420 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Sun, 26 Mar 2017 02:59:13 -0700 Subject: [PATCH 073/305] minor refactor add const changed if for easier to add new conditions --- lib/dictBuilder/zdict.c | 19 ++++++++++--------- 1 file changed, 10 insertions(+), 9 deletions(-) diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index 0cd9cc88e..2c87084ac 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -389,7 +389,7 @@ static U32 ZDICT_checkMerge(dictItem* table, dictItem elt, U32 eltNbToSkip, cons if (u==eltNbToSkip) continue; if ((table[u].pos > elt.pos) && (table[u].pos <= eltEnd)) { /* overlap, existing > new */ /* append */ - U32 addedLength = table[u].pos - elt.pos; + U32 const addedLength = table[u].pos - elt.pos; table[u].length += addedLength; table[u].pos = elt.pos; table[u].savings += elt.savings * addedLength / elt.length; /* rough approx */ @@ -408,7 +408,7 @@ static U32 ZDICT_checkMerge(dictItem* table, dictItem elt, U32 eltNbToSkip, cons if ((table[u].pos + table[u].length >= elt.pos) && (table[u].pos < elt.pos)) { /* overlap, existing < new */ /* append */ - int addedLength = (int)eltEnd - (table[u].pos + table[u].length); + int const addedLength = (int)eltEnd - (table[u].pos + table[u].length); table[u].savings += elt.length / 8; /* rough approx bonus */ if (addedLength > 0) { /* otherwise, elt fully included into existing */ table[u].length += addedLength; @@ -422,13 +422,14 @@ static U32 ZDICT_checkMerge(dictItem* table, dictItem elt, U32 eltNbToSkip, cons return u; } - if ( (MEM_read64(buf + table[u].pos) == MEM_read64(buf + elt.pos + 1)) - && (isIncluded(buf + table[u].pos, buf + elt.pos + 1, table[u].length)) ) { - size_t const addedLength = MAX( (int)elt.length - (int)table[u].length , 1 ); - table[u].pos = elt.pos; - table[u].savings += (U32)(elt.savings * addedLength / elt.length); - table[u].length = MIN(elt.length, table[u].length + 1); - return u; + if (MEM_read64(buf + table[u].pos) == MEM_read64(buf + elt.pos + 1)) { + if (isIncluded(buf + table[u].pos, buf + elt.pos + 1, table[u].length)) { + size_t const addedLength = MAX( (int)elt.length - (int)table[u].length , 1 ); + table[u].pos = elt.pos; + table[u].savings += (U32)(elt.savings * addedLength / elt.length); + table[u].length = MIN(elt.length, table[u].length + 1); + return u; + } } } From 4cf0093571f2aece4ae6f5d1047d8633323aac95 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Sun, 26 Mar 2017 14:51:00 -0700 Subject: [PATCH 074/305] restored bonus rule --- lib/dictBuilder/zdict.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index cd2e3a2f5..bee5e059d 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -393,7 +393,7 @@ static U32 ZDICT_tryMerge(dictItem* table, dictItem elt, U32 eltNbToSkip, const table[u].length += addedLength; table[u].pos = elt.pos; table[u].savings += elt.savings * addedLength / elt.length; /* rough approx */ - // table[u].savings += elt.length / 8; /* rough approx bonus */ + table[u].savings += elt.length / 8; /* rough approx bonus */ elt = table[u]; /* sort : improve rank */ while ((u>1) && (table[u-1].savings < elt.savings)) From 41fefd573add4e4e92555c978504070b1f1123d1 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Sun, 26 Mar 2017 23:52:19 -0700 Subject: [PATCH 075/305] Improved speed tests Better compatibility with Mac OS-X Force attribution to a selected core # (Linux) --- tests/test-zstd-speed.py | 20 +++++++++++++------- 1 file changed, 13 insertions(+), 7 deletions(-) diff --git a/tests/test-zstd-speed.py b/tests/test-zstd-speed.py index 23d4f477c..56108a5ca 100755 --- a/tests/test-zstd-speed.py +++ b/tests/test-zstd-speed.py @@ -14,14 +14,15 @@ # - dir1/zstd and dir2/zstd will be merged in a single results file import argparse -import os +import os # getloadavg import string import subprocess -import time +import time # strftime import traceback import hashlib +import platform # system -script_version = 'v1.1.1 (2016-10-28)' +script_version = 'v1.1.2 (2017-03-26)' default_repo_url = 'https://github.com/facebook/zstd.git' working_dir_name = 'speedTest' working_path = os.getcwd() + '/' + working_dir_name # /path/to/zstd/tests/speedTest @@ -152,10 +153,15 @@ def benchmark_and_compare(branch, commit, last_commit, args, executableName, md5 % (os.getloadavg()[0], args.maxLoadAvg, sleepTime)) time.sleep(sleepTime) start_load = str(os.getloadavg()) - if args.dictionary: - result = execute('programs/%s -rqi5b1e%s -D %s %s' % (executableName, args.lastCLevel, args.dictionary, testFilePath), print_output=True) + osType = platform.system() + if osType == 'Linux': + cpuSelector = "taskset --cpu-list 0" else: - result = execute('programs/%s -rqi5b1e%s %s' % (executableName, args.lastCLevel, testFilePath), print_output=True) + cpuSelector = "" + if args.dictionary: + result = execute('%s programs/%s -rqi5b1e%s -D %s %s' % (cpuSelector, executableName, args.lastCLevel, args.dictionary, testFilePath), print_output=True) + else: + result = execute('%s programs/%s -rqi5b1e%s %s' % (cpuSelector, executableName, args.lastCLevel, testFilePath), print_output=True) end_load = str(os.getloadavg()) linesExpected = args.lastCLevel + 1 if len(result) != linesExpected: @@ -291,7 +297,7 @@ if __name__ == '__main__': log("ERROR: e-mail senders 'mail' or 'mutt' not found") exit(1) - clang_version = execute("clang -v 2>&1 | grep 'clang version' | sed -e 's:.*version \\([0-9.]*\\).*:\\1:' -e 's:\\.\\([0-9][0-9]\\):\\1:g'", verbose)[0]; + clang_version = execute("clang -v 2>&1 | grep ' version ' | sed -e 's:.*version \\([0-9.]*\\).*:\\1:' -e 's:\\.\\([0-9][0-9]\\):\\1:g'", verbose)[0]; gcc_version = execute("gcc -dumpversion", verbose)[0]; if verbose: From 894bf4971374be233df7e954aabb3b19116c1ac7 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 27 Mar 2017 12:19:30 -0700 Subject: [PATCH 076/305] Fix IS_CONSOLE returning 1 for NUL on windows --- programs/platform.h | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/programs/platform.h b/programs/platform.h index 89a9f6cd4..819bec873 100644 --- a/programs/platform.h +++ b/programs/platform.h @@ -100,9 +100,18 @@ extern "C" { #if (defined(__linux__) && (PLATFORM_POSIX_VERSION >= 1)) || (PLATFORM_POSIX_VERSION >= 200112L) || defined(__DJGPP__) # include /* isatty */ # define IS_CONSOLE(stdStream) isatty(fileno(stdStream)) -#elif defined(MSDOS) || defined(OS2) || defined(WIN32) || defined(_WIN32) || defined(__CYGWIN__) +#elif defined(MSDOS) || defined(OS2) || defined(__CYGWIN__) # include /* _isatty */ # define IS_CONSOLE(stdStream) _isatty(_fileno(stdStream)) +#elif defined(WIN32) || defined(_WIN32) +# include /* _isatty */ +# include /* DeviceIoControl, HANDLE, FSCTL_SET_SPARSE */ +# include /* FILE */ +static inline int IS_CONSOLE(FILE* stdStream) +{ + DWORD dummy; + return _isatty(_fileno(stdStream)) && GetConsoleMode((HANDLE)_get_osfhandle(_fileno(stdStream)), &dummy); +} #else # define IS_CONSOLE(stdStream) 0 #endif From efdaf8bb7ca36c80be96226775927b69dfa0cc11 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 27 Mar 2017 12:26:40 -0700 Subject: [PATCH 077/305] Fix inline compile errors --- programs/platform.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/programs/platform.h b/programs/platform.h index 819bec873..ee7819512 100644 --- a/programs/platform.h +++ b/programs/platform.h @@ -107,7 +107,7 @@ extern "C" { # include /* _isatty */ # include /* DeviceIoControl, HANDLE, FSCTL_SET_SPARSE */ # include /* FILE */ -static inline int IS_CONSOLE(FILE* stdStream) +static __inline int IS_CONSOLE(FILE* stdStream) { DWORD dummy; return _isatty(_fileno(stdStream)) && GetConsoleMode((HANDLE)_get_osfhandle(_fileno(stdStream)), &dummy); From 4708394bdd22530da32601eddd906b645f63961c Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Wed, 29 Mar 2017 11:46:57 -0700 Subject: [PATCH 078/305] Remove extra 'F' from skippable magic mask --- lib/decompress/zstd_decompress.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c index 65e6eda70..06337dbd5 100644 --- a/lib/decompress/zstd_decompress.c +++ b/lib/decompress/zstd_decompress.c @@ -1476,7 +1476,7 @@ size_t ZSTD_findFrameCompressedSize(const void *src, size_t srcSize) if (ZSTD_isLegacy(src, srcSize)) return ZSTD_findFrameCompressedSizeLegacy(src, srcSize); #endif if (srcSize >= ZSTD_skippableHeaderSize && - (MEM_readLE32(src) & 0xFFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { + (MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { return ZSTD_skippableHeaderSize + MEM_readLE32((const BYTE*)src + 4); } else { const BYTE* ip = (const BYTE*)src; From 5bde4be5a19aa9c08a389d4becd45b40731b252d Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 29 Mar 2017 12:10:38 -0700 Subject: [PATCH 079/305] fix : bench automatically adapts parameters to srcSize --- programs/bench.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/programs/bench.c b/programs/bench.c index 2643dd45a..d0dac497a 100644 --- a/programs/bench.c +++ b/programs/bench.c @@ -156,7 +156,7 @@ static int BMK_benchMem(const void* srcBuffer, size_t srcSize, ZSTD_compressionParameters *comprParams) { size_t const blockSize = ((g_blockSize>=32 && !g_decodeOnly) ? g_blockSize : srcSize) + (!srcSize) /* avoid div by 0 */ ; - size_t const avgSize = MIN(g_blockSize, (srcSize / nbFiles)); + size_t const avgSize = MIN(blockSize, (srcSize / nbFiles)); U32 const maxNbBlocks = (U32) ((srcSize + (blockSize-1)) / blockSize) + nbFiles; blockParam_t* const blockTable = (blockParam_t*) malloc(maxNbBlocks * sizeof(blockParam_t)); size_t const maxCompressedSize = ZSTD_compressBound(srcSize) + (maxNbBlocks * 1024); /* add some room for safety */ @@ -179,7 +179,7 @@ static int BMK_benchMem(const void* srcBuffer, size_t srcSize, if (strlen(displayName)>17) displayName += strlen(displayName)-17; /* can only display 17 characters */ UTIL_initTimer(&ticksPerSecond); - if (g_decodeOnly) { + if (g_decodeOnly) { /* benchmark only decompression : source must be already compressed */ const char* srcPtr = (const char*) srcBuffer; U64 dSize64 = 0; U32 fileNb; From fcc55ccae44e3c2101cb7d5febb56374f2db47e5 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Wed, 29 Mar 2017 13:08:10 -0700 Subject: [PATCH 080/305] Build windows releases on appveyor as artifacts --- appveyor.yml | 17 ++++++++++++----- 1 file changed, 12 insertions(+), 5 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index a53bce708..7e06cfdfe 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -57,10 +57,10 @@ - ECHO Installing %COMPILER% %PLATFORM% %CONFIGURATION% - SET PATH_ORIGINAL=%PATH% - if [%HOST%]==[mingw] ( - SET "PATH_MINGW32=C:\MinGW\bin;C:\MinGW\usr\bin" && - SET "PATH_MINGW64=C:\msys64\mingw64\bin;C:\msys64\usr\bin" && - COPY C:\msys64\usr\bin\make.exe C:\MinGW\bin\make.exe && - COPY C:\MinGW\bin\gcc.exe C:\MinGW\bin\cc.exe + SET "PATH_MINGW32=C:\mingw-w64\i686-6.3.0-posix-dwarf-rt_v5-rev1\mingw32\bin" && + SET "PATH_MINGW64=C:\mingw-w64\x86_64-6.3.0-posix-seh-rt_v5-rev1\mingw64\bin" && + COPY C:\msys64\usr\bin\make.exe C:\mingw-w64\i686-6.3.0-posix-dwarf-rt_v5-rev1\mingw32\bin\make.exe && + COPY C:\msys64\usr\bin\make.exe C:\mingw-w64\x86_64-6.3.0-posix-seh-rt_v5-rev1\mingw64\bin\make.exe ) - IF [%HOST%]==[visual] IF [%PLATFORM%]==[x64] ( SET ADDITIONALPARAM=/p:LibraryPath="C:\Program Files\Microsoft SDKs\Windows\v7.1\lib\x64;c:\Program Files (x86)\Microsoft Visual Studio 10.0\VC\lib\amd64;C:\Program Files (x86)\Microsoft Visual Studio 10.0\;C:\Program Files (x86)\Microsoft Visual Studio 10.0\lib\amd64;" @@ -85,7 +85,14 @@ SET "CPPFLAGS=-I../../zlib" && SET "LDFLAGS=../../zlib/libz.a" && sh -c "%SCRIPT%" && - ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] COPY programs\zstd.exe zstd_%PLATFORM%.exe && appveyor PushArtifact zstd_%PLATFORM%.exe ) + ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] COPY programs\zstd.exe zstd_%PLATFORM%.exe && appveyor PushArtifact zstd_%PLATFORM%.exe ) && + ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] + lib\dll\example\build_package.bat && + make -C programs DEBUGFLAGS= clean zstdmt && + cp programs\zstd.exe bin\zstdmt.exe && + cd bin\ && 7z a -tzip zstd-win-release-%PLATFORM%.zip * && + appveyor PushArtifact zstd-win-release-%PLATFORM%.zip + ) ) - if [%HOST%]==[visual] ( ECHO *** && From 933ce4a1dd101b8e4d8df2e17d52e232be945e12 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 29 Mar 2017 14:32:15 -0700 Subject: [PATCH 081/305] fix : minmatch 7 conversion minmatch 7 now converted to minmatch 6 for strategies which do not support 7 Used to folded into "default", which applied minmatch 4 --- lib/compress/zstd_compress.c | 22 +++++++++++++--------- lib/compress/zstd_opt.h | 2 ++ 2 files changed, 15 insertions(+), 9 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index f8e3de3fb..bc88baee4 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -962,7 +962,7 @@ static size_t ZSTD_hashPtr(const void* p, U32 hBits, U32 mls) { switch(mls) { - default: + default: /* includes case 3 */ case 4: return ZSTD_hash4Ptr(p, hBits); case 5: return ZSTD_hash5Ptr(p, hBits); case 6: return ZSTD_hash6Ptr(p, hBits); @@ -1085,7 +1085,7 @@ static void ZSTD_compressBlock_fast(ZSTD_CCtx* ctx, const U32 mls = ctx->params.cParams.searchLength; switch(mls) { - default: + default: /* includes case 3 */ case 4 : ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 4); return; case 5 : @@ -1199,7 +1199,7 @@ static void ZSTD_compressBlock_fast_extDict(ZSTD_CCtx* ctx, U32 const mls = ctx->params.cParams.searchLength; switch(mls) { - default: + default: /* includes case 3 */ case 4 : ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 4); return; case 5 : @@ -1353,7 +1353,7 @@ static void ZSTD_compressBlock_doubleFast(ZSTD_CCtx* ctx, const void* src, size_ const U32 mls = ctx->params.cParams.searchLength; switch(mls) { - default: + default: /* includes case 3 */ case 4 : ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 4); return; case 5 : @@ -1503,7 +1503,7 @@ static void ZSTD_compressBlock_doubleFast_extDict(ZSTD_CCtx* ctx, U32 const mls = ctx->params.cParams.searchLength; switch(mls) { - default: + default: /* includes case 3 */ case 4 : ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 4); return; case 5 : @@ -1733,9 +1733,10 @@ static size_t ZSTD_BtFindBestMatch_selectMLS ( { switch(matchLengthSearch) { - default : + default : /* includes case 3 */ case 4 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4); case 5 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5); + case 7 : case 6 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6); } } @@ -1772,9 +1773,10 @@ static size_t ZSTD_BtFindBestMatch_selectMLS_extDict ( { switch(matchLengthSearch) { - default : + default : /* includes case 3 */ case 4 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4); case 5 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5); + case 7 : case 6 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6); } } @@ -1868,9 +1870,10 @@ FORCE_INLINE size_t ZSTD_HcFindBestMatch_selectMLS ( { switch(matchLengthSearch) { - default : + default : /* includes case 3 */ case 4 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4, 0); case 5 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5, 0); + case 7 : case 6 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6, 0); } } @@ -1884,9 +1887,10 @@ FORCE_INLINE size_t ZSTD_HcFindBestMatch_extDict_selectMLS ( { switch(matchLengthSearch) { - default : + default : /* includes case 3 */ case 4 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4, 1); case 5 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5, 1); + case 7 : case 6 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6, 1); } } diff --git a/lib/compress/zstd_opt.h b/lib/compress/zstd_opt.h index ac418b61c..61cdc912a 100644 --- a/lib/compress/zstd_opt.h +++ b/lib/compress/zstd_opt.h @@ -360,6 +360,7 @@ static U32 ZSTD_BtGetAllMatches_selectMLS ( default : case 4 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 4, matches, minMatchLen); case 5 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 5, matches, minMatchLen); + case 7 : case 6 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 6, matches, minMatchLen); } } @@ -387,6 +388,7 @@ static U32 ZSTD_BtGetAllMatches_selectMLS_extDict ( default : case 4 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 4, matches, minMatchLen); case 5 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 5, matches, minMatchLen); + case 7 : case 6 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 6, matches, minMatchLen); } } From dd79c5c9e874f575b98ee5b5a3e947f63b475168 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Wed, 29 Mar 2017 14:40:11 -0700 Subject: [PATCH 082/305] Make pzstd and cmake use gcc/g++ --- appveyor.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/appveyor.yml b/appveyor.yml index 7e06cfdfe..81f76926e 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -147,6 +147,8 @@ test_script: - ECHO Testing %COMPILER% %PLATFORM% %CONFIGURATION% + - SET "CC=gcc" + - SET "CXX=g++" - if [%TEST%]==[cmake] ( mkdir build\cmake\build && cd build\cmake\build && From 2e2e78de47934963d7723ce821043c3a8ad84a32 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 29 Mar 2017 16:02:47 -0700 Subject: [PATCH 083/305] removed unnecessary restriction on minmatchLength it's now transparently translated to nearest value when unsupported (7->6) (3->4) --- doc/zstd_manual.html | 4 ++-- lib/compress/zstd_compress.c | 7 +++---- 2 files changed, 5 insertions(+), 6 deletions(-) diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index 204f56ea5..a66f0a49c 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -1,10 +1,10 @@ -zstd 1.1.4 Manual +zstd 1.1.5 Manual -

zstd 1.1.4 Manual

+

zstd 1.1.5 Manual


Contents

    diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index bc88baee4..27b11b8ef 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -150,9 +150,7 @@ size_t ZSTD_checkCParams(ZSTD_compressionParameters cParams) CLAMPCHECK(cParams.chainLog, ZSTD_CHAINLOG_MIN, ZSTD_CHAINLOG_MAX); CLAMPCHECK(cParams.hashLog, ZSTD_HASHLOG_MIN, ZSTD_HASHLOG_MAX); CLAMPCHECK(cParams.searchLog, ZSTD_SEARCHLOG_MIN, ZSTD_SEARCHLOG_MAX); - { U32 const searchLengthMin = ((cParams.strategy == ZSTD_fast) | (cParams.strategy == ZSTD_greedy)) ? ZSTD_SEARCHLENGTH_MIN+1 : ZSTD_SEARCHLENGTH_MIN; - U32 const searchLengthMax = (cParams.strategy == ZSTD_fast) ? ZSTD_SEARCHLENGTH_MAX : ZSTD_SEARCHLENGTH_MAX-1; - CLAMPCHECK(cParams.searchLength, searchLengthMin, searchLengthMax); } + CLAMPCHECK(cParams.searchLength, ZSTD_SEARCHLENGTH_MIN, ZSTD_SEARCHLENGTH_MAX); CLAMPCHECK(cParams.targetLength, ZSTD_TARGETLENGTH_MIN, ZSTD_TARGETLENGTH_MAX); if ((U32)(cParams.strategy) > (U32)ZSTD_btopt2) return ERROR(compressionParameter_unsupported); return 0; @@ -962,7 +960,8 @@ static size_t ZSTD_hashPtr(const void* p, U32 hBits, U32 mls) { switch(mls) { - default: /* includes case 3 */ + //case 3: return ZSTD_hash3Ptr(p, hBits); + default: case 4: return ZSTD_hash4Ptr(p, hBits); case 5: return ZSTD_hash5Ptr(p, hBits); case 6: return ZSTD_hash6Ptr(p, hBits); From 0ef680370798f368f4e7eb2121101cf4b1edcb48 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 29 Mar 2017 16:58:57 -0700 Subject: [PATCH 084/305] added ZSTD_getFrameContentSize() to ZSTDMT unit test in fuzzer Now tests fail, because ZSTDMT_compress() doesn't fill frame content size correctly. --- tests/fuzzer.c | 55 +++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 54 insertions(+), 1 deletion(-) diff --git a/tests/fuzzer.c b/tests/fuzzer.c index 111ca8243..9d8121806 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -28,6 +28,7 @@ #define ZSTD_STATIC_LINKING_ONLY /* ZSTD_compressContinue, ZSTD_compressBlock */ #include "zstd.h" /* ZSTD_VERSION_STRING */ #include "zstd_errors.h" /* ZSTD_getErrorCode */ +#include "zstdmt_compress.h" #define ZDICT_STATIC_LINKING_ONLY #include "zdict.h" /* ZDICT_trainFromBuffer */ #include "datagen.h" /* RDG_genBuffer */ @@ -133,13 +134,21 @@ static int basicUnitTests(U32 seed, double compressibility) DISPLAYLEVEL(4, "OK : %s \n", errorString); } + DISPLAYLEVEL(4, "test%3i : compress %u bytes : ", testNb++, (U32)CNBuffSize); CHECKPLUS(r, ZSTD_compress(compressedBuffer, ZSTD_compressBound(CNBuffSize), CNBuffer, CNBuffSize, 1), cSize=r ); DISPLAYLEVEL(4, "OK (%u bytes : %.2f%%)\n", (U32)cSize, (double)cSize/CNBuffSize*100); - DISPLAYLEVEL(4, "test%3i : decompressed size test : ", testNb++); + + DISPLAYLEVEL(4, "test%3i : ZSTD_getFrameContentSize test : ", testNb++); + { unsigned long long const rSize = ZSTD_getFrameContentSize(compressedBuffer, cSize); + if (rSize != CNBuffSize) goto _output_error; + } + DISPLAYLEVEL(4, "OK \n"); + + DISPLAYLEVEL(4, "test%3i : ZSTD_findDecompressedSize test : ", testNb++); { unsigned long long const rSize = ZSTD_findDecompressedSize(compressedBuffer, cSize); if (rSize != CNBuffSize) goto _output_error; } @@ -157,6 +166,7 @@ static int basicUnitTests(U32 seed, double compressibility) } } DISPLAYLEVEL(4, "OK \n"); + DISPLAYLEVEL(4, "test%3i : decompress with null dict : ", testNb++); { size_t const r = ZSTD_decompress_usingDict(dctx, decodedBuffer, CNBuffSize, compressedBuffer, cSize, NULL, 0); if (r != CNBuffSize) goto _output_error; } @@ -179,6 +189,49 @@ static int basicUnitTests(U32 seed, double compressibility) if (ZSTD_getErrorCode(r) != ZSTD_error_srcSize_wrong) goto _output_error; } DISPLAYLEVEL(4, "OK \n"); + + /* ZSTDMT Multi-threading test */ + DISPLAYLEVEL(4, "test%3i : create ZSTDMT CCtx : ", testNb++); + { ZSTDMT_CCtx* mtctx = ZSTDMT_createCCtx(2); + if (mtctx==NULL) { + DISPLAY("mtctx : mot enough memory, aborting \n"); + testResult = 1; + goto _end; + } + DISPLAYLEVEL(4, "OK \n"); + + DISPLAYLEVEL(4, "test%3i : compress %u bytes with 2 threads : ", testNb++, (U32)CNBuffSize); + CHECKPLUS(r, ZSTDMT_compressCCtx(mtctx, + compressedBuffer, ZSTD_compressBound(CNBuffSize), + CNBuffer, CNBuffSize, + 1), + cSize=r ); + DISPLAYLEVEL(4, "OK (%u bytes : %.2f%%)\n", (U32)cSize, (double)cSize/CNBuffSize*100); + + DISPLAYLEVEL(4, "test%3i : decompressed size test : ", testNb++); + { unsigned long long const rSize = ZSTD_getFrameContentSize(compressedBuffer, cSize); + if (rSize != CNBuffSize) { + DISPLAY("ZSTD_getFrameContentSize incorrect : %u != %u \n", (U32)rSize, (U32)CNBuffSize); + goto _output_error; + } } + DISPLAYLEVEL(4, "OK \n"); + + DISPLAYLEVEL(4, "test%3i : decompress %u bytes : ", testNb++, (U32)CNBuffSize); + { size_t const r = ZSTD_decompress(decodedBuffer, CNBuffSize, compressedBuffer, cSize); + if (r != CNBuffSize) goto _output_error; } + DISPLAYLEVEL(4, "OK \n"); + + DISPLAYLEVEL(4, "test%3i : check decompressed result : ", testNb++); + { size_t u; + for (u=0; u Date: Wed, 29 Mar 2017 17:09:59 -0700 Subject: [PATCH 085/305] fixed #634 : ZSTDMT_compressCCtx() doesn't provide frame content size in header --- tests/fuzzer.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/fuzzer.c b/tests/fuzzer.c index 9d8121806..fed875843 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -190,7 +190,7 @@ static int basicUnitTests(U32 seed, double compressibility) DISPLAYLEVEL(4, "OK \n"); - /* ZSTDMT Multi-threading test */ + /* ZSTDMT simple MT compression test */ DISPLAYLEVEL(4, "test%3i : create ZSTDMT CCtx : ", testNb++); { ZSTDMT_CCtx* mtctx = ZSTDMT_createCCtx(2); if (mtctx==NULL) { From ca5a8bbe360db56f7d24326854dbefce7fe03b86 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 29 Mar 2017 17:15:27 -0700 Subject: [PATCH 086/305] re-added patch ... --- lib/compress/zstdmt_compress.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/lib/compress/zstdmt_compress.c b/lib/compress/zstdmt_compress.c index 45514a81a..90f79509b 100644 --- a/lib/compress/zstdmt_compress.c +++ b/lib/compress/zstdmt_compress.c @@ -33,7 +33,7 @@ # include # include # include - static unsigned g_debugLevel = 3; + static unsigned g_debugLevel = 2; # define DEBUGLOGRAW(l, ...) if (l<=g_debugLevel) { fprintf(stderr, __VA_ARGS__); } # define DEBUGLOG(l, ...) if (l<=g_debugLevel) { fprintf(stderr, __FILE__ ": "); fprintf(stderr, __VA_ARGS__); fprintf(stderr, " \n"); } @@ -235,7 +235,7 @@ void ZSTDMT_compressChunk(void* jobDescription) if (ZSTD_isError(initError)) { job->cSize = initError; goto _endJob; } } else { /* srcStart points at reloaded section */ size_t const dictModeError = ZSTD_setCCtxParameter(job->cctx, ZSTD_p_forceRawDict, 1); /* Force loading dictionary in "content-only" mode (no header analysis) */ - size_t const initError = ZSTD_compressBegin_advanced(job->cctx, job->srcStart, job->dictSize, job->params, 0); + size_t const initError = ZSTD_compressBegin_advanced(job->cctx, job->srcStart, job->dictSize, job->params, job->fullFrameSize); if (ZSTD_isError(initError) || ZSTD_isError(dictModeError)) { job->cSize = initError; goto _endJob; } ZSTD_setCCtxParameter(job->cctx, ZSTD_p_forceWindow, 1); } From 5152fb2cb26d9d7cf1269d20c8bf29ab5403a44c Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Wed, 29 Mar 2017 18:51:58 -0700 Subject: [PATCH 087/305] Convert all tabs to spaces --- lib/common/bitstream.h | 4 ++-- lib/common/mem.h | 2 +- lib/compress/fse_compress.c | 16 +++++++------- lib/compress/zstd_compress.c | 8 +++---- lib/compress/zstd_opt.h | 8 +++---- lib/dictBuilder/zdict.c | 8 +++---- lib/legacy/zstd_v01.c | 2 +- lib/legacy/zstd_v02.c | 42 ++++++++++++++++++------------------ lib/legacy/zstd_v03.c | 40 +++++++++++++++++----------------- lib/legacy/zstd_v04.c | 8 +++---- lib/legacy/zstd_v05.c | 4 ++-- lib/legacy/zstd_v06.c | 4 ++-- programs/util.h | 10 ++++----- tests/fullbench.c | 4 ++-- 14 files changed, 80 insertions(+), 80 deletions(-) diff --git a/lib/common/bitstream.h b/lib/common/bitstream.h index d3873002e..0e3d2fc55 100644 --- a/lib/common/bitstream.h +++ b/lib/common/bitstream.h @@ -376,8 +376,8 @@ MEM_STATIC size_t BIT_readBitsFast(BIT_DStream_t* bitD, U32 nbBits) if status == BIT_DStream_unfinished, internal register is filled with >= (sizeof(bitD->bitContainer)*8 - 7) bits */ MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD) { - if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should not happen => corruption detected */ - return BIT_DStream_overflow; + if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should not happen => corruption detected */ + return BIT_DStream_overflow; if (bitD->ptr >= bitD->start + sizeof(bitD->bitContainer)) { bitD->ptr -= bitD->bitsConsumed >> 3; diff --git a/lib/common/mem.h b/lib/common/mem.h index 3cacd216a..f049d181b 100644 --- a/lib/common/mem.h +++ b/lib/common/mem.h @@ -122,7 +122,7 @@ MEM_STATIC void MEM_write64(void* memPtr, U64 value) { *(U64*)memPtr = value; } /* __pack instructions are safer, but compiler specific, hence potentially problematic for some compilers */ /* currently only defined for gcc and icc */ #if defined(_MSC_VER) || (defined(__INTEL_COMPILER) && defined(WIN32)) - __pragma( pack(push, 1) ) + __pragma( pack(push, 1) ) typedef union { U16 u16; U32 u32; U64 u64; size_t st; } unalign; __pragma( pack(pop) ) #else diff --git a/lib/compress/fse_compress.c b/lib/compress/fse_compress.c index e520e802b..13654d6e6 100644 --- a/lib/compress/fse_compress.c +++ b/lib/compress/fse_compress.c @@ -476,20 +476,20 @@ void FSE_freeCTable (FSE_CTable* ct) { free(ct); } /* provides the minimum logSize to safely represent a distribution */ static unsigned FSE_minTableLog(size_t srcSize, unsigned maxSymbolValue) { - U32 minBitsSrc = BIT_highbit32((U32)(srcSize - 1)) + 1; - U32 minBitsSymbols = BIT_highbit32(maxSymbolValue) + 2; - U32 minBits = minBitsSrc < minBitsSymbols ? minBitsSrc : minBitsSymbols; - return minBits; + U32 minBitsSrc = BIT_highbit32((U32)(srcSize - 1)) + 1; + U32 minBitsSymbols = BIT_highbit32(maxSymbolValue) + 2; + U32 minBits = minBitsSrc < minBitsSymbols ? minBitsSrc : minBitsSymbols; + return minBits; } unsigned FSE_optimalTableLog_internal(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue, unsigned minus) { - U32 maxBitsSrc = BIT_highbit32((U32)(srcSize - 1)) - minus; + U32 maxBitsSrc = BIT_highbit32((U32)(srcSize - 1)) - minus; U32 tableLog = maxTableLog; - U32 minBits = FSE_minTableLog(srcSize, maxSymbolValue); + U32 minBits = FSE_minTableLog(srcSize, maxSymbolValue); if (tableLog==0) tableLog = FSE_DEFAULT_TABLELOG; - if (maxBitsSrc < tableLog) tableLog = maxBitsSrc; /* Accuracy can be reduced */ - if (minBits > tableLog) tableLog = minBits; /* Need a minimum to safely represent all symbol values */ + if (maxBitsSrc < tableLog) tableLog = maxBitsSrc; /* Accuracy can be reduced */ + if (minBits > tableLog) tableLog = minBits; /* Need a minimum to safely represent all symbol values */ if (tableLog < FSE_MIN_TABLELOG) tableLog = FSE_MIN_TABLELOG; if (tableLog > FSE_MAX_TABLELOG) tableLog = FSE_MAX_TABLELOG; return tableLog; diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 27b11b8ef..c31f8db91 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -1461,8 +1461,8 @@ static void ZSTD_compressBlock_doubleFast_extDict_generic(ZSTD_CCtx* ctx, if (ip <= ilimit) { /* Fill Table */ - hashSmall[ZSTD_hashPtr(base+current+2, hBitsS, mls)] = current+2; - hashLong[ZSTD_hashPtr(base+current+2, hBitsL, 8)] = current+2; + hashSmall[ZSTD_hashPtr(base+current+2, hBitsS, mls)] = current+2; + hashLong[ZSTD_hashPtr(base+current+2, hBitsL, 8)] = current+2; hashSmall[ZSTD_hashPtr(ip-2, hBitsS, mls)] = (U32)(ip-2-base); hashLong[ZSTD_hashPtr(ip-2, hBitsL, 8)] = (U32)(ip-2-base); /* check immediate repcode */ @@ -1587,7 +1587,7 @@ static U32 ZSTD_insertBt1(ZSTD_CCtx* zc, const BYTE* const ip, const U32 mls, co match = dictBase + matchIndex; matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iend, dictEnd, prefixStart); if (matchIndex+matchLength >= dictLimit) - match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ + match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ } if (matchLength > bestLength) { @@ -1666,7 +1666,7 @@ static size_t ZSTD_insertBtAndFindBestMatch ( match = dictBase + matchIndex; matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iend, dictEnd, prefixStart); if (matchIndex+matchLength >= dictLimit) - match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ + match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ } if (matchLength > bestLength) { diff --git a/lib/compress/zstd_opt.h b/lib/compress/zstd_opt.h index 61cdc912a..543761191 100644 --- a/lib/compress/zstd_opt.h +++ b/lib/compress/zstd_opt.h @@ -175,10 +175,10 @@ MEM_STATIC void ZSTD_updatePrice(seqStore_t* seqStorePtr, U32 litLength, const B } /* match offset */ - { BYTE const offCode = (BYTE)ZSTD_highbit32(offset+1); - seqStorePtr->offCodeSum++; - seqStorePtr->offCodeFreq[offCode]++; - } + { BYTE const offCode = (BYTE)ZSTD_highbit32(offset+1); + seqStorePtr->offCodeSum++; + seqStorePtr->offCodeFreq[offCode]++; + } /* match Length */ { const BYTE ML_deltaCode = 36; diff --git a/lib/dictBuilder/zdict.c b/lib/dictBuilder/zdict.c index bee5e059d..179e02eff 100644 --- a/lib/dictBuilder/zdict.c +++ b/lib/dictBuilder/zdict.c @@ -306,10 +306,10 @@ static dictItem ZDICT_analyzePos( /* look backward */ length = MINMATCHLENGTH; while ((length >= MINMATCHLENGTH) & (start > 0)) { - length = ZDICT_count(b + pos, b + suffix[start - 1]); - if (length >= LLIMIT) length = LLIMIT - 1; - lengthList[length]++; - if (length >= MINMATCHLENGTH) start--; + length = ZDICT_count(b + pos, b + suffix[start - 1]); + if (length >= LLIMIT) length = LLIMIT - 1; + lengthList[length]++; + if (length >= MINMATCHLENGTH) start--; } /* largest useful length */ diff --git a/lib/legacy/zstd_v01.c b/lib/legacy/zstd_v01.c index bcacb8d5d..cf5354d6a 100644 --- a/lib/legacy/zstd_v01.c +++ b/lib/legacy/zstd_v01.c @@ -1432,7 +1432,7 @@ typedef struct ZSTD_Cctx_s #else U32 hashTable[HASH_TABLESIZE]; #endif - BYTE buffer[WORKPLACESIZE]; + BYTE buffer[WORKPLACESIZE]; } cctxi_t; diff --git a/lib/legacy/zstd_v02.c b/lib/legacy/zstd_v02.c index 2297b28c8..3cf8f4778 100644 --- a/lib/legacy/zstd_v02.c +++ b/lib/legacy/zstd_v02.c @@ -475,8 +475,8 @@ MEM_STATIC size_t BIT_readBitsFast(BIT_DStream_t* bitD, U32 nbBits) MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD) { - if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should never happen */ - return BIT_DStream_overflow; + if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should never happen */ + return BIT_DStream_overflow; if (bitD->ptr >= bitD->start + sizeof(bitD->bitContainer)) { @@ -1334,8 +1334,8 @@ static size_t FSE_readNCount (short* normalizedCounter, unsigned* maxSVPtr, unsi else { bitCount -= (int)(8 * (iend - 4 - ip)); - ip = iend - 4; - } + ip = iend - 4; + } bitStream = MEM_readLE32(ip) >> (bitCount & 31); } } @@ -2040,7 +2040,7 @@ static size_t HUF_readDTableX4 (U32* DTable, const void* src, size_t srcSize) rankStart[0] = 0; /* forget 0w symbols; this is beginning of weight(1) */ } - /* Build rankVal */ + /* Build rankVal */ { const U32 minBits = tableLog+1 - maxW; U32 nextRankVal = 0; @@ -2374,7 +2374,7 @@ static size_t HUF_readDTableX6 (U32* DTable, const void* src, size_t srcSize) rankStart[0] = 0; /* forget 0w symbols; this is beginning of weight(1) */ } - /* Build rankVal */ + /* Build rankVal */ { const U32 minBits = tableLog+1 - maxW; U32 nextRankVal = 0; @@ -2948,14 +2948,14 @@ static size_t ZSTD_decodeLiteralsBlock(void* ctx, const size_t litSize = (MEM_readLE32(istart) & 0xFFFFFF) >> 2; /* no buffer issue : srcSize >= MIN_CBLOCK_SIZE */ if (litSize > srcSize-11) /* risk of reading too far with wildcopy */ { - if (litSize > srcSize-3) return ERROR(corruption_detected); - memcpy(dctx->litBuffer, istart, litSize); - dctx->litPtr = dctx->litBuffer; - dctx->litSize = litSize; - memset(dctx->litBuffer + dctx->litSize, 0, 8); - return litSize+3; - } - /* direct reference into compressed stream */ + if (litSize > srcSize-3) return ERROR(corruption_detected); + memcpy(dctx->litBuffer, istart, litSize); + dctx->litPtr = dctx->litBuffer; + dctx->litSize = litSize; + memset(dctx->litBuffer + dctx->litSize, 0, 8); + return litSize+3; + } + /* direct reference into compressed stream */ dctx->litPtr = istart+3; dctx->litSize = litSize; return litSize+3; @@ -3515,13 +3515,13 @@ static size_t ZSTD_decompressContinue(ZSTD_DCtx* ctx, void* dst, size_t maxDstSi unsigned ZSTDv02_isError(size_t code) { - return ZSTD_isError(code); + return ZSTD_isError(code); } size_t ZSTDv02_decompress( void* dst, size_t maxOriginalSize, const void* src, size_t compressedSize) { - return ZSTD_decompress(dst, maxOriginalSize, src, compressedSize); + return ZSTD_decompress(dst, maxOriginalSize, src, compressedSize); } size_t ZSTDv02_findFrameCompressedSize(const void *src, size_t compressedSize) @@ -3531,25 +3531,25 @@ size_t ZSTDv02_findFrameCompressedSize(const void *src, size_t compressedSize) ZSTDv02_Dctx* ZSTDv02_createDCtx(void) { - return (ZSTDv02_Dctx*)ZSTD_createDCtx(); + return (ZSTDv02_Dctx*)ZSTD_createDCtx(); } size_t ZSTDv02_freeDCtx(ZSTDv02_Dctx* dctx) { - return ZSTD_freeDCtx((ZSTD_DCtx*)dctx); + return ZSTD_freeDCtx((ZSTD_DCtx*)dctx); } size_t ZSTDv02_resetDCtx(ZSTDv02_Dctx* dctx) { - return ZSTD_resetDCtx((ZSTD_DCtx*)dctx); + return ZSTD_resetDCtx((ZSTD_DCtx*)dctx); } size_t ZSTDv02_nextSrcSizeToDecompress(ZSTDv02_Dctx* dctx) { - return ZSTD_nextSrcSizeToDecompress((ZSTD_DCtx*)dctx); + return ZSTD_nextSrcSizeToDecompress((ZSTD_DCtx*)dctx); } size_t ZSTDv02_decompressContinue(ZSTDv02_Dctx* dctx, void* dst, size_t maxDstSize, const void* src, size_t srcSize) { - return ZSTD_decompressContinue((ZSTD_DCtx*)dctx, dst, maxDstSize, src, srcSize); + return ZSTD_decompressContinue((ZSTD_DCtx*)dctx, dst, maxDstSize, src, srcSize); } diff --git a/lib/legacy/zstd_v03.c b/lib/legacy/zstd_v03.c index ef654931f..f438330a4 100644 --- a/lib/legacy/zstd_v03.c +++ b/lib/legacy/zstd_v03.c @@ -477,8 +477,8 @@ MEM_STATIC size_t BIT_readBitsFast(BIT_DStream_t* bitD, U32 nbBits) MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD) { - if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should never happen */ - return BIT_DStream_overflow; + if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should never happen */ + return BIT_DStream_overflow; if (bitD->ptr >= bitD->start + sizeof(bitD->bitContainer)) { @@ -1335,8 +1335,8 @@ static size_t FSE_readNCount (short* normalizedCounter, unsigned* maxSVPtr, unsi else { bitCount -= (int)(8 * (iend - 4 - ip)); - ip = iend - 4; - } + ip = iend - 4; + } bitStream = MEM_readLE32(ip) >> (bitCount & 31); } } @@ -2037,7 +2037,7 @@ static size_t HUF_readDTableX4 (U32* DTable, const void* src, size_t srcSize) rankStart[0] = 0; /* forget 0w symbols; this is beginning of weight(1) */ } - /* Build rankVal */ + /* Build rankVal */ { const U32 minBits = tableLog+1 - maxW; U32 nextRankVal = 0; @@ -2589,14 +2589,14 @@ static size_t ZSTD_decodeLiteralsBlock(void* ctx, const size_t litSize = (MEM_readLE32(istart) & 0xFFFFFF) >> 2; /* no buffer issue : srcSize >= MIN_CBLOCK_SIZE */ if (litSize > srcSize-11) /* risk of reading too far with wildcopy */ { - if (litSize > srcSize-3) return ERROR(corruption_detected); - memcpy(dctx->litBuffer, istart, litSize); - dctx->litPtr = dctx->litBuffer; - dctx->litSize = litSize; - memset(dctx->litBuffer + dctx->litSize, 0, 8); - return litSize+3; - } - /* direct reference into compressed stream */ + if (litSize > srcSize-3) return ERROR(corruption_detected); + memcpy(dctx->litBuffer, istart, litSize); + dctx->litPtr = dctx->litBuffer; + dctx->litSize = litSize; + memset(dctx->litBuffer + dctx->litSize, 0, 8); + return litSize+3; + } + /* direct reference into compressed stream */ dctx->litPtr = istart+3; dctx->litSize = litSize; return litSize+3; @@ -3156,13 +3156,13 @@ static size_t ZSTD_decompressContinue(ZSTD_DCtx* ctx, void* dst, size_t maxDstSi unsigned ZSTDv03_isError(size_t code) { - return ZSTD_isError(code); + return ZSTD_isError(code); } size_t ZSTDv03_decompress( void* dst, size_t maxOriginalSize, const void* src, size_t compressedSize) { - return ZSTD_decompress(dst, maxOriginalSize, src, compressedSize); + return ZSTD_decompress(dst, maxOriginalSize, src, compressedSize); } size_t ZSTDv03_findFrameCompressedSize(const void* src, size_t srcSize) @@ -3172,25 +3172,25 @@ size_t ZSTDv03_findFrameCompressedSize(const void* src, size_t srcSize) ZSTDv03_Dctx* ZSTDv03_createDCtx(void) { - return (ZSTDv03_Dctx*)ZSTD_createDCtx(); + return (ZSTDv03_Dctx*)ZSTD_createDCtx(); } size_t ZSTDv03_freeDCtx(ZSTDv03_Dctx* dctx) { - return ZSTD_freeDCtx((ZSTD_DCtx*)dctx); + return ZSTD_freeDCtx((ZSTD_DCtx*)dctx); } size_t ZSTDv03_resetDCtx(ZSTDv03_Dctx* dctx) { - return ZSTD_resetDCtx((ZSTD_DCtx*)dctx); + return ZSTD_resetDCtx((ZSTD_DCtx*)dctx); } size_t ZSTDv03_nextSrcSizeToDecompress(ZSTDv03_Dctx* dctx) { - return ZSTD_nextSrcSizeToDecompress((ZSTD_DCtx*)dctx); + return ZSTD_nextSrcSizeToDecompress((ZSTD_DCtx*)dctx); } size_t ZSTDv03_decompressContinue(ZSTDv03_Dctx* dctx, void* dst, size_t maxDstSize, const void* src, size_t srcSize) { - return ZSTD_decompressContinue((ZSTD_DCtx*)dctx, dst, maxDstSize, src, srcSize); + return ZSTD_decompressContinue((ZSTD_DCtx*)dctx, dst, maxDstSize, src, srcSize); } diff --git a/lib/legacy/zstd_v04.c b/lib/legacy/zstd_v04.c index 09040e68e..1a29da92d 100644 --- a/lib/legacy/zstd_v04.c +++ b/lib/legacy/zstd_v04.c @@ -882,8 +882,8 @@ MEM_STATIC size_t BIT_readBitsFast(BIT_DStream_t* bitD, U32 nbBits) MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD) { - if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should never happen */ - return BIT_DStream_overflow; + if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should never happen */ + return BIT_DStream_overflow; if (bitD->ptr >= bitD->start + sizeof(bitD->bitContainer)) { @@ -1451,8 +1451,8 @@ static size_t FSE_readNCount (short* normalizedCounter, unsigned* maxSVPtr, unsi else { bitCount -= (int)(8 * (iend - 4 - ip)); - ip = iend - 4; - } + ip = iend - 4; + } bitStream = MEM_readLE32(ip) >> (bitCount & 31); } } diff --git a/lib/legacy/zstd_v05.c b/lib/legacy/zstd_v05.c index a6f5f5dbb..674f5b0e4 100644 --- a/lib/legacy/zstd_v05.c +++ b/lib/legacy/zstd_v05.c @@ -884,8 +884,8 @@ MEM_STATIC size_t BITv05_readBitsFast(BITv05_DStream_t* bitD, U32 nbBits) MEM_STATIC BITv05_DStream_status BITv05_reloadDStream(BITv05_DStream_t* bitD) { - if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should never happen */ - return BITv05_DStream_overflow; + if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should never happen */ + return BITv05_DStream_overflow; if (bitD->ptr >= bitD->start + sizeof(bitD->bitContainer)) { bitD->ptr -= bitD->bitsConsumed >> 3; diff --git a/lib/legacy/zstd_v06.c b/lib/legacy/zstd_v06.c index a4258b67a..ad8c4cd31 100644 --- a/lib/legacy/zstd_v06.c +++ b/lib/legacy/zstd_v06.c @@ -982,8 +982,8 @@ MEM_STATIC size_t BITv06_readBitsFast(BITv06_DStream_t* bitD, U32 nbBits) if status == unfinished, internal register is filled with >= (sizeof(bitD->bitContainer)*8 - 7) bits */ MEM_STATIC BITv06_DStream_status BITv06_reloadDStream(BITv06_DStream_t* bitD) { - if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should never happen */ - return BITv06_DStream_overflow; + if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should never happen */ + return BITv06_DStream_overflow; if (bitD->ptr >= bitD->start + sizeof(bitD->bitContainer)) { bitD->ptr -= bitD->bitsConsumed >> 3; diff --git a/programs/util.h b/programs/util.h index 9d23cbb8e..9992d79da 100644 --- a/programs/util.h +++ b/programs/util.h @@ -166,8 +166,8 @@ UTIL_STATIC void UTIL_waitForNextTick(UTIL_freq_t ticksPerSecond) * File functions ******************************************/ #if defined(_MSC_VER) - #define chmod _chmod - typedef struct __stat64 stat_t; + #define chmod _chmod + typedef struct __stat64 stat_t; #else typedef struct stat stat_t; #endif @@ -178,9 +178,9 @@ UTIL_STATIC int UTIL_setFileStat(const char *filename, stat_t *statbuf) int res = 0; struct utimbuf timebuf; - timebuf.actime = time(NULL); - timebuf.modtime = statbuf->st_mtime; - res += utime(filename, &timebuf); /* set access and modification times */ + timebuf.actime = time(NULL); + timebuf.modtime = statbuf->st_mtime; + res += utime(filename, &timebuf); /* set access and modification times */ #if !defined(_WIN32) res += chown(filename, statbuf->st_uid, statbuf->st_gid); /* Copy ownership */ diff --git a/tests/fullbench.c b/tests/fullbench.c index 940d315a7..38cf22fa7 100644 --- a/tests/fullbench.c +++ b/tests/fullbench.c @@ -251,14 +251,14 @@ static size_t benchMem(const void* src, size_t srcSize, U32 benchNb) case 13: benchFunction = local_ZSTD_decompressContinue; benchName = "ZSTD_decompressContinue"; break; - case 31: + case 31: benchFunction = local_ZSTD_decodeLiteralsBlock; benchName = "ZSTD_decodeLiteralsBlock"; break; case 32: benchFunction = local_ZSTD_decodeSeqHeaders; benchName = "ZSTD_decodeSeqHeaders"; break; #endif - case 41: + case 41: benchFunction = local_ZSTD_compressStream; benchName = "ZSTD_compressStream"; break; case 42: From 2bcefcc50d8230dad87a7a90f10e3d7f67c4052f Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 29 Mar 2017 22:03:27 -0700 Subject: [PATCH 088/305] fixed Visual Studio projects --- build/VS2008/fuzzer/fuzzer.vcproj | 2 +- build/VS2010/fuzzer/fuzzer.vcxproj | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/build/VS2008/fuzzer/fuzzer.vcproj b/build/VS2008/fuzzer/fuzzer.vcproj index 72540d243..a13624b8a 100644 --- a/build/VS2008/fuzzer/fuzzer.vcproj +++ b/build/VS2008/fuzzer/fuzzer.vcproj @@ -271,7 +271,7 @@ Optimization="2" EnableIntrinsicFunctions="true" OmitFramePointers="true" - AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs" + AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\compression;$(SolutionDir)..\..\programs" PreprocessorDefinitions="WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" diff --git a/build/VS2010/fuzzer/fuzzer.vcxproj b/build/VS2010/fuzzer/fuzzer.vcxproj index e30511a78..fff3a6a4f 100644 --- a/build/VS2010/fuzzer/fuzzer.vcxproj +++ b/build/VS2010/fuzzer/fuzzer.vcxproj @@ -67,22 +67,22 @@ true false - $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(UniversalCRT_IncludePath); + $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compression;$(UniversalCRT_IncludePath); true false - $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(UniversalCRT_IncludePath); + $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compression;$(UniversalCRT_IncludePath); false false - $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(UniversalCRT_IncludePath); + $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compression;$(UniversalCRT_IncludePath); false false - $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(UniversalCRT_IncludePath); + $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compression;$(UniversalCRT_IncludePath); From 3d58a1f9b934914ea05abd93aaf1c3304f18d2e1 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 30 Mar 2017 12:28:09 -0700 Subject: [PATCH 089/305] fixed Visual fuzzer project --- build/VS2008/fuzzer/fuzzer.vcproj | 8 ++++---- build/VS2010/fuzzer/fuzzer.vcxproj | 8 ++++---- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/build/VS2008/fuzzer/fuzzer.vcproj b/build/VS2008/fuzzer/fuzzer.vcproj index a13624b8a..f1719e8ac 100644 --- a/build/VS2008/fuzzer/fuzzer.vcproj +++ b/build/VS2008/fuzzer/fuzzer.vcproj @@ -44,7 +44,7 @@ true false - $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compression;$(UniversalCRT_IncludePath); + $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress;$(UniversalCRT_IncludePath); true false - $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compression;$(UniversalCRT_IncludePath); + $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress;$(UniversalCRT_IncludePath); false false - $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compression;$(UniversalCRT_IncludePath); + $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress;$(UniversalCRT_IncludePath); false false - $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compression;$(UniversalCRT_IncludePath); + $(IncludePath);$(SolutionDir)..\..\lib;$(SolutionDir)..\..\programs;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress;$(UniversalCRT_IncludePath); From 6851db48e00452c4228f8ea2d0c5bda5df9f8e42 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 30 Mar 2017 12:42:44 -0700 Subject: [PATCH 090/305] created contrib/cleanTabs --- Makefile | 4 ++++ contrib/cleanTabs | 2 ++ 2 files changed, 6 insertions(+) create mode 100755 contrib/cleanTabs diff --git a/Makefile b/Makefile index 030fd3fb0..7e57e1680 100644 --- a/Makefile +++ b/Makefile @@ -90,6 +90,10 @@ examples: manual: $(MAKE) -C contrib/gen_html $@ +.PHONY: cleanTabs +cleanTabs: + cd contrib; ./cleanTabs + .PHONY: clean clean: @$(MAKE) -C $(ZSTDDIR) $@ > $(VOID) diff --git a/contrib/cleanTabs b/contrib/cleanTabs new file mode 100755 index 000000000..215913a90 --- /dev/null +++ b/contrib/cleanTabs @@ -0,0 +1,2 @@ +#!/bin/sh +sed -i '' $'s/\t/ /g' ../lib/**/*.{h,c} ../programs/*.{h,c} ../tests/*.c ./**/*.{h,cpp} ../examples/*.c ../zlibWrapper/*.{h,c} From 274f59919db06b63c021a3a949076c59643ad355 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 30 Mar 2017 12:52:14 -0700 Subject: [PATCH 091/305] Changed memory strategy to __packed for gcc Method 1 __packed is always as good or better than memcpy(). But it's not portable, as it depends on compiler extension. For gcc, __pakced directive works fine. Furthermore, gcc has serious performance issues with memcpy() on ARM 32 bits. See #620 --- lib/common/mem.h | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/lib/common/mem.h b/lib/common/mem.h index f049d181b..4773a8b93 100644 --- a/lib/common/mem.h +++ b/lib/common/mem.h @@ -89,8 +89,7 @@ MEM_STATIC void MEM_check(void) { MEM_STATIC_ASSERT((sizeof(size_t)==4) || (size #ifndef MEM_FORCE_MEMORY_ACCESS /* can be defined externally, on command line for example */ # if defined(__GNUC__) && ( defined(__ARM_ARCH_6__) || defined(__ARM_ARCH_6J__) || defined(__ARM_ARCH_6K__) || defined(__ARM_ARCH_6Z__) || defined(__ARM_ARCH_6ZK__) || defined(__ARM_ARCH_6T2__) ) # define MEM_FORCE_MEMORY_ACCESS 2 -# elif defined(__INTEL_COMPILER) /*|| defined(_MSC_VER)*/ || \ - (defined(__GNUC__) && ( defined(__ARM_ARCH_7__) || defined(__ARM_ARCH_7A__) || defined(__ARM_ARCH_7R__) || defined(__ARM_ARCH_7M__) || defined(__ARM_ARCH_7S__) )) +# elif defined(__INTEL_COMPILER) || defined(__GNUC__) # define MEM_FORCE_MEMORY_ACCESS 1 # endif #endif From 0fcb5d70b6073454a732bb40ad2398ac36e1ae7a Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Wed, 29 Mar 2017 13:08:10 -0700 Subject: [PATCH 092/305] Build windows releases on appveyor as artifacts --- appveyor.yml | 17 ++++++++++++----- 1 file changed, 12 insertions(+), 5 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index a53bce708..7e06cfdfe 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -57,10 +57,10 @@ - ECHO Installing %COMPILER% %PLATFORM% %CONFIGURATION% - SET PATH_ORIGINAL=%PATH% - if [%HOST%]==[mingw] ( - SET "PATH_MINGW32=C:\MinGW\bin;C:\MinGW\usr\bin" && - SET "PATH_MINGW64=C:\msys64\mingw64\bin;C:\msys64\usr\bin" && - COPY C:\msys64\usr\bin\make.exe C:\MinGW\bin\make.exe && - COPY C:\MinGW\bin\gcc.exe C:\MinGW\bin\cc.exe + SET "PATH_MINGW32=C:\mingw-w64\i686-6.3.0-posix-dwarf-rt_v5-rev1\mingw32\bin" && + SET "PATH_MINGW64=C:\mingw-w64\x86_64-6.3.0-posix-seh-rt_v5-rev1\mingw64\bin" && + COPY C:\msys64\usr\bin\make.exe C:\mingw-w64\i686-6.3.0-posix-dwarf-rt_v5-rev1\mingw32\bin\make.exe && + COPY C:\msys64\usr\bin\make.exe C:\mingw-w64\x86_64-6.3.0-posix-seh-rt_v5-rev1\mingw64\bin\make.exe ) - IF [%HOST%]==[visual] IF [%PLATFORM%]==[x64] ( SET ADDITIONALPARAM=/p:LibraryPath="C:\Program Files\Microsoft SDKs\Windows\v7.1\lib\x64;c:\Program Files (x86)\Microsoft Visual Studio 10.0\VC\lib\amd64;C:\Program Files (x86)\Microsoft Visual Studio 10.0\;C:\Program Files (x86)\Microsoft Visual Studio 10.0\lib\amd64;" @@ -85,7 +85,14 @@ SET "CPPFLAGS=-I../../zlib" && SET "LDFLAGS=../../zlib/libz.a" && sh -c "%SCRIPT%" && - ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] COPY programs\zstd.exe zstd_%PLATFORM%.exe && appveyor PushArtifact zstd_%PLATFORM%.exe ) + ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] COPY programs\zstd.exe zstd_%PLATFORM%.exe && appveyor PushArtifact zstd_%PLATFORM%.exe ) && + ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] + lib\dll\example\build_package.bat && + make -C programs DEBUGFLAGS= clean zstdmt && + cp programs\zstd.exe bin\zstdmt.exe && + cd bin\ && 7z a -tzip zstd-win-release-%PLATFORM%.zip * && + appveyor PushArtifact zstd-win-release-%PLATFORM%.zip + ) ) - if [%HOST%]==[visual] ( ECHO *** && From 137efc007773e5b5a25bb3cccc7c7a5748985675 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Wed, 29 Mar 2017 14:40:11 -0700 Subject: [PATCH 093/305] Make pzstd and cmake use gcc/g++ --- appveyor.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/appveyor.yml b/appveyor.yml index 7e06cfdfe..81f76926e 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -147,6 +147,8 @@ test_script: - ECHO Testing %COMPILER% %PLATFORM% %CONFIGURATION% + - SET "CC=gcc" + - SET "CXX=g++" - if [%TEST%]==[cmake] ( mkdir build\cmake\build && cd build\cmake\build && From b1c6bb87022404da56cc3015c85494c0ffcec520 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Wed, 29 Mar 2017 18:35:21 -0700 Subject: [PATCH 094/305] Copy files into contrib/linux-kernel/ --- contrib/linux-kernel/include/zstd.h | 775 +++++ contrib/linux-kernel/lib/bitstream.h | 417 +++ contrib/linux-kernel/lib/entropy_common.c | 221 ++ contrib/linux-kernel/lib/error_private.c | 44 + contrib/linux-kernel/lib/error_private.h | 76 + contrib/linux-kernel/lib/fse.h | 694 ++++ contrib/linux-kernel/lib/fse_compress.c | 857 +++++ contrib/linux-kernel/lib/fse_decompress.c | 328 ++ contrib/linux-kernel/lib/huf.h | 260 ++ contrib/linux-kernel/lib/huf_compress.c | 684 ++++ contrib/linux-kernel/lib/huf_decompress.c | 888 +++++ contrib/linux-kernel/lib/mem.h | 374 +++ contrib/linux-kernel/lib/xxhash.c | 869 +++++ contrib/linux-kernel/lib/xxhash.h | 305 ++ contrib/linux-kernel/lib/zstd_common.c | 73 + contrib/linux-kernel/lib/zstd_compress.c | 3400 ++++++++++++++++++++ contrib/linux-kernel/lib/zstd_decompress.c | 2484 ++++++++++++++ contrib/linux-kernel/lib/zstd_errors.h | 75 + contrib/linux-kernel/lib/zstd_internal.h | 283 ++ contrib/linux-kernel/lib/zstd_opt.h | 921 ++++++ 20 files changed, 14028 insertions(+) create mode 100644 contrib/linux-kernel/include/zstd.h create mode 100644 contrib/linux-kernel/lib/bitstream.h create mode 100644 contrib/linux-kernel/lib/entropy_common.c create mode 100644 contrib/linux-kernel/lib/error_private.c create mode 100644 contrib/linux-kernel/lib/error_private.h create mode 100644 contrib/linux-kernel/lib/fse.h create mode 100644 contrib/linux-kernel/lib/fse_compress.c create mode 100644 contrib/linux-kernel/lib/fse_decompress.c create mode 100644 contrib/linux-kernel/lib/huf.h create mode 100644 contrib/linux-kernel/lib/huf_compress.c create mode 100644 contrib/linux-kernel/lib/huf_decompress.c create mode 100644 contrib/linux-kernel/lib/mem.h create mode 100644 contrib/linux-kernel/lib/xxhash.c create mode 100644 contrib/linux-kernel/lib/xxhash.h create mode 100644 contrib/linux-kernel/lib/zstd_common.c create mode 100644 contrib/linux-kernel/lib/zstd_compress.c create mode 100644 contrib/linux-kernel/lib/zstd_decompress.c create mode 100644 contrib/linux-kernel/lib/zstd_errors.h create mode 100644 contrib/linux-kernel/lib/zstd_internal.h create mode 100644 contrib/linux-kernel/lib/zstd_opt.h diff --git a/contrib/linux-kernel/include/zstd.h b/contrib/linux-kernel/include/zstd.h new file mode 100644 index 000000000..4531a84bf --- /dev/null +++ b/contrib/linux-kernel/include/zstd.h @@ -0,0 +1,775 @@ +/* + * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + +#if defined (__cplusplus) +extern "C" { +#endif + +#ifndef ZSTD_H_235446 +#define ZSTD_H_235446 + +/* ====== Dependency ======*/ +#include /* size_t */ + + +/* ===== ZSTDLIB_API : control library symbols visibility ===== */ +#if defined(__GNUC__) && (__GNUC__ >= 4) +# define ZSTDLIB_VISIBILITY __attribute__ ((visibility ("default"))) +#else +# define ZSTDLIB_VISIBILITY +#endif +#if defined(ZSTD_DLL_EXPORT) && (ZSTD_DLL_EXPORT==1) +# define ZSTDLIB_API __declspec(dllexport) ZSTDLIB_VISIBILITY +#elif defined(ZSTD_DLL_IMPORT) && (ZSTD_DLL_IMPORT==1) +# define ZSTDLIB_API __declspec(dllimport) ZSTDLIB_VISIBILITY /* It isn't required but allows to generate better code, saving a function pointer load from the IAT and an indirect jump.*/ +#else +# define ZSTDLIB_API ZSTDLIB_VISIBILITY +#endif + + +/******************************************************************************************************* + Introduction + + zstd, short for Zstandard, is a fast lossless compression algorithm, targeting real-time compression scenarios + at zlib-level and better compression ratios. The zstd compression library provides in-memory compression and + decompression functions. The library supports compression levels from 1 up to ZSTD_maxCLevel() which is 22. + Levels >= 20, labeled `--ultra`, should be used with caution, as they require more memory. + Compression can be done in: + - a single step (described as Simple API) + - a single step, reusing a context (described as Explicit memory management) + - unbounded multiple steps (described as Streaming compression) + The compression ratio achievable on small data can be highly improved using compression with a dictionary in: + - a single step (described as Simple dictionary API) + - a single step, reusing a dictionary (described as Fast dictionary API) + + Advanced experimental functions can be accessed using #define ZSTD_STATIC_LINKING_ONLY before including zstd.h. + These APIs shall never be used with a dynamic library. + They are not "stable", their definition may change in the future. Only static linking is allowed. +*********************************************************************************************************/ + +/*------ Version ------*/ +#define ZSTD_VERSION_MAJOR 1 +#define ZSTD_VERSION_MINOR 1 +#define ZSTD_VERSION_RELEASE 5 + +#define ZSTD_LIB_VERSION ZSTD_VERSION_MAJOR.ZSTD_VERSION_MINOR.ZSTD_VERSION_RELEASE +#define ZSTD_QUOTE(str) #str +#define ZSTD_EXPAND_AND_QUOTE(str) ZSTD_QUOTE(str) +#define ZSTD_VERSION_STRING ZSTD_EXPAND_AND_QUOTE(ZSTD_LIB_VERSION) + +#define ZSTD_VERSION_NUMBER (ZSTD_VERSION_MAJOR *100*100 + ZSTD_VERSION_MINOR *100 + ZSTD_VERSION_RELEASE) +ZSTDLIB_API unsigned ZSTD_versionNumber(void); /**< library version number; to be used when checking dll version */ + + +/*************************************** +* Simple API +***************************************/ +/*! ZSTD_compress() : + Compresses `src` content as a single zstd compressed frame into already allocated `dst`. + Hint : compression runs faster if `dstCapacity` >= `ZSTD_compressBound(srcSize)`. + @return : compressed size written into `dst` (<= `dstCapacity), + or an error code if it fails (which can be tested using ZSTD_isError()). */ +ZSTDLIB_API size_t ZSTD_compress( void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + int compressionLevel); + +/*! ZSTD_decompress() : + `compressedSize` : must be the _exact_ size of some number of compressed and/or skippable frames. + `dstCapacity` is an upper bound of originalSize. + If user cannot imply a maximum upper bound, it's better to use streaming mode to decompress data. + @return : the number of bytes decompressed into `dst` (<= `dstCapacity`), + or an errorCode if it fails (which can be tested using ZSTD_isError()). */ +ZSTDLIB_API size_t ZSTD_decompress( void* dst, size_t dstCapacity, + const void* src, size_t compressedSize); + +/*! ZSTD_getDecompressedSize() : +* NOTE: This function is planned to be obsolete, in favour of ZSTD_getFrameContentSize. +* ZSTD_getFrameContentSize functions the same way, returning the decompressed size of a single +* frame, but distinguishes empty frames from frames with an unknown size, or errors. +* +* Additionally, ZSTD_findDecompressedSize can be used instead. It can handle multiple +* concatenated frames in one buffer, and so is more general. +* As a result however, it requires more computation and entire frames to be passed to it, +* as opposed to ZSTD_getFrameContentSize which requires only a single frame's header. +* +* 'src' is the start of a zstd compressed frame. +* @return : content size to be decompressed, as a 64-bits value _if known_, 0 otherwise. +* note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode. +* When `return==0`, data to decompress could be any size. +* In which case, it's necessary to use streaming mode to decompress data. +* Optionally, application can still use ZSTD_decompress() while relying on implied limits. +* (For example, data may be necessarily cut into blocks <= 16 KB). +* note 2 : decompressed size is always present when compression is done with ZSTD_compress() +* note 3 : decompressed size can be very large (64-bits value), +* potentially larger than what local system can handle as a single memory segment. +* In which case, it's necessary to use streaming mode to decompress data. +* note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified. +* Always ensure result fits within application's authorized limits. +* Each application can set its own limits. +* note 5 : when `return==0`, if precise failure cause is needed, use ZSTD_getFrameParams() to know more. */ +ZSTDLIB_API unsigned long long ZSTD_getDecompressedSize(const void* src, size_t srcSize); + + +/*====== Helper functions ======*/ +ZSTDLIB_API int ZSTD_maxCLevel(void); /*!< maximum compression level available */ +ZSTDLIB_API size_t ZSTD_compressBound(size_t srcSize); /*!< maximum compressed size in worst case scenario */ +ZSTDLIB_API unsigned ZSTD_isError(size_t code); /*!< tells if a `size_t` function result is an error code */ +ZSTDLIB_API const char* ZSTD_getErrorName(size_t code); /*!< provides readable string from an error code */ + + +/*************************************** +* Explicit memory management +***************************************/ +/*= Compression context +* When compressing many times, +* it is recommended to allocate a context just once, and re-use it for each successive compression operation. +* This will make workload friendlier for system's memory. +* Use one context per thread for parallel execution in multi-threaded environments. */ +typedef struct ZSTD_CCtx_s ZSTD_CCtx; +ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx(void); +ZSTDLIB_API size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx); + +/*! ZSTD_compressCCtx() : + Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). */ +ZSTDLIB_API size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel); + +/*= Decompression context +* When decompressing many times, +* it is recommended to allocate a context just once, and re-use it for each successive compression operation. +* This will make workload friendlier for system's memory. +* Use one context per thread for parallel execution in multi-threaded environments. */ +typedef struct ZSTD_DCtx_s ZSTD_DCtx; +ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx(void); +ZSTDLIB_API size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx); + +/*! ZSTD_decompressDCtx() : +* Same as ZSTD_decompress(), requires an allocated ZSTD_DCtx (see ZSTD_createDCtx()). */ +ZSTDLIB_API size_t ZSTD_decompressDCtx(ZSTD_DCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); + + +/************************** +* Simple dictionary API +***************************/ +/*! ZSTD_compress_usingDict() : +* Compression using a predefined Dictionary (see dictBuilder/zdict.h). +* Note : This function loads the dictionary, resulting in significant startup delay. +* Note : When `dict == NULL || dictSize < 8` no dictionary is used. */ +ZSTDLIB_API size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict,size_t dictSize, + int compressionLevel); + +/*! ZSTD_decompress_usingDict() : +* Decompression using a predefined Dictionary (see dictBuilder/zdict.h). +* Dictionary must be identical to the one used during compression. +* Note : This function loads the dictionary, resulting in significant startup delay. +* Note : When `dict == NULL || dictSize < 8` no dictionary is used. */ +ZSTDLIB_API size_t ZSTD_decompress_usingDict(ZSTD_DCtx* dctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict,size_t dictSize); + + +/**************************** +* Fast dictionary API +****************************/ +typedef struct ZSTD_CDict_s ZSTD_CDict; + +/*! ZSTD_createCDict() : +* When compressing multiple messages / blocks with the same dictionary, it's recommended to load it just once. +* ZSTD_createCDict() will create a digested dictionary, ready to start future compression operations without startup delay. +* ZSTD_CDict can be created once and used by multiple threads concurrently, as its usage is read-only. +* `dictBuffer` can be released after ZSTD_CDict creation, as its content is copied within CDict */ +ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict(const void* dictBuffer, size_t dictSize, int compressionLevel); + +/*! ZSTD_freeCDict() : +* Function frees memory allocated by ZSTD_createCDict(). */ +ZSTDLIB_API size_t ZSTD_freeCDict(ZSTD_CDict* CDict); + +/*! ZSTD_compress_usingCDict() : +* Compression using a digested Dictionary. +* Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. +* Note that compression level is decided during dictionary creation. */ +ZSTDLIB_API size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const ZSTD_CDict* cdict); + + +typedef struct ZSTD_DDict_s ZSTD_DDict; + +/*! ZSTD_createDDict() : +* Create a digested dictionary, ready to start decompression operation without startup delay. +* dictBuffer can be released after DDict creation, as its content is copied inside DDict */ +ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict(const void* dictBuffer, size_t dictSize); + +/*! ZSTD_freeDDict() : +* Function frees memory allocated with ZSTD_createDDict() */ +ZSTDLIB_API size_t ZSTD_freeDDict(ZSTD_DDict* ddict); + +/*! ZSTD_decompress_usingDDict() : +* Decompression using a digested Dictionary. +* Faster startup than ZSTD_decompress_usingDict(), recommended when same dictionary is used multiple times. */ +ZSTDLIB_API size_t ZSTD_decompress_usingDDict(ZSTD_DCtx* dctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const ZSTD_DDict* ddict); + + +/**************************** +* Streaming +****************************/ + +typedef struct ZSTD_inBuffer_s { + const void* src; /**< start of input buffer */ + size_t size; /**< size of input buffer */ + size_t pos; /**< position where reading stopped. Will be updated. Necessarily 0 <= pos <= size */ +} ZSTD_inBuffer; + +typedef struct ZSTD_outBuffer_s { + void* dst; /**< start of output buffer */ + size_t size; /**< size of output buffer */ + size_t pos; /**< position where writing stopped. Will be updated. Necessarily 0 <= pos <= size */ +} ZSTD_outBuffer; + + + +/*-*********************************************************************** +* Streaming compression - HowTo +* +* A ZSTD_CStream object is required to track streaming operation. +* Use ZSTD_createCStream() and ZSTD_freeCStream() to create/release resources. +* ZSTD_CStream objects can be reused multiple times on consecutive compression operations. +* It is recommended to re-use ZSTD_CStream in situations where many streaming operations will be achieved consecutively, +* since it will play nicer with system's memory, by re-using already allocated memory. +* Use one separate ZSTD_CStream per thread for parallel execution. +* +* Start a new compression by initializing ZSTD_CStream. +* Use ZSTD_initCStream() to start a new compression operation. +* Use ZSTD_initCStream_usingDict() or ZSTD_initCStream_usingCDict() for a compression which requires a dictionary (experimental section) +* +* Use ZSTD_compressStream() repetitively to consume input stream. +* The function will automatically update both `pos` fields. +* Note that it may not consume the entire input, in which case `pos < size`, +* and it's up to the caller to present again remaining data. +* @return : a size hint, preferred nb of bytes to use as input for next function call +* or an error code, which can be tested using ZSTD_isError(). +* Note 1 : it's just a hint, to help latency a little, any other value will work fine. +* Note 2 : size hint is guaranteed to be <= ZSTD_CStreamInSize() +* +* At any moment, it's possible to flush whatever data remains within internal buffer, using ZSTD_flushStream(). +* `output->pos` will be updated. +* Note that some content might still be left within internal buffer if `output->size` is too small. +* @return : nb of bytes still present within internal buffer (0 if it's empty) +* or an error code, which can be tested using ZSTD_isError(). +* +* ZSTD_endStream() instructs to finish a frame. +* It will perform a flush and write frame epilogue. +* The epilogue is required for decoders to consider a frame completed. +* Similar to ZSTD_flushStream(), it may not be able to flush the full content if `output->size` is too small. +* In which case, call again ZSTD_endStream() to complete the flush. +* @return : nb of bytes still present within internal buffer (0 if it's empty, hence compression completed) +* or an error code, which can be tested using ZSTD_isError(). +* +* *******************************************************************/ + +typedef struct ZSTD_CStream_s ZSTD_CStream; +/*===== ZSTD_CStream management functions =====*/ +ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream(void); +ZSTDLIB_API size_t ZSTD_freeCStream(ZSTD_CStream* zcs); + +/*===== Streaming compression functions =====*/ +ZSTDLIB_API size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel); +ZSTDLIB_API size_t ZSTD_compressStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input); +ZSTDLIB_API size_t ZSTD_flushStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output); +ZSTDLIB_API size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output); + +ZSTDLIB_API size_t ZSTD_CStreamInSize(void); /**< recommended size for input buffer */ +ZSTDLIB_API size_t ZSTD_CStreamOutSize(void); /**< recommended size for output buffer. Guarantee to successfully flush at least one complete compressed block in all circumstances. */ + + + +/*-*************************************************************************** +* Streaming decompression - HowTo +* +* A ZSTD_DStream object is required to track streaming operations. +* Use ZSTD_createDStream() and ZSTD_freeDStream() to create/release resources. +* ZSTD_DStream objects can be re-used multiple times. +* +* Use ZSTD_initDStream() to start a new decompression operation, +* or ZSTD_initDStream_usingDict() if decompression requires a dictionary. +* @return : recommended first input size +* +* Use ZSTD_decompressStream() repetitively to consume your input. +* The function will update both `pos` fields. +* If `input.pos < input.size`, some input has not been consumed. +* It's up to the caller to present again remaining data. +* If `output.pos < output.size`, decoder has flushed everything it could. +* @return : 0 when a frame is completely decoded and fully flushed, +* an error code, which can be tested using ZSTD_isError(), +* any other value > 0, which means there is still some decoding to do to complete current frame. +* The return value is a suggested next input size (a hint to improve latency) that will never load more than the current frame. +* *******************************************************************************/ + +typedef struct ZSTD_DStream_s ZSTD_DStream; +/*===== ZSTD_DStream management functions =====*/ +ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream(void); +ZSTDLIB_API size_t ZSTD_freeDStream(ZSTD_DStream* zds); + +/*===== Streaming decompression functions =====*/ +ZSTDLIB_API size_t ZSTD_initDStream(ZSTD_DStream* zds); +ZSTDLIB_API size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input); + +ZSTDLIB_API size_t ZSTD_DStreamInSize(void); /*!< recommended size for input buffer */ +ZSTDLIB_API size_t ZSTD_DStreamOutSize(void); /*!< recommended size for output buffer. Guarantee to successfully flush at least one complete block in all circumstances. */ + +#endif /* ZSTD_H_235446 */ + + +#if defined(ZSTD_STATIC_LINKING_ONLY) && !defined(ZSTD_H_ZSTD_STATIC_LINKING_ONLY) +#define ZSTD_H_ZSTD_STATIC_LINKING_ONLY + +/**************************************************************************************** + * START OF ADVANCED AND EXPERIMENTAL FUNCTIONS + * The definitions in this section are considered experimental. + * They should never be used with a dynamic library, as they may change in the future. + * They are provided for advanced usages. + * Use them only in association with static linking. + * ***************************************************************************************/ + +/* --- Constants ---*/ +#define ZSTD_MAGICNUMBER 0xFD2FB528 /* >= v0.8.0 */ +#define ZSTD_MAGIC_SKIPPABLE_START 0x184D2A50U + +#define ZSTD_CONTENTSIZE_UNKNOWN (0ULL - 1) +#define ZSTD_CONTENTSIZE_ERROR (0ULL - 2) + +#define ZSTD_WINDOWLOG_MAX_32 27 +#define ZSTD_WINDOWLOG_MAX_64 27 +#define ZSTD_WINDOWLOG_MAX ((unsigned)(sizeof(size_t) == 4 ? ZSTD_WINDOWLOG_MAX_32 : ZSTD_WINDOWLOG_MAX_64)) +#define ZSTD_WINDOWLOG_MIN 10 +#define ZSTD_HASHLOG_MAX ZSTD_WINDOWLOG_MAX +#define ZSTD_HASHLOG_MIN 6 +#define ZSTD_CHAINLOG_MAX (ZSTD_WINDOWLOG_MAX+1) +#define ZSTD_CHAINLOG_MIN ZSTD_HASHLOG_MIN +#define ZSTD_HASHLOG3_MAX 17 +#define ZSTD_SEARCHLOG_MAX (ZSTD_WINDOWLOG_MAX-1) +#define ZSTD_SEARCHLOG_MIN 1 +#define ZSTD_SEARCHLENGTH_MAX 7 /* only for ZSTD_fast, other strategies are limited to 6 */ +#define ZSTD_SEARCHLENGTH_MIN 3 /* only for ZSTD_btopt, other strategies are limited to 4 */ +#define ZSTD_TARGETLENGTH_MIN 4 +#define ZSTD_TARGETLENGTH_MAX 999 + +#define ZSTD_FRAMEHEADERSIZE_MAX 18 /* for static allocation */ +#define ZSTD_FRAMEHEADERSIZE_MIN 6 +static const size_t ZSTD_frameHeaderSize_prefix = 5; +static const size_t ZSTD_frameHeaderSize_min = ZSTD_FRAMEHEADERSIZE_MIN; +static const size_t ZSTD_frameHeaderSize_max = ZSTD_FRAMEHEADERSIZE_MAX; +static const size_t ZSTD_skippableHeaderSize = 8; /* magic number + skippable frame length */ + + +/*--- Advanced types ---*/ +typedef enum { ZSTD_fast, ZSTD_dfast, ZSTD_greedy, ZSTD_lazy, ZSTD_lazy2, ZSTD_btlazy2, ZSTD_btopt, ZSTD_btopt2 } ZSTD_strategy; /* from faster to stronger */ + +typedef struct { + unsigned windowLog; /**< largest match distance : larger == more compression, more memory needed during decompression */ + unsigned chainLog; /**< fully searched segment : larger == more compression, slower, more memory (useless for fast) */ + unsigned hashLog; /**< dispatch table : larger == faster, more memory */ + unsigned searchLog; /**< nb of searches : larger == more compression, slower */ + unsigned searchLength; /**< match length searched : larger == faster decompression, sometimes less compression */ + unsigned targetLength; /**< acceptable match size for optimal parser (only) : larger == more compression, slower */ + ZSTD_strategy strategy; +} ZSTD_compressionParameters; + +typedef struct { + unsigned contentSizeFlag; /**< 1: content size will be in frame header (when known) */ + unsigned checksumFlag; /**< 1: generate a 32-bits checksum at end of frame, for error detection */ + unsigned noDictIDFlag; /**< 1: no dictID will be saved into frame header (if dictionary compression) */ +} ZSTD_frameParameters; + +typedef struct { + ZSTD_compressionParameters cParams; + ZSTD_frameParameters fParams; +} ZSTD_parameters; + +/*= Custom memory allocation functions */ +typedef void* (*ZSTD_allocFunction) (void* opaque, size_t size); +typedef void (*ZSTD_freeFunction) (void* opaque, void* address); +typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; void* opaque; } ZSTD_customMem; + +/*************************************** +* Compressed size functions +***************************************/ + +/*! ZSTD_findFrameCompressedSize() : + * `src` should point to the start of a ZSTD encoded frame or skippable frame + * `srcSize` must be at least as large as the frame + * @return : the compressed size of the frame pointed to by `src`, suitable to pass to + * `ZSTD_decompress` or similar, or an error code if given invalid input. */ +ZSTDLIB_API size_t ZSTD_findFrameCompressedSize(const void* src, size_t srcSize); + +/*************************************** +* Decompressed size functions +***************************************/ +/*! ZSTD_getFrameContentSize() : +* `src` should point to the start of a ZSTD encoded frame +* `srcSize` must be at least as large as the frame header. A value greater than or equal +* to `ZSTD_frameHeaderSize_max` is guaranteed to be large enough in all cases. +* @return : decompressed size of the frame pointed to be `src` if known, otherwise +* - ZSTD_CONTENTSIZE_UNKNOWN if the size cannot be determined +* - ZSTD_CONTENTSIZE_ERROR if an error occurred (e.g. invalid magic number, srcSize too small) */ +ZSTDLIB_API unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize); + +/*! ZSTD_findDecompressedSize() : +* `src` should point the start of a series of ZSTD encoded and/or skippable frames +* `srcSize` must be the _exact_ size of this series +* (i.e. there should be a frame boundary exactly `srcSize` bytes after `src`) +* @return : the decompressed size of all data in the contained frames, as a 64-bit value _if known_ +* - if the decompressed size cannot be determined: ZSTD_CONTENTSIZE_UNKNOWN +* - if an error occurred: ZSTD_CONTENTSIZE_ERROR +* +* note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode. +* When `return==ZSTD_CONTENTSIZE_UNKNOWN`, data to decompress could be any size. +* In which case, it's necessary to use streaming mode to decompress data. +* Optionally, application can still use ZSTD_decompress() while relying on implied limits. +* (For example, data may be necessarily cut into blocks <= 16 KB). +* note 2 : decompressed size is always present when compression is done with ZSTD_compress() +* note 3 : decompressed size can be very large (64-bits value), +* potentially larger than what local system can handle as a single memory segment. +* In which case, it's necessary to use streaming mode to decompress data. +* note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified. +* Always ensure result fits within application's authorized limits. +* Each application can set its own limits. +* note 5 : ZSTD_findDecompressedSize handles multiple frames, and so it must traverse the input to +* read each contained frame header. This is efficient as most of the data is skipped, +* however it does mean that all frame data must be present and valid. */ +ZSTDLIB_API unsigned long long ZSTD_findDecompressedSize(const void* src, size_t srcSize); + + +/*************************************** +* Advanced compression functions +***************************************/ +/*! ZSTD_estimateCCtxSize() : + * Gives the amount of memory allocated for a ZSTD_CCtx given a set of compression parameters. + * `frameContentSize` is an optional parameter, provide `0` if unknown */ +ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams); + +/*! ZSTD_createCCtx_advanced() : + * Create a ZSTD compression context using external alloc and free functions */ +ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx_advanced(ZSTD_customMem customMem); + +/*! ZSTD_sizeofCCtx() : + * Gives the amount of memory used by a given ZSTD_CCtx */ +ZSTDLIB_API size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx); + +typedef enum { + ZSTD_p_forceWindow, /* Force back-references to remain < windowSize, even when referencing Dictionary content (default:0) */ + ZSTD_p_forceRawDict /* Force loading dictionary in "content-only" mode (no header analysis) */ +} ZSTD_CCtxParameter; +/*! ZSTD_setCCtxParameter() : + * Set advanced parameters, selected through enum ZSTD_CCtxParameter + * @result : 0, or an error code (which can be tested with ZSTD_isError()) */ +ZSTDLIB_API size_t ZSTD_setCCtxParameter(ZSTD_CCtx* cctx, ZSTD_CCtxParameter param, unsigned value); + +/*! ZSTD_createCDict_byReference() : + * Create a digested dictionary for compression + * Dictionary content is simply referenced, and therefore stays in dictBuffer. + * It is important that dictBuffer outlives CDict, it must remain read accessible throughout the lifetime of CDict */ +ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_byReference(const void* dictBuffer, size_t dictSize, int compressionLevel); + +/*! ZSTD_createCDict_advanced() : + * Create a ZSTD_CDict using external alloc and free, and customized compression parameters */ +ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_advanced(const void* dict, size_t dictSize, unsigned byReference, + ZSTD_parameters params, ZSTD_customMem customMem); + +/*! ZSTD_sizeof_CDict() : + * Gives the amount of memory used by a given ZSTD_sizeof_CDict */ +ZSTDLIB_API size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict); + +/*! ZSTD_getCParams() : +* @return ZSTD_compressionParameters structure for a selected compression level and estimated srcSize. +* `estimatedSrcSize` value is optional, select 0 if not known */ +ZSTDLIB_API ZSTD_compressionParameters ZSTD_getCParams(int compressionLevel, unsigned long long estimatedSrcSize, size_t dictSize); + +/*! ZSTD_getParams() : +* same as ZSTD_getCParams(), but @return a full `ZSTD_parameters` object instead of sub-component `ZSTD_compressionParameters`. +* All fields of `ZSTD_frameParameters` are set to default (0) */ +ZSTDLIB_API ZSTD_parameters ZSTD_getParams(int compressionLevel, unsigned long long estimatedSrcSize, size_t dictSize); + +/*! ZSTD_checkCParams() : +* Ensure param values remain within authorized range */ +ZSTDLIB_API size_t ZSTD_checkCParams(ZSTD_compressionParameters params); + +/*! ZSTD_adjustCParams() : +* optimize params for a given `srcSize` and `dictSize`. +* both values are optional, select `0` if unknown. */ +ZSTDLIB_API ZSTD_compressionParameters ZSTD_adjustCParams(ZSTD_compressionParameters cPar, unsigned long long srcSize, size_t dictSize); + +/*! ZSTD_compress_advanced() : +* Same as ZSTD_compress_usingDict(), with fine-tune control of each compression parameter */ +ZSTDLIB_API size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict,size_t dictSize, + ZSTD_parameters params); + + +/*--- Advanced decompression functions ---*/ + +/*! ZSTD_isFrame() : + * Tells if the content of `buffer` starts with a valid Frame Identifier. + * Note : Frame Identifier is 4 bytes. If `size < 4`, @return will always be 0. + * Note 2 : Legacy Frame Identifiers are considered valid only if Legacy Support is enabled. + * Note 3 : Skippable Frame Identifiers are considered valid. */ +ZSTDLIB_API unsigned ZSTD_isFrame(const void* buffer, size_t size); + +/*! ZSTD_estimateDCtxSize() : + * Gives the potential amount of memory allocated to create a ZSTD_DCtx */ +ZSTDLIB_API size_t ZSTD_estimateDCtxSize(void); + +/*! ZSTD_createDCtx_advanced() : + * Create a ZSTD decompression context using external alloc and free functions */ +ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx_advanced(ZSTD_customMem customMem); + +/*! ZSTD_sizeof_DCtx() : + * Gives the amount of memory used by a given ZSTD_DCtx */ +ZSTDLIB_API size_t ZSTD_sizeof_DCtx(const ZSTD_DCtx* dctx); + +/*! ZSTD_createDDict_byReference() : + * Create a digested dictionary, ready to start decompression operation without startup delay. + * Dictionary content is simply referenced, and therefore stays in dictBuffer. + * It is important that dictBuffer outlives DDict, it must remain read accessible throughout the lifetime of DDict */ +ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_byReference(const void* dictBuffer, size_t dictSize); + +/*! ZSTD_createDDict_advanced() : + * Create a ZSTD_DDict using external alloc and free, optionally by reference */ +ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize, + unsigned byReference, ZSTD_customMem customMem); + +/*! ZSTD_sizeof_DDict() : + * Gives the amount of memory used by a given ZSTD_DDict */ +ZSTDLIB_API size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict); + +/*! ZSTD_getDictID_fromDict() : + * Provides the dictID stored within dictionary. + * if @return == 0, the dictionary is not conformant with Zstandard specification. + * It can still be loaded, but as a content-only dictionary. */ +ZSTDLIB_API unsigned ZSTD_getDictID_fromDict(const void* dict, size_t dictSize); + +/*! ZSTD_getDictID_fromDDict() : + * Provides the dictID of the dictionary loaded into `ddict`. + * If @return == 0, the dictionary is not conformant to Zstandard specification, or empty. + * Non-conformant dictionaries can still be loaded, but as content-only dictionaries. */ +ZSTDLIB_API unsigned ZSTD_getDictID_fromDDict(const ZSTD_DDict* ddict); + +/*! ZSTD_getDictID_fromFrame() : + * Provides the dictID required to decompressed the frame stored within `src`. + * If @return == 0, the dictID could not be decoded. + * This could for one of the following reasons : + * - The frame does not require a dictionary to be decoded (most common case). + * - The frame was built with dictID intentionally removed. Whatever dictionary is necessary is a hidden information. + * Note : this use case also happens when using a non-conformant dictionary. + * - `srcSize` is too small, and as a result, the frame header could not be decoded (only possible if `srcSize < ZSTD_FRAMEHEADERSIZE_MAX`). + * - This is not a Zstandard frame. + * When identifying the exact failure cause, it's possible to used ZSTD_getFrameParams(), which will provide a more precise error code. */ +ZSTDLIB_API unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize); + + +/******************************************************************** +* Advanced streaming functions +********************************************************************/ + +/*===== Advanced Streaming compression functions =====*/ +ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem); +ZSTDLIB_API size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize); /**< pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ +ZSTDLIB_API size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */ +ZSTDLIB_API size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize, + ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */ +ZSTDLIB_API size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict); /**< note : cdict will just be referenced, and must outlive compression session */ +ZSTDLIB_API size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize); /**< re-use compression parameters from previous init; skip dictionary loading stage; zcs must be init at least once before. note: pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ +ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs); + + +/*===== Advanced Streaming decompression functions =====*/ +typedef enum { DStream_p_maxWindowSize } ZSTD_DStreamParameter_e; +ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem); +ZSTDLIB_API size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */ +ZSTDLIB_API size_t ZSTD_setDStreamParameter(ZSTD_DStream* zds, ZSTD_DStreamParameter_e paramType, unsigned paramValue); +ZSTDLIB_API size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, const ZSTD_DDict* ddict); /**< note : ddict will just be referenced, and must outlive decompression session */ +ZSTDLIB_API size_t ZSTD_resetDStream(ZSTD_DStream* zds); /**< re-use decompression parameters from previous init; saves dictionary loading */ +ZSTDLIB_API size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds); + + +/********************************************************************* +* Buffer-less and synchronous inner streaming functions +* +* This is an advanced API, giving full control over buffer management, for users which need direct control over memory. +* But it's also a complex one, with many restrictions (documented below). +* Prefer using normal streaming API for an easier experience +********************************************************************* */ + +/** + Buffer-less streaming compression (synchronous mode) + + A ZSTD_CCtx object is required to track streaming operations. + Use ZSTD_createCCtx() / ZSTD_freeCCtx() to manage resource. + ZSTD_CCtx object can be re-used multiple times within successive compression operations. + + Start by initializing a context. + Use ZSTD_compressBegin(), or ZSTD_compressBegin_usingDict() for dictionary compression, + or ZSTD_compressBegin_advanced(), for finer parameter control. + It's also possible to duplicate a reference context which has already been initialized, using ZSTD_copyCCtx() + + Then, consume your input using ZSTD_compressContinue(). + There are some important considerations to keep in mind when using this advanced function : + - ZSTD_compressContinue() has no internal buffer. It uses externally provided buffer only. + - Interface is synchronous : input is consumed entirely and produce 1+ (or more) compressed blocks. + - Caller must ensure there is enough space in `dst` to store compressed data under worst case scenario. + Worst case evaluation is provided by ZSTD_compressBound(). + ZSTD_compressContinue() doesn't guarantee recover after a failed compression. + - ZSTD_compressContinue() presumes prior input ***is still accessible and unmodified*** (up to maximum distance size, see WindowLog). + It remembers all previous contiguous blocks, plus one separated memory segment (which can itself consists of multiple contiguous blocks) + - ZSTD_compressContinue() detects that prior input has been overwritten when `src` buffer overlaps. + In which case, it will "discard" the relevant memory section from its history. + + Finish a frame with ZSTD_compressEnd(), which will write the last block(s) and optional checksum. + It's possible to use srcSize==0, in which case, it will write a final empty block to end the frame. + Without last block mark, frames will be considered unfinished (corrupted) by decoders. + + `ZSTD_CCtx` object can be re-used (ZSTD_compressBegin()) to compress some new frame. +*/ + +/*===== Buffer-less streaming compression functions =====*/ +ZSTDLIB_API size_t ZSTD_compressBegin(ZSTD_CCtx* cctx, int compressionLevel); +ZSTDLIB_API size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel); +ZSTDLIB_API size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */ +ZSTDLIB_API size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**< note: if pledgedSrcSize can be 0, indicating unknown size. if it is non-zero, it must be accurate. for 0 size frames, use compressBegin_advanced */ +ZSTDLIB_API size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: if pledgedSrcSize can be 0, indicating unknown size. if it is non-zero, it must be accurate. for 0 size frames, use compressBegin_advanced */ +ZSTDLIB_API size_t ZSTD_compressContinue(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); +ZSTDLIB_API size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); + + + +/*- + Buffer-less streaming decompression (synchronous mode) + + A ZSTD_DCtx object is required to track streaming operations. + Use ZSTD_createDCtx() / ZSTD_freeDCtx() to manage it. + A ZSTD_DCtx object can be re-used multiple times. + + First typical operation is to retrieve frame parameters, using ZSTD_getFrameParams(). + It fills a ZSTD_frameParams structure which provide important information to correctly decode the frame, + such as the minimum rolling buffer size to allocate to decompress data (`windowSize`), + and the dictionary ID used. + (Note : content size is optional, it may not be present. 0 means : content size unknown). + Note that these values could be wrong, either because of data malformation, or because an attacker is spoofing deliberate false information. + As a consequence, check that values remain within valid application range, especially `windowSize`, before allocation. + Each application can set its own limit, depending on local restrictions. For extended interoperability, it is recommended to support at least 8 MB. + Frame parameters are extracted from the beginning of the compressed frame. + Data fragment must be large enough to ensure successful decoding, typically `ZSTD_frameHeaderSize_max` bytes. + @result : 0 : successful decoding, the `ZSTD_frameParams` structure is correctly filled. + >0 : `srcSize` is too small, please provide at least @result bytes on next attempt. + errorCode, which can be tested using ZSTD_isError(). + + Start decompression, with ZSTD_decompressBegin() or ZSTD_decompressBegin_usingDict(). + Alternatively, you can copy a prepared context, using ZSTD_copyDCtx(). + + Then use ZSTD_nextSrcSizeToDecompress() and ZSTD_decompressContinue() alternatively. + ZSTD_nextSrcSizeToDecompress() tells how many bytes to provide as 'srcSize' to ZSTD_decompressContinue(). + ZSTD_decompressContinue() requires this _exact_ amount of bytes, or it will fail. + + @result of ZSTD_decompressContinue() is the number of bytes regenerated within 'dst' (necessarily <= dstCapacity). + It can be zero, which is not an error; it just means ZSTD_decompressContinue() has decoded some metadata item. + It can also be an error code, which can be tested with ZSTD_isError(). + + ZSTD_decompressContinue() needs previous data blocks during decompression, up to `windowSize`. + They should preferably be located contiguously, prior to current block. + Alternatively, a round buffer of sufficient size is also possible. Sufficient size is determined by frame parameters. + ZSTD_decompressContinue() is very sensitive to contiguity, + if 2 blocks don't follow each other, make sure that either the compressor breaks contiguity at the same place, + or that previous contiguous segment is large enough to properly handle maximum back-reference. + + A frame is fully decoded when ZSTD_nextSrcSizeToDecompress() returns zero. + Context can then be reset to start a new decompression. + + Note : it's possible to know if next input to present is a header or a block, using ZSTD_nextInputType(). + This information is not required to properly decode a frame. + + == Special case : skippable frames == + + Skippable frames allow integration of user-defined data into a flow of concatenated frames. + Skippable frames will be ignored (skipped) by a decompressor. The format of skippable frames is as follows : + a) Skippable frame ID - 4 Bytes, Little endian format, any value from 0x184D2A50 to 0x184D2A5F + b) Frame Size - 4 Bytes, Little endian format, unsigned 32-bits + c) Frame Content - any content (User Data) of length equal to Frame Size + For skippable frames ZSTD_decompressContinue() always returns 0. + For skippable frames ZSTD_getFrameParams() returns fparamsPtr->windowLog==0 what means that a frame is skippable. + Note : If fparamsPtr->frameContentSize==0, it is ambiguous: the frame might actually be a Zstd encoded frame with no content. + For purposes of decompression, it is valid in both cases to skip the frame using + ZSTD_findFrameCompressedSize to find its size in bytes. + It also returns Frame Size as fparamsPtr->frameContentSize. +*/ + +typedef struct { + unsigned long long frameContentSize; + unsigned windowSize; + unsigned dictID; + unsigned checksumFlag; +} ZSTD_frameParams; + +/*===== Buffer-less streaming decompression functions =====*/ +ZSTDLIB_API size_t ZSTD_getFrameParams(ZSTD_frameParams* fparamsPtr, const void* src, size_t srcSize); /**< doesn't consume input, see details below */ +ZSTDLIB_API size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx); +ZSTDLIB_API size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx* dctx, const void* dict, size_t dictSize); +ZSTDLIB_API void ZSTD_copyDCtx(ZSTD_DCtx* dctx, const ZSTD_DCtx* preparedDCtx); +ZSTDLIB_API size_t ZSTD_nextSrcSizeToDecompress(ZSTD_DCtx* dctx); +ZSTDLIB_API size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); +typedef enum { ZSTDnit_frameHeader, ZSTDnit_blockHeader, ZSTDnit_block, ZSTDnit_lastBlock, ZSTDnit_checksum, ZSTDnit_skippableFrame } ZSTD_nextInputType_e; +ZSTDLIB_API ZSTD_nextInputType_e ZSTD_nextInputType(ZSTD_DCtx* dctx); + +/** + Block functions + + Block functions produce and decode raw zstd blocks, without frame metadata. + Frame metadata cost is typically ~18 bytes, which can be non-negligible for very small blocks (< 100 bytes). + User will have to take in charge required information to regenerate data, such as compressed and content sizes. + + A few rules to respect : + - Compressing and decompressing require a context structure + + Use ZSTD_createCCtx() and ZSTD_createDCtx() + - It is necessary to init context before starting + + compression : ZSTD_compressBegin() + + decompression : ZSTD_decompressBegin() + + variants _usingDict() are also allowed + + copyCCtx() and copyDCtx() work too + - Block size is limited, it must be <= ZSTD_getBlockSizeMax() + + If you need to compress more, cut data into multiple blocks + + Consider using the regular ZSTD_compress() instead, as frame metadata costs become negligible when source size is large. + - When a block is considered not compressible enough, ZSTD_compressBlock() result will be zero. + In which case, nothing is produced into `dst`. + + User must test for such outcome and deal directly with uncompressed data + + ZSTD_decompressBlock() doesn't accept uncompressed data as input !!! + + In case of multiple successive blocks, decoder must be informed of uncompressed block existence to follow proper history. + Use ZSTD_insertBlock() in such a case. +*/ + +#define ZSTD_BLOCKSIZE_ABSOLUTEMAX (128 * 1024) /* define, for static allocation */ +/*===== Raw zstd block functions =====*/ +ZSTDLIB_API size_t ZSTD_getBlockSizeMax(ZSTD_CCtx* cctx); +ZSTDLIB_API size_t ZSTD_compressBlock (ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); +ZSTDLIB_API size_t ZSTD_decompressBlock(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); +ZSTDLIB_API size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize); /**< insert block into `dctx` history. Useful for uncompressed blocks */ + + +#endif /* ZSTD_H_ZSTD_STATIC_LINKING_ONLY */ + +#if defined (__cplusplus) +} +#endif diff --git a/contrib/linux-kernel/lib/bitstream.h b/contrib/linux-kernel/lib/bitstream.h new file mode 100644 index 000000000..0e3d2fc55 --- /dev/null +++ b/contrib/linux-kernel/lib/bitstream.h @@ -0,0 +1,417 @@ +/* ****************************************************************** + bitstream + Part of FSE library + header file (to include) + Copyright (C) 2013-2016, Yann Collet. + + BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php) + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are + met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the + distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + You can contact the author at : + - Source repository : https://github.com/Cyan4973/FiniteStateEntropy +****************************************************************** */ +#ifndef BITSTREAM_H_MODULE +#define BITSTREAM_H_MODULE + +#if defined (__cplusplus) +extern "C" { +#endif + + +/* +* This API consists of small unitary functions, which must be inlined for best performance. +* Since link-time-optimization is not available for all compilers, +* these functions are defined into a .h to be included. +*/ + +/*-**************************************** +* Dependencies +******************************************/ +#include "mem.h" /* unaligned access routines */ +#include "error_private.h" /* error codes and messages */ + + +/*========================================= +* Target specific +=========================================*/ +#if defined(__BMI__) && defined(__GNUC__) +# include /* support for bextr (experimental) */ +#endif + +#define STREAM_ACCUMULATOR_MIN_32 25 +#define STREAM_ACCUMULATOR_MIN_64 57 +#define STREAM_ACCUMULATOR_MIN ((U32)(MEM_32bits() ? STREAM_ACCUMULATOR_MIN_32 : STREAM_ACCUMULATOR_MIN_64)) + +/*-****************************************** +* bitStream encoding API (write forward) +********************************************/ +/* bitStream can mix input from multiple sources. +* A critical property of these streams is that they encode and decode in **reverse** direction. +* So the first bit sequence you add will be the last to be read, like a LIFO stack. +*/ +typedef struct +{ + size_t bitContainer; + int bitPos; + char* startPtr; + char* ptr; + char* endPtr; +} BIT_CStream_t; + +MEM_STATIC size_t BIT_initCStream(BIT_CStream_t* bitC, void* dstBuffer, size_t dstCapacity); +MEM_STATIC void BIT_addBits(BIT_CStream_t* bitC, size_t value, unsigned nbBits); +MEM_STATIC void BIT_flushBits(BIT_CStream_t* bitC); +MEM_STATIC size_t BIT_closeCStream(BIT_CStream_t* bitC); + +/* Start with initCStream, providing the size of buffer to write into. +* bitStream will never write outside of this buffer. +* `dstCapacity` must be >= sizeof(bitD->bitContainer), otherwise @return will be an error code. +* +* bits are first added to a local register. +* Local register is size_t, hence 64-bits on 64-bits systems, or 32-bits on 32-bits systems. +* Writing data into memory is an explicit operation, performed by the flushBits function. +* Hence keep track how many bits are potentially stored into local register to avoid register overflow. +* After a flushBits, a maximum of 7 bits might still be stored into local register. +* +* Avoid storing elements of more than 24 bits if you want compatibility with 32-bits bitstream readers. +* +* Last operation is to close the bitStream. +* The function returns the final size of CStream in bytes. +* If data couldn't fit into `dstBuffer`, it will return a 0 ( == not storable) +*/ + + +/*-******************************************** +* bitStream decoding API (read backward) +**********************************************/ +typedef struct +{ + size_t bitContainer; + unsigned bitsConsumed; + const char* ptr; + const char* start; +} BIT_DStream_t; + +typedef enum { BIT_DStream_unfinished = 0, + BIT_DStream_endOfBuffer = 1, + BIT_DStream_completed = 2, + BIT_DStream_overflow = 3 } BIT_DStream_status; /* result of BIT_reloadDStream() */ + /* 1,2,4,8 would be better for bitmap combinations, but slows down performance a bit ... :( */ + +MEM_STATIC size_t BIT_initDStream(BIT_DStream_t* bitD, const void* srcBuffer, size_t srcSize); +MEM_STATIC size_t BIT_readBits(BIT_DStream_t* bitD, unsigned nbBits); +MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD); +MEM_STATIC unsigned BIT_endOfDStream(const BIT_DStream_t* bitD); + + +/* Start by invoking BIT_initDStream(). +* A chunk of the bitStream is then stored into a local register. +* Local register size is 64-bits on 64-bits systems, 32-bits on 32-bits systems (size_t). +* You can then retrieve bitFields stored into the local register, **in reverse order**. +* Local register is explicitly reloaded from memory by the BIT_reloadDStream() method. +* A reload guarantee a minimum of ((8*sizeof(bitD->bitContainer))-7) bits when its result is BIT_DStream_unfinished. +* Otherwise, it can be less than that, so proceed accordingly. +* Checking if DStream has reached its end can be performed with BIT_endOfDStream(). +*/ + + +/*-**************************************** +* unsafe API +******************************************/ +MEM_STATIC void BIT_addBitsFast(BIT_CStream_t* bitC, size_t value, unsigned nbBits); +/* faster, but works only if value is "clean", meaning all high bits above nbBits are 0 */ + +MEM_STATIC void BIT_flushBitsFast(BIT_CStream_t* bitC); +/* unsafe version; does not check buffer overflow */ + +MEM_STATIC size_t BIT_readBitsFast(BIT_DStream_t* bitD, unsigned nbBits); +/* faster, but works only if nbBits >= 1 */ + + + +/*-************************************************************** +* Internal functions +****************************************************************/ +MEM_STATIC unsigned BIT_highbit32 (register U32 val) +{ +# if defined(_MSC_VER) /* Visual */ + unsigned long r=0; + _BitScanReverse ( &r, val ); + return (unsigned) r; +# elif defined(__GNUC__) && (__GNUC__ >= 3) /* Use GCC Intrinsic */ + return 31 - __builtin_clz (val); +# else /* Software version */ + static const unsigned DeBruijnClz[32] = { 0, 9, 1, 10, 13, 21, 2, 29, 11, 14, 16, 18, 22, 25, 3, 30, 8, 12, 20, 28, 15, 17, 24, 7, 19, 27, 23, 6, 26, 5, 4, 31 }; + U32 v = val; + v |= v >> 1; + v |= v >> 2; + v |= v >> 4; + v |= v >> 8; + v |= v >> 16; + return DeBruijnClz[ (U32) (v * 0x07C4ACDDU) >> 27]; +# endif +} + +/*===== Local Constants =====*/ +static const unsigned BIT_mask[] = { 0, 1, 3, 7, 0xF, 0x1F, 0x3F, 0x7F, 0xFF, 0x1FF, 0x3FF, 0x7FF, 0xFFF, 0x1FFF, 0x3FFF, 0x7FFF, 0xFFFF, 0x1FFFF, 0x3FFFF, 0x7FFFF, 0xFFFFF, 0x1FFFFF, 0x3FFFFF, 0x7FFFFF, 0xFFFFFF, 0x1FFFFFF, 0x3FFFFFF }; /* up to 26 bits */ + + +/*-************************************************************** +* bitStream encoding +****************************************************************/ +/*! BIT_initCStream() : + * `dstCapacity` must be > sizeof(void*) + * @return : 0 if success, + otherwise an error code (can be tested using ERR_isError() ) */ +MEM_STATIC size_t BIT_initCStream(BIT_CStream_t* bitC, void* startPtr, size_t dstCapacity) +{ + bitC->bitContainer = 0; + bitC->bitPos = 0; + bitC->startPtr = (char*)startPtr; + bitC->ptr = bitC->startPtr; + bitC->endPtr = bitC->startPtr + dstCapacity - sizeof(bitC->ptr); + if (dstCapacity <= sizeof(bitC->ptr)) return ERROR(dstSize_tooSmall); + return 0; +} + +/*! BIT_addBits() : + can add up to 26 bits into `bitC`. + Does not check for register overflow ! */ +MEM_STATIC void BIT_addBits(BIT_CStream_t* bitC, size_t value, unsigned nbBits) +{ + bitC->bitContainer |= (value & BIT_mask[nbBits]) << bitC->bitPos; + bitC->bitPos += nbBits; +} + +/*! BIT_addBitsFast() : + * works only if `value` is _clean_, meaning all high bits above nbBits are 0 */ +MEM_STATIC void BIT_addBitsFast(BIT_CStream_t* bitC, size_t value, unsigned nbBits) +{ + bitC->bitContainer |= value << bitC->bitPos; + bitC->bitPos += nbBits; +} + +/*! BIT_flushBitsFast() : + * unsafe version; does not check buffer overflow */ +MEM_STATIC void BIT_flushBitsFast(BIT_CStream_t* bitC) +{ + size_t const nbBytes = bitC->bitPos >> 3; + MEM_writeLEST(bitC->ptr, bitC->bitContainer); + bitC->ptr += nbBytes; + bitC->bitPos &= 7; + bitC->bitContainer >>= nbBytes*8; /* if bitPos >= sizeof(bitContainer)*8 --> undefined behavior */ +} + +/*! BIT_flushBits() : + * safe version; check for buffer overflow, and prevents it. + * note : does not signal buffer overflow. This will be revealed later on using BIT_closeCStream() */ +MEM_STATIC void BIT_flushBits(BIT_CStream_t* bitC) +{ + size_t const nbBytes = bitC->bitPos >> 3; + MEM_writeLEST(bitC->ptr, bitC->bitContainer); + bitC->ptr += nbBytes; + if (bitC->ptr > bitC->endPtr) bitC->ptr = bitC->endPtr; + bitC->bitPos &= 7; + bitC->bitContainer >>= nbBytes*8; /* if bitPos >= sizeof(bitContainer)*8 --> undefined behavior */ +} + +/*! BIT_closeCStream() : + * @return : size of CStream, in bytes, + or 0 if it could not fit into dstBuffer */ +MEM_STATIC size_t BIT_closeCStream(BIT_CStream_t* bitC) +{ + BIT_addBitsFast(bitC, 1, 1); /* endMark */ + BIT_flushBits(bitC); + + if (bitC->ptr >= bitC->endPtr) return 0; /* doesn't fit within authorized budget : cancel */ + + return (bitC->ptr - bitC->startPtr) + (bitC->bitPos > 0); +} + + +/*-******************************************************** +* bitStream decoding +**********************************************************/ +/*! BIT_initDStream() : +* Initialize a BIT_DStream_t. +* `bitD` : a pointer to an already allocated BIT_DStream_t structure. +* `srcSize` must be the *exact* size of the bitStream, in bytes. +* @return : size of stream (== srcSize) or an errorCode if a problem is detected +*/ +MEM_STATIC size_t BIT_initDStream(BIT_DStream_t* bitD, const void* srcBuffer, size_t srcSize) +{ + if (srcSize < 1) { memset(bitD, 0, sizeof(*bitD)); return ERROR(srcSize_wrong); } + + if (srcSize >= sizeof(bitD->bitContainer)) { /* normal case */ + bitD->start = (const char*)srcBuffer; + bitD->ptr = (const char*)srcBuffer + srcSize - sizeof(bitD->bitContainer); + bitD->bitContainer = MEM_readLEST(bitD->ptr); + { BYTE const lastByte = ((const BYTE*)srcBuffer)[srcSize-1]; + bitD->bitsConsumed = lastByte ? 8 - BIT_highbit32(lastByte) : 0; /* ensures bitsConsumed is always set */ + if (lastByte == 0) return ERROR(GENERIC); /* endMark not present */ } + } else { + bitD->start = (const char*)srcBuffer; + bitD->ptr = bitD->start; + bitD->bitContainer = *(const BYTE*)(bitD->start); + switch(srcSize) + { + case 7: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[6]) << (sizeof(bitD->bitContainer)*8 - 16); + case 6: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[5]) << (sizeof(bitD->bitContainer)*8 - 24); + case 5: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[4]) << (sizeof(bitD->bitContainer)*8 - 32); + case 4: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[3]) << 24; + case 3: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[2]) << 16; + case 2: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[1]) << 8; + default:; + } + { BYTE const lastByte = ((const BYTE*)srcBuffer)[srcSize-1]; + bitD->bitsConsumed = lastByte ? 8 - BIT_highbit32(lastByte) : 0; + if (lastByte == 0) return ERROR(GENERIC); /* endMark not present */ } + bitD->bitsConsumed += (U32)(sizeof(bitD->bitContainer) - srcSize)*8; + } + + return srcSize; +} + +MEM_STATIC size_t BIT_getUpperBits(size_t bitContainer, U32 const start) +{ + return bitContainer >> start; +} + +MEM_STATIC size_t BIT_getMiddleBits(size_t bitContainer, U32 const start, U32 const nbBits) +{ +#if defined(__BMI__) && defined(__GNUC__) && __GNUC__*1000+__GNUC_MINOR__ >= 4008 /* experimental */ +# if defined(__x86_64__) + if (sizeof(bitContainer)==8) + return _bextr_u64(bitContainer, start, nbBits); + else +# endif + return _bextr_u32(bitContainer, start, nbBits); +#else + return (bitContainer >> start) & BIT_mask[nbBits]; +#endif +} + +MEM_STATIC size_t BIT_getLowerBits(size_t bitContainer, U32 const nbBits) +{ + return bitContainer & BIT_mask[nbBits]; +} + +/*! BIT_lookBits() : + * Provides next n bits from local register. + * local register is not modified. + * On 32-bits, maxNbBits==24. + * On 64-bits, maxNbBits==56. + * @return : value extracted + */ + MEM_STATIC size_t BIT_lookBits(const BIT_DStream_t* bitD, U32 nbBits) +{ +#if defined(__BMI__) && defined(__GNUC__) /* experimental; fails if bitD->bitsConsumed + nbBits > sizeof(bitD->bitContainer)*8 */ + return BIT_getMiddleBits(bitD->bitContainer, (sizeof(bitD->bitContainer)*8) - bitD->bitsConsumed - nbBits, nbBits); +#else + U32 const bitMask = sizeof(bitD->bitContainer)*8 - 1; + return ((bitD->bitContainer << (bitD->bitsConsumed & bitMask)) >> 1) >> ((bitMask-nbBits) & bitMask); +#endif +} + +/*! BIT_lookBitsFast() : +* unsafe version; only works only if nbBits >= 1 */ +MEM_STATIC size_t BIT_lookBitsFast(const BIT_DStream_t* bitD, U32 nbBits) +{ + U32 const bitMask = sizeof(bitD->bitContainer)*8 - 1; + return (bitD->bitContainer << (bitD->bitsConsumed & bitMask)) >> (((bitMask+1)-nbBits) & bitMask); +} + +MEM_STATIC void BIT_skipBits(BIT_DStream_t* bitD, U32 nbBits) +{ + bitD->bitsConsumed += nbBits; +} + +/*! BIT_readBits() : + * Read (consume) next n bits from local register and update. + * Pay attention to not read more than nbBits contained into local register. + * @return : extracted value. + */ +MEM_STATIC size_t BIT_readBits(BIT_DStream_t* bitD, U32 nbBits) +{ + size_t const value = BIT_lookBits(bitD, nbBits); + BIT_skipBits(bitD, nbBits); + return value; +} + +/*! BIT_readBitsFast() : +* unsafe version; only works only if nbBits >= 1 */ +MEM_STATIC size_t BIT_readBitsFast(BIT_DStream_t* bitD, U32 nbBits) +{ + size_t const value = BIT_lookBitsFast(bitD, nbBits); + BIT_skipBits(bitD, nbBits); + return value; +} + +/*! BIT_reloadDStream() : +* Refill `bitD` from buffer previously set in BIT_initDStream() . +* This function is safe, it guarantees it will not read beyond src buffer. +* @return : status of `BIT_DStream_t` internal register. + if status == BIT_DStream_unfinished, internal register is filled with >= (sizeof(bitD->bitContainer)*8 - 7) bits */ +MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD) +{ + if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should not happen => corruption detected */ + return BIT_DStream_overflow; + + if (bitD->ptr >= bitD->start + sizeof(bitD->bitContainer)) { + bitD->ptr -= bitD->bitsConsumed >> 3; + bitD->bitsConsumed &= 7; + bitD->bitContainer = MEM_readLEST(bitD->ptr); + return BIT_DStream_unfinished; + } + if (bitD->ptr == bitD->start) { + if (bitD->bitsConsumed < sizeof(bitD->bitContainer)*8) return BIT_DStream_endOfBuffer; + return BIT_DStream_completed; + } + { U32 nbBytes = bitD->bitsConsumed >> 3; + BIT_DStream_status result = BIT_DStream_unfinished; + if (bitD->ptr - nbBytes < bitD->start) { + nbBytes = (U32)(bitD->ptr - bitD->start); /* ptr > start */ + result = BIT_DStream_endOfBuffer; + } + bitD->ptr -= nbBytes; + bitD->bitsConsumed -= nbBytes*8; + bitD->bitContainer = MEM_readLEST(bitD->ptr); /* reminder : srcSize > sizeof(bitD) */ + return result; + } +} + +/*! BIT_endOfDStream() : +* @return Tells if DStream has exactly reached its end (all bits consumed). +*/ +MEM_STATIC unsigned BIT_endOfDStream(const BIT_DStream_t* DStream) +{ + return ((DStream->ptr == DStream->start) && (DStream->bitsConsumed == sizeof(DStream->bitContainer)*8)); +} + +#if defined (__cplusplus) +} +#endif + +#endif /* BITSTREAM_H_MODULE */ diff --git a/contrib/linux-kernel/lib/entropy_common.c b/contrib/linux-kernel/lib/entropy_common.c new file mode 100644 index 000000000..b37a082fe --- /dev/null +++ b/contrib/linux-kernel/lib/entropy_common.c @@ -0,0 +1,221 @@ +/* + Common functions of New Generation Entropy library + Copyright (C) 2016, Yann Collet. + + BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php) + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are + met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the + distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + You can contact the author at : + - FSE+HUF source repository : https://github.com/Cyan4973/FiniteStateEntropy + - Public forum : https://groups.google.com/forum/#!forum/lz4c +*************************************************************************** */ + +/* ************************************* +* Dependencies +***************************************/ +#include "mem.h" +#include "error_private.h" /* ERR_*, ERROR */ +#define FSE_STATIC_LINKING_ONLY /* FSE_MIN_TABLELOG */ +#include "fse.h" +#define HUF_STATIC_LINKING_ONLY /* HUF_TABLELOG_ABSOLUTEMAX */ +#include "huf.h" + + +/*=== Version ===*/ +unsigned FSE_versionNumber(void) { return FSE_VERSION_NUMBER; } + + +/*=== Error Management ===*/ +unsigned FSE_isError(size_t code) { return ERR_isError(code); } +const char* FSE_getErrorName(size_t code) { return ERR_getErrorName(code); } + +unsigned HUF_isError(size_t code) { return ERR_isError(code); } +const char* HUF_getErrorName(size_t code) { return ERR_getErrorName(code); } + + +/*-************************************************************** +* FSE NCount encoding-decoding +****************************************************************/ +size_t FSE_readNCount (short* normalizedCounter, unsigned* maxSVPtr, unsigned* tableLogPtr, + const void* headerBuffer, size_t hbSize) +{ + const BYTE* const istart = (const BYTE*) headerBuffer; + const BYTE* const iend = istart + hbSize; + const BYTE* ip = istart; + int nbBits; + int remaining; + int threshold; + U32 bitStream; + int bitCount; + unsigned charnum = 0; + int previous0 = 0; + + if (hbSize < 4) return ERROR(srcSize_wrong); + bitStream = MEM_readLE32(ip); + nbBits = (bitStream & 0xF) + FSE_MIN_TABLELOG; /* extract tableLog */ + if (nbBits > FSE_TABLELOG_ABSOLUTE_MAX) return ERROR(tableLog_tooLarge); + bitStream >>= 4; + bitCount = 4; + *tableLogPtr = nbBits; + remaining = (1<1) & (charnum<=*maxSVPtr)) { + if (previous0) { + unsigned n0 = charnum; + while ((bitStream & 0xFFFF) == 0xFFFF) { + n0 += 24; + if (ip < iend-5) { + ip += 2; + bitStream = MEM_readLE32(ip) >> bitCount; + } else { + bitStream >>= 16; + bitCount += 16; + } } + while ((bitStream & 3) == 3) { + n0 += 3; + bitStream >>= 2; + bitCount += 2; + } + n0 += bitStream & 3; + bitCount += 2; + if (n0 > *maxSVPtr) return ERROR(maxSymbolValue_tooSmall); + while (charnum < n0) normalizedCounter[charnum++] = 0; + if ((ip <= iend-7) || (ip + (bitCount>>3) <= iend-4)) { + ip += bitCount>>3; + bitCount &= 7; + bitStream = MEM_readLE32(ip) >> bitCount; + } else { + bitStream >>= 2; + } } + { int const max = (2*threshold-1) - remaining; + int count; + + if ((bitStream & (threshold-1)) < (U32)max) { + count = bitStream & (threshold-1); + bitCount += nbBits-1; + } else { + count = bitStream & (2*threshold-1); + if (count >= threshold) count -= max; + bitCount += nbBits; + } + + count--; /* extra accuracy */ + remaining -= count < 0 ? -count : count; /* -1 means +1 */ + normalizedCounter[charnum++] = (short)count; + previous0 = !count; + while (remaining < threshold) { + nbBits--; + threshold >>= 1; + } + + if ((ip <= iend-7) || (ip + (bitCount>>3) <= iend-4)) { + ip += bitCount>>3; + bitCount &= 7; + } else { + bitCount -= (int)(8 * (iend - 4 - ip)); + ip = iend - 4; + } + bitStream = MEM_readLE32(ip) >> (bitCount & 31); + } } /* while ((remaining>1) & (charnum<=*maxSVPtr)) */ + if (remaining != 1) return ERROR(corruption_detected); + if (bitCount > 32) return ERROR(corruption_detected); + *maxSVPtr = charnum-1; + + ip += (bitCount+7)>>3; + return ip-istart; +} + + +/*! HUF_readStats() : + Read compact Huffman tree, saved by HUF_writeCTable(). + `huffWeight` is destination buffer. + `rankStats` is assumed to be a table of at least HUF_TABLELOG_MAX U32. + @return : size read from `src` , or an error Code . + Note : Needed by HUF_readCTable() and HUF_readDTableX?() . +*/ +size_t HUF_readStats(BYTE* huffWeight, size_t hwSize, U32* rankStats, + U32* nbSymbolsPtr, U32* tableLogPtr, + const void* src, size_t srcSize) +{ + U32 weightTotal; + const BYTE* ip = (const BYTE*) src; + size_t iSize; + size_t oSize; + + if (!srcSize) return ERROR(srcSize_wrong); + iSize = ip[0]; + /* memset(huffWeight, 0, hwSize); *//* is not necessary, even though some analyzer complain ... */ + + if (iSize >= 128) { /* special header */ + oSize = iSize - 127; + iSize = ((oSize+1)/2); + if (iSize+1 > srcSize) return ERROR(srcSize_wrong); + if (oSize >= hwSize) return ERROR(corruption_detected); + ip += 1; + { U32 n; + for (n=0; n> 4; + huffWeight[n+1] = ip[n/2] & 15; + } } } + else { /* header compressed with FSE (normal case) */ + FSE_DTable fseWorkspace[FSE_DTABLE_SIZE_U32(6)]; /* 6 is max possible tableLog for HUF header (maybe even 5, to be tested) */ + if (iSize+1 > srcSize) return ERROR(srcSize_wrong); + oSize = FSE_decompress_wksp(huffWeight, hwSize-1, ip+1, iSize, fseWorkspace, 6); /* max (hwSize-1) values decoded, as last one is implied */ + if (FSE_isError(oSize)) return oSize; + } + + /* collect weight stats */ + memset(rankStats, 0, (HUF_TABLELOG_MAX + 1) * sizeof(U32)); + weightTotal = 0; + { U32 n; for (n=0; n= HUF_TABLELOG_MAX) return ERROR(corruption_detected); + rankStats[huffWeight[n]]++; + weightTotal += (1 << huffWeight[n]) >> 1; + } } + if (weightTotal == 0) return ERROR(corruption_detected); + + /* get last non-null symbol weight (implied, total must be 2^n) */ + { U32 const tableLog = BIT_highbit32(weightTotal) + 1; + if (tableLog > HUF_TABLELOG_MAX) return ERROR(corruption_detected); + *tableLogPtr = tableLog; + /* determine last weight */ + { U32 const total = 1 << tableLog; + U32 const rest = total - weightTotal; + U32 const verif = 1 << BIT_highbit32(rest); + U32 const lastWeight = BIT_highbit32(rest) + 1; + if (verif != rest) return ERROR(corruption_detected); /* last value must be a clean power of 2 */ + huffWeight[oSize] = (BYTE)lastWeight; + rankStats[lastWeight]++; + } } + + /* check tree construction validity */ + if ((rankStats[1] < 2) || (rankStats[1] & 1)) return ERROR(corruption_detected); /* by construction : at least 2 elts of rank 1, must be even */ + + /* results */ + *nbSymbolsPtr = (U32)(oSize+1); + return iSize+1; +} diff --git a/contrib/linux-kernel/lib/error_private.c b/contrib/linux-kernel/lib/error_private.c new file mode 100644 index 000000000..44ae20104 --- /dev/null +++ b/contrib/linux-kernel/lib/error_private.c @@ -0,0 +1,44 @@ +/** + * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + +/* The purpose of this file is to have a single list of error strings embedded in binary */ + +#include "error_private.h" + +const char* ERR_getErrorString(ERR_enum code) +{ + static const char* const notErrorCode = "Unspecified error code"; + switch( code ) + { + case PREFIX(no_error): return "No error detected"; + case PREFIX(GENERIC): return "Error (generic)"; + case PREFIX(prefix_unknown): return "Unknown frame descriptor"; + case PREFIX(version_unsupported): return "Version not supported"; + case PREFIX(parameter_unknown): return "Unknown parameter type"; + case PREFIX(frameParameter_unsupported): return "Unsupported frame parameter"; + case PREFIX(frameParameter_unsupportedBy32bits): return "Frame parameter unsupported in 32-bits mode"; + case PREFIX(frameParameter_windowTooLarge): return "Frame requires too much memory for decoding"; + case PREFIX(compressionParameter_unsupported): return "Compression parameter is out of bound"; + case PREFIX(init_missing): return "Context should be init first"; + case PREFIX(memory_allocation): return "Allocation error : not enough memory"; + case PREFIX(stage_wrong): return "Operation not authorized at current processing stage"; + case PREFIX(dstSize_tooSmall): return "Destination buffer is too small"; + case PREFIX(srcSize_wrong): return "Src size incorrect"; + case PREFIX(corruption_detected): return "Corrupted block detected"; + case PREFIX(checksum_wrong): return "Restored data doesn't match checksum"; + case PREFIX(tableLog_tooLarge): return "tableLog requires too much memory : unsupported"; + case PREFIX(maxSymbolValue_tooLarge): return "Unsupported max Symbol Value : too large"; + case PREFIX(maxSymbolValue_tooSmall): return "Specified maxSymbolValue is too small"; + case PREFIX(dictionary_corrupted): return "Dictionary is corrupted"; + case PREFIX(dictionary_wrong): return "Dictionary mismatch"; + case PREFIX(dictionaryCreation_failed): return "Cannot create Dictionary from provided samples"; + case PREFIX(maxCode): + default: return notErrorCode; + } +} diff --git a/contrib/linux-kernel/lib/error_private.h b/contrib/linux-kernel/lib/error_private.h new file mode 100644 index 000000000..1bc2e4954 --- /dev/null +++ b/contrib/linux-kernel/lib/error_private.h @@ -0,0 +1,76 @@ +/** + * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + +/* Note : this module is expected to remain private, do not expose it */ + +#ifndef ERROR_H_MODULE +#define ERROR_H_MODULE + +#if defined (__cplusplus) +extern "C" { +#endif + + +/* **************************************** +* Dependencies +******************************************/ +#include /* size_t */ +#include "zstd_errors.h" /* enum list */ + + +/* **************************************** +* Compiler-specific +******************************************/ +#if defined(__GNUC__) +# define ERR_STATIC static __attribute__((unused)) +#elif defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) +# define ERR_STATIC static inline +#elif defined(_MSC_VER) +# define ERR_STATIC static __inline +#else +# define ERR_STATIC static /* this version may generate warnings for unused static functions; disable the relevant warning */ +#endif + + +/*-**************************************** +* Customization (error_public.h) +******************************************/ +typedef ZSTD_ErrorCode ERR_enum; +#define PREFIX(name) ZSTD_error_##name + + +/*-**************************************** +* Error codes handling +******************************************/ +#ifdef ERROR +# undef ERROR /* reported already defined on VS 2015 (Rich Geldreich) */ +#endif +#define ERROR(name) ((size_t)-PREFIX(name)) + +ERR_STATIC unsigned ERR_isError(size_t code) { return (code > ERROR(maxCode)); } + +ERR_STATIC ERR_enum ERR_getErrorCode(size_t code) { if (!ERR_isError(code)) return (ERR_enum)0; return (ERR_enum) (0-code); } + + +/*-**************************************** +* Error Strings +******************************************/ + +const char* ERR_getErrorString(ERR_enum code); /* error_private.c */ + +ERR_STATIC const char* ERR_getErrorName(size_t code) +{ + return ERR_getErrorString(ERR_getErrorCode(code)); +} + +#if defined (__cplusplus) +} +#endif + +#endif /* ERROR_H_MODULE */ diff --git a/contrib/linux-kernel/lib/fse.h b/contrib/linux-kernel/lib/fse.h new file mode 100644 index 000000000..baac39032 --- /dev/null +++ b/contrib/linux-kernel/lib/fse.h @@ -0,0 +1,694 @@ +/* ****************************************************************** + FSE : Finite State Entropy codec + Public Prototypes declaration + Copyright (C) 2013-2016, Yann Collet. + + BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php) + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are + met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the + distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + You can contact the author at : + - Source repository : https://github.com/Cyan4973/FiniteStateEntropy +****************************************************************** */ +#ifndef FSE_H +#define FSE_H + +#if defined (__cplusplus) +extern "C" { +#endif + + +/*-***************************************** +* Dependencies +******************************************/ +#include /* size_t, ptrdiff_t */ + + +/*-***************************************** +* FSE_PUBLIC_API : control library symbols visibility +******************************************/ +#if defined(FSE_DLL_EXPORT) && (FSE_DLL_EXPORT==1) && defined(__GNUC__) && (__GNUC__ >= 4) +# define FSE_PUBLIC_API __attribute__ ((visibility ("default"))) +#elif defined(FSE_DLL_EXPORT) && (FSE_DLL_EXPORT==1) /* Visual expected */ +# define FSE_PUBLIC_API __declspec(dllexport) +#elif defined(FSE_DLL_IMPORT) && (FSE_DLL_IMPORT==1) +# define FSE_PUBLIC_API __declspec(dllimport) /* It isn't required but allows to generate better code, saving a function pointer load from the IAT and an indirect jump.*/ +#else +# define FSE_PUBLIC_API +#endif + +/*------ Version ------*/ +#define FSE_VERSION_MAJOR 0 +#define FSE_VERSION_MINOR 9 +#define FSE_VERSION_RELEASE 0 + +#define FSE_LIB_VERSION FSE_VERSION_MAJOR.FSE_VERSION_MINOR.FSE_VERSION_RELEASE +#define FSE_QUOTE(str) #str +#define FSE_EXPAND_AND_QUOTE(str) FSE_QUOTE(str) +#define FSE_VERSION_STRING FSE_EXPAND_AND_QUOTE(FSE_LIB_VERSION) + +#define FSE_VERSION_NUMBER (FSE_VERSION_MAJOR *100*100 + FSE_VERSION_MINOR *100 + FSE_VERSION_RELEASE) +FSE_PUBLIC_API unsigned FSE_versionNumber(void); /**< library version number; to be used when checking dll version */ + +/*-**************************************** +* FSE simple functions +******************************************/ +/*! FSE_compress() : + Compress content of buffer 'src', of size 'srcSize', into destination buffer 'dst'. + 'dst' buffer must be already allocated. Compression runs faster is dstCapacity >= FSE_compressBound(srcSize). + @return : size of compressed data (<= dstCapacity). + Special values : if return == 0, srcData is not compressible => Nothing is stored within dst !!! + if return == 1, srcData is a single byte symbol * srcSize times. Use RLE compression instead. + if FSE_isError(return), compression failed (more details using FSE_getErrorName()) +*/ +FSE_PUBLIC_API size_t FSE_compress(void* dst, size_t dstCapacity, + const void* src, size_t srcSize); + +/*! FSE_decompress(): + Decompress FSE data from buffer 'cSrc', of size 'cSrcSize', + into already allocated destination buffer 'dst', of size 'dstCapacity'. + @return : size of regenerated data (<= maxDstSize), + or an error code, which can be tested using FSE_isError() . + + ** Important ** : FSE_decompress() does not decompress non-compressible nor RLE data !!! + Why ? : making this distinction requires a header. + Header management is intentionally delegated to the user layer, which can better manage special cases. +*/ +FSE_PUBLIC_API size_t FSE_decompress(void* dst, size_t dstCapacity, + const void* cSrc, size_t cSrcSize); + + +/*-***************************************** +* Tool functions +******************************************/ +FSE_PUBLIC_API size_t FSE_compressBound(size_t size); /* maximum compressed size */ + +/* Error Management */ +FSE_PUBLIC_API unsigned FSE_isError(size_t code); /* tells if a return value is an error code */ +FSE_PUBLIC_API const char* FSE_getErrorName(size_t code); /* provides error code string (useful for debugging) */ + + +/*-***************************************** +* FSE advanced functions +******************************************/ +/*! FSE_compress2() : + Same as FSE_compress(), but allows the selection of 'maxSymbolValue' and 'tableLog' + Both parameters can be defined as '0' to mean : use default value + @return : size of compressed data + Special values : if return == 0, srcData is not compressible => Nothing is stored within cSrc !!! + if return == 1, srcData is a single byte symbol * srcSize times. Use RLE compression. + if FSE_isError(return), it's an error code. +*/ +FSE_PUBLIC_API size_t FSE_compress2 (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog); + + +/*-***************************************** +* FSE detailed API +******************************************/ +/*! +FSE_compress() does the following: +1. count symbol occurrence from source[] into table count[] +2. normalize counters so that sum(count[]) == Power_of_2 (2^tableLog) +3. save normalized counters to memory buffer using writeNCount() +4. build encoding table 'CTable' from normalized counters +5. encode the data stream using encoding table 'CTable' + +FSE_decompress() does the following: +1. read normalized counters with readNCount() +2. build decoding table 'DTable' from normalized counters +3. decode the data stream using decoding table 'DTable' + +The following API allows targeting specific sub-functions for advanced tasks. +For example, it's possible to compress several blocks using the same 'CTable', +or to save and provide normalized distribution using external method. +*/ + +/* *** COMPRESSION *** */ + +/*! FSE_count(): + Provides the precise count of each byte within a table 'count'. + 'count' is a table of unsigned int, of minimum size (*maxSymbolValuePtr+1). + *maxSymbolValuePtr will be updated if detected smaller than initial value. + @return : the count of the most frequent symbol (which is not identified). + if return == srcSize, there is only one symbol. + Can also return an error code, which can be tested with FSE_isError(). */ +FSE_PUBLIC_API size_t FSE_count(unsigned* count, unsigned* maxSymbolValuePtr, const void* src, size_t srcSize); + +/*! FSE_optimalTableLog(): + dynamically downsize 'tableLog' when conditions are met. + It saves CPU time, by using smaller tables, while preserving or even improving compression ratio. + @return : recommended tableLog (necessarily <= 'maxTableLog') */ +FSE_PUBLIC_API unsigned FSE_optimalTableLog(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue); + +/*! FSE_normalizeCount(): + normalize counts so that sum(count[]) == Power_of_2 (2^tableLog) + 'normalizedCounter' is a table of short, of minimum size (maxSymbolValue+1). + @return : tableLog, + or an errorCode, which can be tested using FSE_isError() */ +FSE_PUBLIC_API size_t FSE_normalizeCount(short* normalizedCounter, unsigned tableLog, const unsigned* count, size_t srcSize, unsigned maxSymbolValue); + +/*! FSE_NCountWriteBound(): + Provides the maximum possible size of an FSE normalized table, given 'maxSymbolValue' and 'tableLog'. + Typically useful for allocation purpose. */ +FSE_PUBLIC_API size_t FSE_NCountWriteBound(unsigned maxSymbolValue, unsigned tableLog); + +/*! FSE_writeNCount(): + Compactly save 'normalizedCounter' into 'buffer'. + @return : size of the compressed table, + or an errorCode, which can be tested using FSE_isError(). */ +FSE_PUBLIC_API size_t FSE_writeNCount (void* buffer, size_t bufferSize, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog); + + +/*! Constructor and Destructor of FSE_CTable. + Note that FSE_CTable size depends on 'tableLog' and 'maxSymbolValue' */ +typedef unsigned FSE_CTable; /* don't allocate that. It's only meant to be more restrictive than void* */ +FSE_PUBLIC_API FSE_CTable* FSE_createCTable (unsigned tableLog, unsigned maxSymbolValue); +FSE_PUBLIC_API void FSE_freeCTable (FSE_CTable* ct); + +/*! FSE_buildCTable(): + Builds `ct`, which must be already allocated, using FSE_createCTable(). + @return : 0, or an errorCode, which can be tested using FSE_isError() */ +FSE_PUBLIC_API size_t FSE_buildCTable(FSE_CTable* ct, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog); + +/*! FSE_compress_usingCTable(): + Compress `src` using `ct` into `dst` which must be already allocated. + @return : size of compressed data (<= `dstCapacity`), + or 0 if compressed data could not fit into `dst`, + or an errorCode, which can be tested using FSE_isError() */ +FSE_PUBLIC_API size_t FSE_compress_usingCTable (void* dst, size_t dstCapacity, const void* src, size_t srcSize, const FSE_CTable* ct); + +/*! +Tutorial : +---------- +The first step is to count all symbols. FSE_count() does this job very fast. +Result will be saved into 'count', a table of unsigned int, which must be already allocated, and have 'maxSymbolValuePtr[0]+1' cells. +'src' is a table of bytes of size 'srcSize'. All values within 'src' MUST be <= maxSymbolValuePtr[0] +maxSymbolValuePtr[0] will be updated, with its real value (necessarily <= original value) +FSE_count() will return the number of occurrence of the most frequent symbol. +This can be used to know if there is a single symbol within 'src', and to quickly evaluate its compressibility. +If there is an error, the function will return an ErrorCode (which can be tested using FSE_isError()). + +The next step is to normalize the frequencies. +FSE_normalizeCount() will ensure that sum of frequencies is == 2 ^'tableLog'. +It also guarantees a minimum of 1 to any Symbol with frequency >= 1. +You can use 'tableLog'==0 to mean "use default tableLog value". +If you are unsure of which tableLog value to use, you can ask FSE_optimalTableLog(), +which will provide the optimal valid tableLog given sourceSize, maxSymbolValue, and a user-defined maximum (0 means "default"). + +The result of FSE_normalizeCount() will be saved into a table, +called 'normalizedCounter', which is a table of signed short. +'normalizedCounter' must be already allocated, and have at least 'maxSymbolValue+1' cells. +The return value is tableLog if everything proceeded as expected. +It is 0 if there is a single symbol within distribution. +If there is an error (ex: invalid tableLog value), the function will return an ErrorCode (which can be tested using FSE_isError()). + +'normalizedCounter' can be saved in a compact manner to a memory area using FSE_writeNCount(). +'buffer' must be already allocated. +For guaranteed success, buffer size must be at least FSE_headerBound(). +The result of the function is the number of bytes written into 'buffer'. +If there is an error, the function will return an ErrorCode (which can be tested using FSE_isError(); ex : buffer size too small). + +'normalizedCounter' can then be used to create the compression table 'CTable'. +The space required by 'CTable' must be already allocated, using FSE_createCTable(). +You can then use FSE_buildCTable() to fill 'CTable'. +If there is an error, both functions will return an ErrorCode (which can be tested using FSE_isError()). + +'CTable' can then be used to compress 'src', with FSE_compress_usingCTable(). +Similar to FSE_count(), the convention is that 'src' is assumed to be a table of char of size 'srcSize' +The function returns the size of compressed data (without header), necessarily <= `dstCapacity`. +If it returns '0', compressed data could not fit into 'dst'. +If there is an error, the function will return an ErrorCode (which can be tested using FSE_isError()). +*/ + + +/* *** DECOMPRESSION *** */ + +/*! FSE_readNCount(): + Read compactly saved 'normalizedCounter' from 'rBuffer'. + @return : size read from 'rBuffer', + or an errorCode, which can be tested using FSE_isError(). + maxSymbolValuePtr[0] and tableLogPtr[0] will also be updated with their respective values */ +FSE_PUBLIC_API size_t FSE_readNCount (short* normalizedCounter, unsigned* maxSymbolValuePtr, unsigned* tableLogPtr, const void* rBuffer, size_t rBuffSize); + +/*! Constructor and Destructor of FSE_DTable. + Note that its size depends on 'tableLog' */ +typedef unsigned FSE_DTable; /* don't allocate that. It's just a way to be more restrictive than void* */ +FSE_PUBLIC_API FSE_DTable* FSE_createDTable(unsigned tableLog); +FSE_PUBLIC_API void FSE_freeDTable(FSE_DTable* dt); + +/*! FSE_buildDTable(): + Builds 'dt', which must be already allocated, using FSE_createDTable(). + return : 0, or an errorCode, which can be tested using FSE_isError() */ +FSE_PUBLIC_API size_t FSE_buildDTable (FSE_DTable* dt, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog); + +/*! FSE_decompress_usingDTable(): + Decompress compressed source `cSrc` of size `cSrcSize` using `dt` + into `dst` which must be already allocated. + @return : size of regenerated data (necessarily <= `dstCapacity`), + or an errorCode, which can be tested using FSE_isError() */ +FSE_PUBLIC_API size_t FSE_decompress_usingDTable(void* dst, size_t dstCapacity, const void* cSrc, size_t cSrcSize, const FSE_DTable* dt); + +/*! +Tutorial : +---------- +(Note : these functions only decompress FSE-compressed blocks. + If block is uncompressed, use memcpy() instead + If block is a single repeated byte, use memset() instead ) + +The first step is to obtain the normalized frequencies of symbols. +This can be performed by FSE_readNCount() if it was saved using FSE_writeNCount(). +'normalizedCounter' must be already allocated, and have at least 'maxSymbolValuePtr[0]+1' cells of signed short. +In practice, that means it's necessary to know 'maxSymbolValue' beforehand, +or size the table to handle worst case situations (typically 256). +FSE_readNCount() will provide 'tableLog' and 'maxSymbolValue'. +The result of FSE_readNCount() is the number of bytes read from 'rBuffer'. +Note that 'rBufferSize' must be at least 4 bytes, even if useful information is less than that. +If there is an error, the function will return an error code, which can be tested using FSE_isError(). + +The next step is to build the decompression tables 'FSE_DTable' from 'normalizedCounter'. +This is performed by the function FSE_buildDTable(). +The space required by 'FSE_DTable' must be already allocated using FSE_createDTable(). +If there is an error, the function will return an error code, which can be tested using FSE_isError(). + +`FSE_DTable` can then be used to decompress `cSrc`, with FSE_decompress_usingDTable(). +`cSrcSize` must be strictly correct, otherwise decompression will fail. +FSE_decompress_usingDTable() result will tell how many bytes were regenerated (<=`dstCapacity`). +If there is an error, the function will return an error code, which can be tested using FSE_isError(). (ex: dst buffer too small) +*/ + + +#ifdef FSE_STATIC_LINKING_ONLY + +/* *** Dependency *** */ +#include "bitstream.h" + + +/* ***************************************** +* Static allocation +*******************************************/ +/* FSE buffer bounds */ +#define FSE_NCOUNTBOUND 512 +#define FSE_BLOCKBOUND(size) (size + (size>>7)) +#define FSE_COMPRESSBOUND(size) (FSE_NCOUNTBOUND + FSE_BLOCKBOUND(size)) /* Macro version, useful for static allocation */ + +/* It is possible to statically allocate FSE CTable/DTable as a table of FSE_CTable/FSE_DTable using below macros */ +#define FSE_CTABLE_SIZE_U32(maxTableLog, maxSymbolValue) (1 + (1<<(maxTableLog-1)) + ((maxSymbolValue+1)*2)) +#define FSE_DTABLE_SIZE_U32(maxTableLog) (1 + (1<= `1024` unsigned + */ +size_t FSE_count_wksp(unsigned* count, unsigned* maxSymbolValuePtr, + const void* source, size_t sourceSize, unsigned* workSpace); + +/** FSE_countFast() : + * same as FSE_count(), but blindly trusts that all byte values within src are <= *maxSymbolValuePtr + */ +size_t FSE_countFast(unsigned* count, unsigned* maxSymbolValuePtr, const void* src, size_t srcSize); + +/* FSE_countFast_wksp() : + * Same as FSE_countFast(), but using an externally provided scratch buffer. + * `workSpace` must be a table of minimum `1024` unsigned + */ +size_t FSE_countFast_wksp(unsigned* count, unsigned* maxSymbolValuePtr, const void* src, size_t srcSize, unsigned* workSpace); + +/*! FSE_count_simple + * Same as FSE_countFast(), but does not use any additional memory (not even on stack). + * This function is unsafe, and will segfault if any value within `src` is `> *maxSymbolValuePtr` (presuming it's also the size of `count`). +*/ +size_t FSE_count_simple(unsigned* count, unsigned* maxSymbolValuePtr, const void* src, size_t srcSize); + + + +unsigned FSE_optimalTableLog_internal(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue, unsigned minus); +/**< same as FSE_optimalTableLog(), which used `minus==2` */ + +/* FSE_compress_wksp() : + * Same as FSE_compress2(), but using an externally allocated scratch buffer (`workSpace`). + * FSE_WKSP_SIZE_U32() provides the minimum size required for `workSpace` as a table of FSE_CTable. + */ +#define FSE_WKSP_SIZE_U32(maxTableLog, maxSymbolValue) ( FSE_CTABLE_SIZE_U32(maxTableLog, maxSymbolValue) + (1<<((maxTableLog>2)?(maxTableLog-2):0)) ) +size_t FSE_compress_wksp (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize); + +size_t FSE_buildCTable_raw (FSE_CTable* ct, unsigned nbBits); +/**< build a fake FSE_CTable, designed for a flat distribution, where each symbol uses nbBits */ + +size_t FSE_buildCTable_rle (FSE_CTable* ct, unsigned char symbolValue); +/**< build a fake FSE_CTable, designed to compress always the same symbolValue */ + +/* FSE_buildCTable_wksp() : + * Same as FSE_buildCTable(), but using an externally allocated scratch buffer (`workSpace`). + * `wkspSize` must be >= `(1<= BIT_DStream_completed + +When it's done, verify decompression is fully completed, by checking both DStream and the relevant states. +Checking if DStream has reached its end is performed by : + BIT_endOfDStream(&DStream); +Check also the states. There might be some symbols left there, if some high probability ones (>50%) are possible. + FSE_endOfDState(&DState); +*/ + + +/* ***************************************** +* FSE unsafe API +*******************************************/ +static unsigned char FSE_decodeSymbolFast(FSE_DState_t* DStatePtr, BIT_DStream_t* bitD); +/* faster, but works only if nbBits is always >= 1 (otherwise, result will be corrupted) */ + + +/* ***************************************** +* Implementation of inlined functions +*******************************************/ +typedef struct { + int deltaFindState; + U32 deltaNbBits; +} FSE_symbolCompressionTransform; /* total 8 bytes */ + +MEM_STATIC void FSE_initCState(FSE_CState_t* statePtr, const FSE_CTable* ct) +{ + const void* ptr = ct; + const U16* u16ptr = (const U16*) ptr; + const U32 tableLog = MEM_read16(ptr); + statePtr->value = (ptrdiff_t)1<stateTable = u16ptr+2; + statePtr->symbolTT = ((const U32*)ct + 1 + (tableLog ? (1<<(tableLog-1)) : 1)); + statePtr->stateLog = tableLog; +} + + +/*! FSE_initCState2() : +* Same as FSE_initCState(), but the first symbol to include (which will be the last to be read) +* uses the smallest state value possible, saving the cost of this symbol */ +MEM_STATIC void FSE_initCState2(FSE_CState_t* statePtr, const FSE_CTable* ct, U32 symbol) +{ + FSE_initCState(statePtr, ct); + { const FSE_symbolCompressionTransform symbolTT = ((const FSE_symbolCompressionTransform*)(statePtr->symbolTT))[symbol]; + const U16* stateTable = (const U16*)(statePtr->stateTable); + U32 nbBitsOut = (U32)((symbolTT.deltaNbBits + (1<<15)) >> 16); + statePtr->value = (nbBitsOut << 16) - symbolTT.deltaNbBits; + statePtr->value = stateTable[(statePtr->value >> nbBitsOut) + symbolTT.deltaFindState]; + } +} + +MEM_STATIC void FSE_encodeSymbol(BIT_CStream_t* bitC, FSE_CState_t* statePtr, U32 symbol) +{ + const FSE_symbolCompressionTransform symbolTT = ((const FSE_symbolCompressionTransform*)(statePtr->symbolTT))[symbol]; + const U16* const stateTable = (const U16*)(statePtr->stateTable); + U32 nbBitsOut = (U32)((statePtr->value + symbolTT.deltaNbBits) >> 16); + BIT_addBits(bitC, statePtr->value, nbBitsOut); + statePtr->value = stateTable[ (statePtr->value >> nbBitsOut) + symbolTT.deltaFindState]; +} + +MEM_STATIC void FSE_flushCState(BIT_CStream_t* bitC, const FSE_CState_t* statePtr) +{ + BIT_addBits(bitC, statePtr->value, statePtr->stateLog); + BIT_flushBits(bitC); +} + + +/* ====== Decompression ====== */ + +typedef struct { + U16 tableLog; + U16 fastMode; +} FSE_DTableHeader; /* sizeof U32 */ + +typedef struct +{ + unsigned short newState; + unsigned char symbol; + unsigned char nbBits; +} FSE_decode_t; /* size == U32 */ + +MEM_STATIC void FSE_initDState(FSE_DState_t* DStatePtr, BIT_DStream_t* bitD, const FSE_DTable* dt) +{ + const void* ptr = dt; + const FSE_DTableHeader* const DTableH = (const FSE_DTableHeader*)ptr; + DStatePtr->state = BIT_readBits(bitD, DTableH->tableLog); + BIT_reloadDStream(bitD); + DStatePtr->table = dt + 1; +} + +MEM_STATIC BYTE FSE_peekSymbol(const FSE_DState_t* DStatePtr) +{ + FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; + return DInfo.symbol; +} + +MEM_STATIC void FSE_updateState(FSE_DState_t* DStatePtr, BIT_DStream_t* bitD) +{ + FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; + U32 const nbBits = DInfo.nbBits; + size_t const lowBits = BIT_readBits(bitD, nbBits); + DStatePtr->state = DInfo.newState + lowBits; +} + +MEM_STATIC BYTE FSE_decodeSymbol(FSE_DState_t* DStatePtr, BIT_DStream_t* bitD) +{ + FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; + U32 const nbBits = DInfo.nbBits; + BYTE const symbol = DInfo.symbol; + size_t const lowBits = BIT_readBits(bitD, nbBits); + + DStatePtr->state = DInfo.newState + lowBits; + return symbol; +} + +/*! FSE_decodeSymbolFast() : + unsafe, only works if no symbol has a probability > 50% */ +MEM_STATIC BYTE FSE_decodeSymbolFast(FSE_DState_t* DStatePtr, BIT_DStream_t* bitD) +{ + FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; + U32 const nbBits = DInfo.nbBits; + BYTE const symbol = DInfo.symbol; + size_t const lowBits = BIT_readBitsFast(bitD, nbBits); + + DStatePtr->state = DInfo.newState + lowBits; + return symbol; +} + +MEM_STATIC unsigned FSE_endOfDState(const FSE_DState_t* DStatePtr) +{ + return DStatePtr->state == 0; +} + + + +#ifndef FSE_COMMONDEFS_ONLY + +/* ************************************************************** +* Tuning parameters +****************************************************************/ +/*!MEMORY_USAGE : +* Memory usage formula : N->2^N Bytes (examples : 10 -> 1KB; 12 -> 4KB ; 16 -> 64KB; 20 -> 1MB; etc.) +* Increasing memory usage improves compression ratio +* Reduced memory usage can improve speed, due to cache effect +* Recommended max value is 14, for 16KB, which nicely fits into Intel x86 L1 cache */ +#ifndef FSE_MAX_MEMORY_USAGE +# define FSE_MAX_MEMORY_USAGE 14 +#endif +#ifndef FSE_DEFAULT_MEMORY_USAGE +# define FSE_DEFAULT_MEMORY_USAGE 13 +#endif + +/*!FSE_MAX_SYMBOL_VALUE : +* Maximum symbol value authorized. +* Required for proper stack allocation */ +#ifndef FSE_MAX_SYMBOL_VALUE +# define FSE_MAX_SYMBOL_VALUE 255 +#endif + +/* ************************************************************** +* template functions type & suffix +****************************************************************/ +#define FSE_FUNCTION_TYPE BYTE +#define FSE_FUNCTION_EXTENSION +#define FSE_DECODE_TYPE FSE_decode_t + + +#endif /* !FSE_COMMONDEFS_ONLY */ + + +/* *************************************************************** +* Constants +*****************************************************************/ +#define FSE_MAX_TABLELOG (FSE_MAX_MEMORY_USAGE-2) +#define FSE_MAX_TABLESIZE (1U< FSE_TABLELOG_ABSOLUTE_MAX +# error "FSE_MAX_TABLELOG > FSE_TABLELOG_ABSOLUTE_MAX is not supported" +#endif + +#define FSE_TABLESTEP(tableSize) ((tableSize>>1) + (tableSize>>3) + 3) + + +#endif /* FSE_STATIC_LINKING_ONLY */ + + +#if defined (__cplusplus) +} +#endif + +#endif /* FSE_H */ diff --git a/contrib/linux-kernel/lib/fse_compress.c b/contrib/linux-kernel/lib/fse_compress.c new file mode 100644 index 000000000..13654d6e6 --- /dev/null +++ b/contrib/linux-kernel/lib/fse_compress.c @@ -0,0 +1,857 @@ +/* ****************************************************************** + FSE : Finite State Entropy encoder + Copyright (C) 2013-2015, Yann Collet. + + BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php) + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are + met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the + distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + You can contact the author at : + - FSE source repository : https://github.com/Cyan4973/FiniteStateEntropy + - Public forum : https://groups.google.com/forum/#!forum/lz4c +****************************************************************** */ + +/* ************************************************************** +* Compiler specifics +****************************************************************/ +#ifdef _MSC_VER /* Visual Studio */ +# define FORCE_INLINE static __forceinline +# include /* For Visual 2005 */ +# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ +# pragma warning(disable : 4214) /* disable: C4214: non-int bitfields */ +#else +# if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ +# ifdef __GNUC__ +# define FORCE_INLINE static inline __attribute__((always_inline)) +# else +# define FORCE_INLINE static inline +# endif +# else +# define FORCE_INLINE static +# endif /* __STDC_VERSION__ */ +#endif + + +/* ************************************************************** +* Includes +****************************************************************/ +#include /* malloc, free, qsort */ +#include /* memcpy, memset */ +#include /* printf (debug) */ +#include "bitstream.h" +#define FSE_STATIC_LINKING_ONLY +#include "fse.h" + + +/* ************************************************************** +* Error Management +****************************************************************/ +#define FSE_STATIC_ASSERT(c) { enum { FSE_static_assert = 1/(int)(!!(c)) }; } /* use only *after* variable declarations */ + + +/* ************************************************************** +* Templates +****************************************************************/ +/* + designed to be included + for type-specific functions (template emulation in C) + Objective is to write these functions only once, for improved maintenance +*/ + +/* safety checks */ +#ifndef FSE_FUNCTION_EXTENSION +# error "FSE_FUNCTION_EXTENSION must be defined" +#endif +#ifndef FSE_FUNCTION_TYPE +# error "FSE_FUNCTION_TYPE must be defined" +#endif + +/* Function names */ +#define FSE_CAT(X,Y) X##Y +#define FSE_FUNCTION_NAME(X,Y) FSE_CAT(X,Y) +#define FSE_TYPE_NAME(X,Y) FSE_CAT(X,Y) + + +/* Function templates */ + +/* FSE_buildCTable_wksp() : + * Same as FSE_buildCTable(), but using an externally allocated scratch buffer (`workSpace`). + * wkspSize should be sized to handle worst case situation, which is `1<>1 : 1) ; + FSE_symbolCompressionTransform* const symbolTT = (FSE_symbolCompressionTransform*) (FSCT); + U32 const step = FSE_TABLESTEP(tableSize); + U32 cumul[FSE_MAX_SYMBOL_VALUE+2]; + + FSE_FUNCTION_TYPE* const tableSymbol = (FSE_FUNCTION_TYPE*)workSpace; + U32 highThreshold = tableSize-1; + + /* CTable header */ + if (((size_t)1 << tableLog) * sizeof(FSE_FUNCTION_TYPE) > wkspSize) return ERROR(tableLog_tooLarge); + tableU16[-2] = (U16) tableLog; + tableU16[-1] = (U16) maxSymbolValue; + + /* For explanations on how to distribute symbol values over the table : + * http://fastcompression.blogspot.fr/2014/02/fse-distributing-symbol-values.html */ + + /* symbol start positions */ + { U32 u; + cumul[0] = 0; + for (u=1; u<=maxSymbolValue+1; u++) { + if (normalizedCounter[u-1]==-1) { /* Low proba symbol */ + cumul[u] = cumul[u-1] + 1; + tableSymbol[highThreshold--] = (FSE_FUNCTION_TYPE)(u-1); + } else { + cumul[u] = cumul[u-1] + normalizedCounter[u-1]; + } } + cumul[maxSymbolValue+1] = tableSize+1; + } + + /* Spread symbols */ + { U32 position = 0; + U32 symbol; + for (symbol=0; symbol<=maxSymbolValue; symbol++) { + int nbOccurences; + for (nbOccurences=0; nbOccurences highThreshold) position = (position + step) & tableMask; /* Low proba area */ + } } + + if (position!=0) return ERROR(GENERIC); /* Must have gone through all positions */ + } + + /* Build table */ + { U32 u; for (u=0; u> 3) + 3; + return maxSymbolValue ? maxHeaderSize : FSE_NCOUNTBOUND; /* maxSymbolValue==0 ? use default */ +} + +static size_t FSE_writeNCount_generic (void* header, size_t headerBufferSize, + const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog, + unsigned writeIsSafe) +{ + BYTE* const ostart = (BYTE*) header; + BYTE* out = ostart; + BYTE* const oend = ostart + headerBufferSize; + int nbBits; + const int tableSize = 1 << tableLog; + int remaining; + int threshold; + U32 bitStream; + int bitCount; + unsigned charnum = 0; + int previous0 = 0; + + bitStream = 0; + bitCount = 0; + /* Table Size */ + bitStream += (tableLog-FSE_MIN_TABLELOG) << bitCount; + bitCount += 4; + + /* Init */ + remaining = tableSize+1; /* +1 for extra accuracy */ + threshold = tableSize; + nbBits = tableLog+1; + + while (remaining>1) { /* stops at 1 */ + if (previous0) { + unsigned start = charnum; + while (!normalizedCounter[charnum]) charnum++; + while (charnum >= start+24) { + start+=24; + bitStream += 0xFFFFU << bitCount; + if ((!writeIsSafe) && (out > oend-2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ + out[0] = (BYTE) bitStream; + out[1] = (BYTE)(bitStream>>8); + out+=2; + bitStream>>=16; + } + while (charnum >= start+3) { + start+=3; + bitStream += 3 << bitCount; + bitCount += 2; + } + bitStream += (charnum-start) << bitCount; + bitCount += 2; + if (bitCount>16) { + if ((!writeIsSafe) && (out > oend - 2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ + out[0] = (BYTE)bitStream; + out[1] = (BYTE)(bitStream>>8); + out += 2; + bitStream >>= 16; + bitCount -= 16; + } } + { int count = normalizedCounter[charnum++]; + int const max = (2*threshold-1)-remaining; + remaining -= count < 0 ? -count : count; + count++; /* +1 for extra accuracy */ + if (count>=threshold) count += max; /* [0..max[ [max..threshold[ (...) [threshold+max 2*threshold[ */ + bitStream += count << bitCount; + bitCount += nbBits; + bitCount -= (count>=1; + } + if (bitCount>16) { + if ((!writeIsSafe) && (out > oend - 2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ + out[0] = (BYTE)bitStream; + out[1] = (BYTE)(bitStream>>8); + out += 2; + bitStream >>= 16; + bitCount -= 16; + } } + + /* flush remaining bitStream */ + if ((!writeIsSafe) && (out > oend - 2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ + out[0] = (BYTE)bitStream; + out[1] = (BYTE)(bitStream>>8); + out+= (bitCount+7) /8; + + if (charnum > maxSymbolValue + 1) return ERROR(GENERIC); + + return (out-ostart); +} + + +size_t FSE_writeNCount (void* buffer, size_t bufferSize, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog) +{ + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); /* Unsupported */ + if (tableLog < FSE_MIN_TABLELOG) return ERROR(GENERIC); /* Unsupported */ + + if (bufferSize < FSE_NCountWriteBound(maxSymbolValue, tableLog)) + return FSE_writeNCount_generic(buffer, bufferSize, normalizedCounter, maxSymbolValue, tableLog, 0); + + return FSE_writeNCount_generic(buffer, bufferSize, normalizedCounter, maxSymbolValue, tableLog, 1); +} + + + +/*-************************************************************** +* Counting histogram +****************************************************************/ +/*! FSE_count_simple + This function counts byte values within `src`, and store the histogram into table `count`. + It doesn't use any additional memory. + But this function is unsafe : it doesn't check that all values within `src` can fit into `count`. + For this reason, prefer using a table `count` with 256 elements. + @return : count of most numerous element +*/ +size_t FSE_count_simple(unsigned* count, unsigned* maxSymbolValuePtr, + const void* src, size_t srcSize) +{ + const BYTE* ip = (const BYTE*)src; + const BYTE* const end = ip + srcSize; + unsigned maxSymbolValue = *maxSymbolValuePtr; + unsigned max=0; + + memset(count, 0, (maxSymbolValue+1)*sizeof(*count)); + if (srcSize==0) { *maxSymbolValuePtr = 0; return 0; } + + while (ip max) max = count[s]; } + + return (size_t)max; +} + + +/* FSE_count_parallel_wksp() : + * Same as FSE_count_parallel(), but using an externally provided scratch buffer. + * `workSpace` size must be a minimum of `1024 * sizeof(unsigned)`` */ +static size_t FSE_count_parallel_wksp( + unsigned* count, unsigned* maxSymbolValuePtr, + const void* source, size_t sourceSize, + unsigned checkMax, unsigned* const workSpace) +{ + const BYTE* ip = (const BYTE*)source; + const BYTE* const iend = ip+sourceSize; + unsigned maxSymbolValue = *maxSymbolValuePtr; + unsigned max=0; + U32* const Counting1 = workSpace; + U32* const Counting2 = Counting1 + 256; + U32* const Counting3 = Counting2 + 256; + U32* const Counting4 = Counting3 + 256; + + memset(Counting1, 0, 4*256*sizeof(unsigned)); + + /* safety checks */ + if (!sourceSize) { + memset(count, 0, maxSymbolValue + 1); + *maxSymbolValuePtr = 0; + return 0; + } + if (!maxSymbolValue) maxSymbolValue = 255; /* 0 == default */ + + /* by stripes of 16 bytes */ + { U32 cached = MEM_read32(ip); ip += 4; + while (ip < iend-15) { + U32 c = cached; cached = MEM_read32(ip); ip += 4; + Counting1[(BYTE) c ]++; + Counting2[(BYTE)(c>>8) ]++; + Counting3[(BYTE)(c>>16)]++; + Counting4[ c>>24 ]++; + c = cached; cached = MEM_read32(ip); ip += 4; + Counting1[(BYTE) c ]++; + Counting2[(BYTE)(c>>8) ]++; + Counting3[(BYTE)(c>>16)]++; + Counting4[ c>>24 ]++; + c = cached; cached = MEM_read32(ip); ip += 4; + Counting1[(BYTE) c ]++; + Counting2[(BYTE)(c>>8) ]++; + Counting3[(BYTE)(c>>16)]++; + Counting4[ c>>24 ]++; + c = cached; cached = MEM_read32(ip); ip += 4; + Counting1[(BYTE) c ]++; + Counting2[(BYTE)(c>>8) ]++; + Counting3[(BYTE)(c>>16)]++; + Counting4[ c>>24 ]++; + } + ip-=4; + } + + /* finish last symbols */ + while (ipmaxSymbolValue; s--) { + Counting1[s] += Counting2[s] + Counting3[s] + Counting4[s]; + if (Counting1[s]) return ERROR(maxSymbolValue_tooSmall); + } } + + { U32 s; for (s=0; s<=maxSymbolValue; s++) { + count[s] = Counting1[s] + Counting2[s] + Counting3[s] + Counting4[s]; + if (count[s] > max) max = count[s]; + } } + + while (!count[maxSymbolValue]) maxSymbolValue--; + *maxSymbolValuePtr = maxSymbolValue; + return (size_t)max; +} + +/* FSE_countFast_wksp() : + * Same as FSE_countFast(), but using an externally provided scratch buffer. + * `workSpace` size must be table of >= `1024` unsigned */ +size_t FSE_countFast_wksp(unsigned* count, unsigned* maxSymbolValuePtr, + const void* source, size_t sourceSize, unsigned* workSpace) +{ + if (sourceSize < 1500) return FSE_count_simple(count, maxSymbolValuePtr, source, sourceSize); + return FSE_count_parallel_wksp(count, maxSymbolValuePtr, source, sourceSize, 0, workSpace); +} + +/* fast variant (unsafe : won't check if src contains values beyond count[] limit) */ +size_t FSE_countFast(unsigned* count, unsigned* maxSymbolValuePtr, + const void* source, size_t sourceSize) +{ + unsigned tmpCounters[1024]; + return FSE_countFast_wksp(count, maxSymbolValuePtr, source, sourceSize, tmpCounters); +} + +/* FSE_count_wksp() : + * Same as FSE_count(), but using an externally provided scratch buffer. + * `workSpace` size must be table of >= `1024` unsigned */ +size_t FSE_count_wksp(unsigned* count, unsigned* maxSymbolValuePtr, + const void* source, size_t sourceSize, unsigned* workSpace) +{ + if (*maxSymbolValuePtr < 255) + return FSE_count_parallel_wksp(count, maxSymbolValuePtr, source, sourceSize, 1, workSpace); + *maxSymbolValuePtr = 255; + return FSE_countFast_wksp(count, maxSymbolValuePtr, source, sourceSize, workSpace); +} + +size_t FSE_count(unsigned* count, unsigned* maxSymbolValuePtr, + const void* src, size_t srcSize) +{ + unsigned tmpCounters[1024]; + return FSE_count_wksp(count, maxSymbolValuePtr, src, srcSize, tmpCounters); +} + + + +/*-************************************************************** +* FSE Compression Code +****************************************************************/ +/*! FSE_sizeof_CTable() : + FSE_CTable is a variable size structure which contains : + `U16 tableLog;` + `U16 maxSymbolValue;` + `U16 nextStateNumber[1 << tableLog];` // This size is variable + `FSE_symbolCompressionTransform symbolTT[maxSymbolValue+1];` // This size is variable +Allocation is manual (C standard does not support variable-size structures). +*/ +size_t FSE_sizeof_CTable (unsigned maxSymbolValue, unsigned tableLog) +{ + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); + return FSE_CTABLE_SIZE_U32 (tableLog, maxSymbolValue) * sizeof(U32); +} + +FSE_CTable* FSE_createCTable (unsigned maxSymbolValue, unsigned tableLog) +{ + size_t size; + if (tableLog > FSE_TABLELOG_ABSOLUTE_MAX) tableLog = FSE_TABLELOG_ABSOLUTE_MAX; + size = FSE_CTABLE_SIZE_U32 (tableLog, maxSymbolValue) * sizeof(U32); + return (FSE_CTable*)malloc(size); +} + +void FSE_freeCTable (FSE_CTable* ct) { free(ct); } + +/* provides the minimum logSize to safely represent a distribution */ +static unsigned FSE_minTableLog(size_t srcSize, unsigned maxSymbolValue) +{ + U32 minBitsSrc = BIT_highbit32((U32)(srcSize - 1)) + 1; + U32 minBitsSymbols = BIT_highbit32(maxSymbolValue) + 2; + U32 minBits = minBitsSrc < minBitsSymbols ? minBitsSrc : minBitsSymbols; + return minBits; +} + +unsigned FSE_optimalTableLog_internal(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue, unsigned minus) +{ + U32 maxBitsSrc = BIT_highbit32((U32)(srcSize - 1)) - minus; + U32 tableLog = maxTableLog; + U32 minBits = FSE_minTableLog(srcSize, maxSymbolValue); + if (tableLog==0) tableLog = FSE_DEFAULT_TABLELOG; + if (maxBitsSrc < tableLog) tableLog = maxBitsSrc; /* Accuracy can be reduced */ + if (minBits > tableLog) tableLog = minBits; /* Need a minimum to safely represent all symbol values */ + if (tableLog < FSE_MIN_TABLELOG) tableLog = FSE_MIN_TABLELOG; + if (tableLog > FSE_MAX_TABLELOG) tableLog = FSE_MAX_TABLELOG; + return tableLog; +} + +unsigned FSE_optimalTableLog(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue) +{ + return FSE_optimalTableLog_internal(maxTableLog, srcSize, maxSymbolValue, 2); +} + + +/* Secondary normalization method. + To be used when primary method fails. */ + +static size_t FSE_normalizeM2(short* norm, U32 tableLog, const unsigned* count, size_t total, U32 maxSymbolValue) +{ + short const NOT_YET_ASSIGNED = -2; + U32 s; + U32 distributed = 0; + U32 ToDistribute; + + /* Init */ + U32 const lowThreshold = (U32)(total >> tableLog); + U32 lowOne = (U32)((total * 3) >> (tableLog + 1)); + + for (s=0; s<=maxSymbolValue; s++) { + if (count[s] == 0) { + norm[s]=0; + continue; + } + if (count[s] <= lowThreshold) { + norm[s] = -1; + distributed++; + total -= count[s]; + continue; + } + if (count[s] <= lowOne) { + norm[s] = 1; + distributed++; + total -= count[s]; + continue; + } + + norm[s]=NOT_YET_ASSIGNED; + } + ToDistribute = (1 << tableLog) - distributed; + + if ((total / ToDistribute) > lowOne) { + /* risk of rounding to zero */ + lowOne = (U32)((total * 3) / (ToDistribute * 2)); + for (s=0; s<=maxSymbolValue; s++) { + if ((norm[s] == NOT_YET_ASSIGNED) && (count[s] <= lowOne)) { + norm[s] = 1; + distributed++; + total -= count[s]; + continue; + } } + ToDistribute = (1 << tableLog) - distributed; + } + + if (distributed == maxSymbolValue+1) { + /* all values are pretty poor; + probably incompressible data (should have already been detected); + find max, then give all remaining points to max */ + U32 maxV = 0, maxC = 0; + for (s=0; s<=maxSymbolValue; s++) + if (count[s] > maxC) maxV=s, maxC=count[s]; + norm[maxV] += (short)ToDistribute; + return 0; + } + + if (total == 0) { + /* all of the symbols were low enough for the lowOne or lowThreshold */ + for (s=0; ToDistribute > 0; s = (s+1)%(maxSymbolValue+1)) + if (norm[s] > 0) ToDistribute--, norm[s]++; + return 0; + } + + { U64 const vStepLog = 62 - tableLog; + U64 const mid = (1ULL << (vStepLog-1)) - 1; + U64 const rStep = ((((U64)1<> vStepLog); + U32 const sEnd = (U32)(end >> vStepLog); + U32 const weight = sEnd - sStart; + if (weight < 1) + return ERROR(GENERIC); + norm[s] = (short)weight; + tmpTotal = end; + } } } + + return 0; +} + + +size_t FSE_normalizeCount (short* normalizedCounter, unsigned tableLog, + const unsigned* count, size_t total, + unsigned maxSymbolValue) +{ + /* Sanity checks */ + if (tableLog==0) tableLog = FSE_DEFAULT_TABLELOG; + if (tableLog < FSE_MIN_TABLELOG) return ERROR(GENERIC); /* Unsupported size */ + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); /* Unsupported size */ + if (tableLog < FSE_minTableLog(total, maxSymbolValue)) return ERROR(GENERIC); /* Too small tableLog, compression potentially impossible */ + + { U32 const rtbTable[] = { 0, 473195, 504333, 520860, 550000, 700000, 750000, 830000 }; + U64 const scale = 62 - tableLog; + U64 const step = ((U64)1<<62) / total; /* <== here, one division ! */ + U64 const vStep = 1ULL<<(scale-20); + int stillToDistribute = 1<> tableLog); + + for (s=0; s<=maxSymbolValue; s++) { + if (count[s] == total) return 0; /* rle special case */ + if (count[s] == 0) { normalizedCounter[s]=0; continue; } + if (count[s] <= lowThreshold) { + normalizedCounter[s] = -1; + stillToDistribute--; + } else { + short proba = (short)((count[s]*step) >> scale); + if (proba<8) { + U64 restToBeat = vStep * rtbTable[proba]; + proba += (count[s]*step) - ((U64)proba< restToBeat; + } + if (proba > largestP) largestP=proba, largest=s; + normalizedCounter[s] = proba; + stillToDistribute -= proba; + } } + if (-stillToDistribute >= (normalizedCounter[largest] >> 1)) { + /* corner case, need another normalization method */ + size_t const errorCode = FSE_normalizeM2(normalizedCounter, tableLog, count, total, maxSymbolValue); + if (FSE_isError(errorCode)) return errorCode; + } + else normalizedCounter[largest] += (short)stillToDistribute; + } + +#if 0 + { /* Print Table (debug) */ + U32 s; + U32 nTotal = 0; + for (s=0; s<=maxSymbolValue; s++) + printf("%3i: %4i \n", s, normalizedCounter[s]); + for (s=0; s<=maxSymbolValue; s++) + nTotal += abs(normalizedCounter[s]); + if (nTotal != (1U<>1); /* assumption : tableLog >= 1 */ + FSE_symbolCompressionTransform* const symbolTT = (FSE_symbolCompressionTransform*) (FSCT); + unsigned s; + + /* Sanity checks */ + if (nbBits < 1) return ERROR(GENERIC); /* min size */ + + /* header */ + tableU16[-2] = (U16) nbBits; + tableU16[-1] = (U16) maxSymbolValue; + + /* Build table */ + for (s=0; s FSE_MAX_TABLELOG*4+7 ) && (srcSize & 2)) { /* test bit 2 */ + FSE_encodeSymbol(&bitC, &CState2, *--ip); + FSE_encodeSymbol(&bitC, &CState1, *--ip); + FSE_FLUSHBITS(&bitC); + } + + /* 2 or 4 encoding per loop */ + while ( ip>istart ) { + + FSE_encodeSymbol(&bitC, &CState2, *--ip); + + if (sizeof(bitC.bitContainer)*8 < FSE_MAX_TABLELOG*2+7 ) /* this test must be static */ + FSE_FLUSHBITS(&bitC); + + FSE_encodeSymbol(&bitC, &CState1, *--ip); + + if (sizeof(bitC.bitContainer)*8 > FSE_MAX_TABLELOG*4+7 ) { /* this test must be static */ + FSE_encodeSymbol(&bitC, &CState2, *--ip); + FSE_encodeSymbol(&bitC, &CState1, *--ip); + } + + FSE_FLUSHBITS(&bitC); + } + + FSE_flushCState(&bitC, &CState2); + FSE_flushCState(&bitC, &CState1); + return BIT_closeCStream(&bitC); +} + +size_t FSE_compress_usingCTable (void* dst, size_t dstSize, + const void* src, size_t srcSize, + const FSE_CTable* ct) +{ + unsigned const fast = (dstSize >= FSE_BLOCKBOUND(srcSize)); + + if (fast) + return FSE_compress_usingCTable_generic(dst, dstSize, src, srcSize, ct, 1); + else + return FSE_compress_usingCTable_generic(dst, dstSize, src, srcSize, ct, 0); +} + + +size_t FSE_compressBound(size_t size) { return FSE_COMPRESSBOUND(size); } + +#define CHECK_V_F(e, f) size_t const e = f; if (ERR_isError(e)) return f +#define CHECK_F(f) { CHECK_V_F(_var_err__, f); } + +/* FSE_compress_wksp() : + * Same as FSE_compress2(), but using an externally allocated scratch buffer (`workSpace`). + * `wkspSize` size must be `(1< not compressible */ + if (maxCount < (srcSize >> 7)) return 0; /* Heuristic : not compressible enough */ + } + + tableLog = FSE_optimalTableLog(tableLog, srcSize, maxSymbolValue); + CHECK_F( FSE_normalizeCount(norm, tableLog, count, srcSize, maxSymbolValue) ); + + /* Write table description header */ + { CHECK_V_F(nc_err, FSE_writeNCount(op, oend-op, norm, maxSymbolValue, tableLog) ); + op += nc_err; + } + + /* Compress */ + CHECK_F( FSE_buildCTable_wksp(CTable, norm, maxSymbolValue, tableLog, scratchBuffer, scratchBufferSize) ); + { CHECK_V_F(cSize, FSE_compress_usingCTable(op, oend - op, src, srcSize, CTable) ); + if (cSize == 0) return 0; /* not enough space for compressed data */ + op += cSize; + } + + /* check compressibility */ + if ( (size_t)(op-ostart) >= srcSize-1 ) return 0; + + return op-ostart; +} + +typedef struct { + FSE_CTable CTable_max[FSE_CTABLE_SIZE_U32(FSE_MAX_TABLELOG, FSE_MAX_SYMBOL_VALUE)]; + BYTE scratchBuffer[1 << FSE_MAX_TABLELOG]; +} fseWkspMax_t; + +size_t FSE_compress2 (void* dst, size_t dstCapacity, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog) +{ + fseWkspMax_t scratchBuffer; + FSE_STATIC_ASSERT(sizeof(scratchBuffer) >= FSE_WKSP_SIZE_U32(FSE_MAX_TABLELOG, FSE_MAX_SYMBOL_VALUE)); /* compilation failures here means scratchBuffer is not large enough */ + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); + return FSE_compress_wksp(dst, dstCapacity, src, srcSize, maxSymbolValue, tableLog, &scratchBuffer, sizeof(scratchBuffer)); +} + +size_t FSE_compress (void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + return FSE_compress2(dst, dstCapacity, src, srcSize, FSE_MAX_SYMBOL_VALUE, FSE_DEFAULT_TABLELOG); +} + + +#endif /* FSE_COMMONDEFS_ONLY */ diff --git a/contrib/linux-kernel/lib/fse_decompress.c b/contrib/linux-kernel/lib/fse_decompress.c new file mode 100644 index 000000000..8474a4c07 --- /dev/null +++ b/contrib/linux-kernel/lib/fse_decompress.c @@ -0,0 +1,328 @@ +/* ****************************************************************** + FSE : Finite State Entropy decoder + Copyright (C) 2013-2015, Yann Collet. + + BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php) + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are + met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the + distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + You can contact the author at : + - FSE source repository : https://github.com/Cyan4973/FiniteStateEntropy + - Public forum : https://groups.google.com/forum/#!forum/lz4c +****************************************************************** */ + + +/* ************************************************************** +* Compiler specifics +****************************************************************/ +#ifdef _MSC_VER /* Visual Studio */ +# define FORCE_INLINE static __forceinline +# include /* For Visual 2005 */ +# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ +# pragma warning(disable : 4214) /* disable: C4214: non-int bitfields */ +#else +# if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ +# ifdef __GNUC__ +# define FORCE_INLINE static inline __attribute__((always_inline)) +# else +# define FORCE_INLINE static inline +# endif +# else +# define FORCE_INLINE static +# endif /* __STDC_VERSION__ */ +#endif + + +/* ************************************************************** +* Includes +****************************************************************/ +#include /* malloc, free, qsort */ +#include /* memcpy, memset */ +#include "bitstream.h" +#define FSE_STATIC_LINKING_ONLY +#include "fse.h" + + +/* ************************************************************** +* Error Management +****************************************************************/ +#define FSE_isError ERR_isError +#define FSE_STATIC_ASSERT(c) { enum { FSE_static_assert = 1/(int)(!!(c)) }; } /* use only *after* variable declarations */ + +/* check and forward error code */ +#define CHECK_F(f) { size_t const e = f; if (FSE_isError(e)) return e; } + + +/* ************************************************************** +* Templates +****************************************************************/ +/* + designed to be included + for type-specific functions (template emulation in C) + Objective is to write these functions only once, for improved maintenance +*/ + +/* safety checks */ +#ifndef FSE_FUNCTION_EXTENSION +# error "FSE_FUNCTION_EXTENSION must be defined" +#endif +#ifndef FSE_FUNCTION_TYPE +# error "FSE_FUNCTION_TYPE must be defined" +#endif + +/* Function names */ +#define FSE_CAT(X,Y) X##Y +#define FSE_FUNCTION_NAME(X,Y) FSE_CAT(X,Y) +#define FSE_TYPE_NAME(X,Y) FSE_CAT(X,Y) + + +/* Function templates */ +FSE_DTable* FSE_createDTable (unsigned tableLog) +{ + if (tableLog > FSE_TABLELOG_ABSOLUTE_MAX) tableLog = FSE_TABLELOG_ABSOLUTE_MAX; + return (FSE_DTable*)malloc( FSE_DTABLE_SIZE_U32(tableLog) * sizeof (U32) ); +} + +void FSE_freeDTable (FSE_DTable* dt) +{ + free(dt); +} + +size_t FSE_buildDTable(FSE_DTable* dt, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog) +{ + void* const tdPtr = dt+1; /* because *dt is unsigned, 32-bits aligned on 32-bits */ + FSE_DECODE_TYPE* const tableDecode = (FSE_DECODE_TYPE*) (tdPtr); + U16 symbolNext[FSE_MAX_SYMBOL_VALUE+1]; + + U32 const maxSV1 = maxSymbolValue + 1; + U32 const tableSize = 1 << tableLog; + U32 highThreshold = tableSize-1; + + /* Sanity Checks */ + if (maxSymbolValue > FSE_MAX_SYMBOL_VALUE) return ERROR(maxSymbolValue_tooLarge); + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); + + /* Init, lay down lowprob symbols */ + { FSE_DTableHeader DTableH; + DTableH.tableLog = (U16)tableLog; + DTableH.fastMode = 1; + { S16 const largeLimit= (S16)(1 << (tableLog-1)); + U32 s; + for (s=0; s= largeLimit) DTableH.fastMode=0; + symbolNext[s] = normalizedCounter[s]; + } } } + memcpy(dt, &DTableH, sizeof(DTableH)); + } + + /* Spread symbols */ + { U32 const tableMask = tableSize-1; + U32 const step = FSE_TABLESTEP(tableSize); + U32 s, position = 0; + for (s=0; s highThreshold) position = (position + step) & tableMask; /* lowprob area */ + } } + if (position!=0) return ERROR(GENERIC); /* position must reach all cells once, otherwise normalizedCounter is incorrect */ + } + + /* Build Decoding table */ + { U32 u; + for (u=0; utableLog = 0; + DTableH->fastMode = 0; + + cell->newState = 0; + cell->symbol = symbolValue; + cell->nbBits = 0; + + return 0; +} + + +size_t FSE_buildDTable_raw (FSE_DTable* dt, unsigned nbBits) +{ + void* ptr = dt; + FSE_DTableHeader* const DTableH = (FSE_DTableHeader*)ptr; + void* dPtr = dt + 1; + FSE_decode_t* const dinfo = (FSE_decode_t*)dPtr; + const unsigned tableSize = 1 << nbBits; + const unsigned tableMask = tableSize - 1; + const unsigned maxSV1 = tableMask+1; + unsigned s; + + /* Sanity checks */ + if (nbBits < 1) return ERROR(GENERIC); /* min size */ + + /* Build Decoding Table */ + DTableH->tableLog = (U16)nbBits; + DTableH->fastMode = 1; + for (s=0; s sizeof(bitD.bitContainer)*8) /* This test must be static */ + BIT_reloadDStream(&bitD); + + op[1] = FSE_GETSYMBOL(&state2); + + if (FSE_MAX_TABLELOG*4+7 > sizeof(bitD.bitContainer)*8) /* This test must be static */ + { if (BIT_reloadDStream(&bitD) > BIT_DStream_unfinished) { op+=2; break; } } + + op[2] = FSE_GETSYMBOL(&state1); + + if (FSE_MAX_TABLELOG*2+7 > sizeof(bitD.bitContainer)*8) /* This test must be static */ + BIT_reloadDStream(&bitD); + + op[3] = FSE_GETSYMBOL(&state2); + } + + /* tail */ + /* note : BIT_reloadDStream(&bitD) >= FSE_DStream_partiallyFilled; Ends at exactly BIT_DStream_completed */ + while (1) { + if (op>(omax-2)) return ERROR(dstSize_tooSmall); + *op++ = FSE_GETSYMBOL(&state1); + if (BIT_reloadDStream(&bitD)==BIT_DStream_overflow) { + *op++ = FSE_GETSYMBOL(&state2); + break; + } + + if (op>(omax-2)) return ERROR(dstSize_tooSmall); + *op++ = FSE_GETSYMBOL(&state2); + if (BIT_reloadDStream(&bitD)==BIT_DStream_overflow) { + *op++ = FSE_GETSYMBOL(&state1); + break; + } } + + return op-ostart; +} + + +size_t FSE_decompress_usingDTable(void* dst, size_t originalSize, + const void* cSrc, size_t cSrcSize, + const FSE_DTable* dt) +{ + const void* ptr = dt; + const FSE_DTableHeader* DTableH = (const FSE_DTableHeader*)ptr; + const U32 fastMode = DTableH->fastMode; + + /* select fast mode (static) */ + if (fastMode) return FSE_decompress_usingDTable_generic(dst, originalSize, cSrc, cSrcSize, dt, 1); + return FSE_decompress_usingDTable_generic(dst, originalSize, cSrc, cSrcSize, dt, 0); +} + + +size_t FSE_decompress_wksp(void* dst, size_t dstCapacity, const void* cSrc, size_t cSrcSize, FSE_DTable* workSpace, unsigned maxLog) +{ + const BYTE* const istart = (const BYTE*)cSrc; + const BYTE* ip = istart; + short counting[FSE_MAX_SYMBOL_VALUE+1]; + unsigned tableLog; + unsigned maxSymbolValue = FSE_MAX_SYMBOL_VALUE; + + /* normal FSE decoding mode */ + size_t const NCountLength = FSE_readNCount (counting, &maxSymbolValue, &tableLog, istart, cSrcSize); + if (FSE_isError(NCountLength)) return NCountLength; + //if (NCountLength >= cSrcSize) return ERROR(srcSize_wrong); /* too small input size; supposed to be already checked in NCountLength, only remaining case : NCountLength==cSrcSize */ + if (tableLog > maxLog) return ERROR(tableLog_tooLarge); + ip += NCountLength; + cSrcSize -= NCountLength; + + CHECK_F( FSE_buildDTable (workSpace, counting, maxSymbolValue, tableLog) ); + + return FSE_decompress_usingDTable (dst, dstCapacity, ip, cSrcSize, workSpace); /* always return, even if it is an error code */ +} + + +typedef FSE_DTable DTable_max_t[FSE_DTABLE_SIZE_U32(FSE_MAX_TABLELOG)]; + +size_t FSE_decompress(void* dst, size_t dstCapacity, const void* cSrc, size_t cSrcSize) +{ + DTable_max_t dt; /* Static analyzer seems unable to understand this table will be properly initialized later */ + return FSE_decompress_wksp(dst, dstCapacity, cSrc, cSrcSize, dt, FSE_MAX_TABLELOG); +} + + + +#endif /* FSE_COMMONDEFS_ONLY */ diff --git a/contrib/linux-kernel/lib/huf.h b/contrib/linux-kernel/lib/huf.h new file mode 100644 index 000000000..e5572760a --- /dev/null +++ b/contrib/linux-kernel/lib/huf.h @@ -0,0 +1,260 @@ +/* ****************************************************************** + Huffman coder, part of New Generation Entropy library + header file + Copyright (C) 2013-2016, Yann Collet. + + BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php) + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are + met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the + distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + You can contact the author at : + - Source repository : https://github.com/Cyan4973/FiniteStateEntropy +****************************************************************** */ +#ifndef HUF_H_298734234 +#define HUF_H_298734234 + +#if defined (__cplusplus) +extern "C" { +#endif + + +/* *** Dependencies *** */ +#include /* size_t */ + + +/* *** simple functions *** */ +/** +HUF_compress() : + Compress content from buffer 'src', of size 'srcSize', into buffer 'dst'. + 'dst' buffer must be already allocated. + Compression runs faster if `dstCapacity` >= HUF_compressBound(srcSize). + `srcSize` must be <= `HUF_BLOCKSIZE_MAX` == 128 KB. + @return : size of compressed data (<= `dstCapacity`). + Special values : if return == 0, srcData is not compressible => Nothing is stored within dst !!! + if return == 1, srcData is a single repeated byte symbol (RLE compression). + if HUF_isError(return), compression failed (more details using HUF_getErrorName()) +*/ +size_t HUF_compress(void* dst, size_t dstCapacity, + const void* src, size_t srcSize); + +/** +HUF_decompress() : + Decompress HUF data from buffer 'cSrc', of size 'cSrcSize', + into already allocated buffer 'dst', of minimum size 'dstSize'. + `originalSize` : **must** be the ***exact*** size of original (uncompressed) data. + Note : in contrast with FSE, HUF_decompress can regenerate + RLE (cSrcSize==1) and uncompressed (cSrcSize==dstSize) data, + because it knows size to regenerate. + @return : size of regenerated data (== originalSize), + or an error code, which can be tested using HUF_isError() +*/ +size_t HUF_decompress(void* dst, size_t originalSize, + const void* cSrc, size_t cSrcSize); + + +/* *** Tool functions *** */ +#define HUF_BLOCKSIZE_MAX (128 * 1024) /**< maximum input size for a single block compressed with HUF_compress */ +size_t HUF_compressBound(size_t size); /**< maximum compressed size (worst case) */ + +/* Error Management */ +unsigned HUF_isError(size_t code); /**< tells if a return value is an error code */ +const char* HUF_getErrorName(size_t code); /**< provides error code string (useful for debugging) */ + + +/* *** Advanced function *** */ + +/** HUF_compress2() : + * Same as HUF_compress(), but offers direct control over `maxSymbolValue` and `tableLog` . + * `tableLog` must be `<= HUF_TABLELOG_MAX` . */ +size_t HUF_compress2 (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog); + +/** HUF_compress4X_wksp() : +* Same as HUF_compress2(), but uses externally allocated `workSpace`, which must be a table of >= 1024 unsigned */ +size_t HUF_compress4X_wksp (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize); /**< `workSpace` must be a table of at least HUF_WORKSPACE_SIZE_U32 unsigned */ + + + +#ifdef HUF_STATIC_LINKING_ONLY + +/* *** Dependencies *** */ +#include "mem.h" /* U32 */ + + +/* *** Constants *** */ +#define HUF_TABLELOG_MAX 12 /* max configured tableLog (for static allocation); can be modified up to HUF_ABSOLUTEMAX_TABLELOG */ +#define HUF_TABLELOG_DEFAULT 11 /* tableLog by default, when not specified */ +#define HUF_SYMBOLVALUE_MAX 255 + +#define HUF_TABLELOG_ABSOLUTEMAX 15 /* absolute limit of HUF_MAX_TABLELOG. Beyond that value, code does not work */ +#if (HUF_TABLELOG_MAX > HUF_TABLELOG_ABSOLUTEMAX) +# error "HUF_TABLELOG_MAX is too large !" +#endif + + +/* **************************************** +* Static allocation +******************************************/ +/* HUF buffer bounds */ +#define HUF_CTABLEBOUND 129 +#define HUF_BLOCKBOUND(size) (size + (size>>8) + 8) /* only true if incompressible pre-filtered with fast heuristic */ +#define HUF_COMPRESSBOUND(size) (HUF_CTABLEBOUND + HUF_BLOCKBOUND(size)) /* Macro version, useful for static allocation */ + +/* static allocation of HUF's Compression Table */ +#define HUF_CREATE_STATIC_CTABLE(name, maxSymbolValue) \ + U32 name##hb[maxSymbolValue+1]; \ + void* name##hv = &(name##hb); \ + HUF_CElt* name = (HUF_CElt*)(name##hv) /* no final ; */ + +/* static allocation of HUF's DTable */ +typedef U32 HUF_DTable; +#define HUF_DTABLE_SIZE(maxTableLog) (1 + (1<<(maxTableLog))) +#define HUF_CREATE_STATIC_DTABLEX2(DTable, maxTableLog) \ + HUF_DTable DTable[HUF_DTABLE_SIZE((maxTableLog)-1)] = { ((U32)((maxTableLog)-1) * 0x01000001) } +#define HUF_CREATE_STATIC_DTABLEX4(DTable, maxTableLog) \ + HUF_DTable DTable[HUF_DTABLE_SIZE(maxTableLog)] = { ((U32)(maxTableLog) * 0x01000001) } + +/* The workspace must have alignment at least 4 and be at least this large */ +#define HUF_WORKSPACE_SIZE (6 << 10) +#define HUF_WORKSPACE_SIZE_U32 (HUF_WORKSPACE_SIZE / sizeof(U32)) + + +/* **************************************** +* Advanced decompression functions +******************************************/ +size_t HUF_decompress4X2 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< single-symbol decoder */ +size_t HUF_decompress4X4 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< double-symbols decoder */ + +size_t HUF_decompress4X_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< decodes RLE and uncompressed */ +size_t HUF_decompress4X_hufOnly(HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< considers RLE and uncompressed as errors */ +size_t HUF_decompress4X2_DCtx(HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< single-symbol decoder */ +size_t HUF_decompress4X4_DCtx(HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< double-symbols decoder */ + + +/* **************************************** +* HUF detailed API +******************************************/ +/*! +HUF_compress() does the following: +1. count symbol occurrence from source[] into table count[] using FSE_count() +2. (optional) refine tableLog using HUF_optimalTableLog() +3. build Huffman table from count using HUF_buildCTable() +4. save Huffman table to memory buffer using HUF_writeCTable() +5. encode the data stream using HUF_compress4X_usingCTable() + +The following API allows targeting specific sub-functions for advanced tasks. +For example, it's possible to compress several blocks using the same 'CTable', +or to save and regenerate 'CTable' using external methods. +*/ +/* FSE_count() : find it within "fse.h" */ +unsigned HUF_optimalTableLog(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue); +typedef struct HUF_CElt_s HUF_CElt; /* incomplete type */ +size_t HUF_buildCTable (HUF_CElt* CTable, const unsigned* count, unsigned maxSymbolValue, unsigned maxNbBits); +size_t HUF_writeCTable (void* dst, size_t maxDstSize, const HUF_CElt* CTable, unsigned maxSymbolValue, unsigned huffLog); +size_t HUF_compress4X_usingCTable(void* dst, size_t dstSize, const void* src, size_t srcSize, const HUF_CElt* CTable); + +typedef enum { + HUF_repeat_none, /**< Cannot use the previous table */ + HUF_repeat_check, /**< Can use the previous table but it must be checked. Note : The previous table must have been constructed by HUF_compress{1, 4}X_repeat */ + HUF_repeat_valid /**< Can use the previous table and it is asumed to be valid */ + } HUF_repeat; +/** HUF_compress4X_repeat() : +* Same as HUF_compress4X_wksp(), but considers using hufTable if *repeat != HUF_repeat_none. +* If it uses hufTable it does not modify hufTable or repeat. +* If it doesn't, it sets *repeat = HUF_repeat_none, and it sets hufTable to the table used. +* If preferRepeat then the old table will always be used if valid. */ +size_t HUF_compress4X_repeat(void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize, HUF_CElt* hufTable, HUF_repeat* repeat, int preferRepeat); /**< `workSpace` must be a table of at least HUF_WORKSPACE_SIZE_U32 unsigned */ + +/** HUF_buildCTable_wksp() : + * Same as HUF_buildCTable(), but using externally allocated scratch buffer. + * `workSpace` must be aligned on 4-bytes boundaries, and be at least as large as a table of 1024 unsigned. + */ +size_t HUF_buildCTable_wksp (HUF_CElt* tree, const U32* count, U32 maxSymbolValue, U32 maxNbBits, void* workSpace, size_t wkspSize); + +/*! HUF_readStats() : + Read compact Huffman tree, saved by HUF_writeCTable(). + `huffWeight` is destination buffer. + @return : size read from `src` , or an error Code . + Note : Needed by HUF_readCTable() and HUF_readDTableXn() . */ +size_t HUF_readStats(BYTE* huffWeight, size_t hwSize, U32* rankStats, + U32* nbSymbolsPtr, U32* tableLogPtr, + const void* src, size_t srcSize); + +/** HUF_readCTable() : +* Loading a CTable saved with HUF_writeCTable() */ +size_t HUF_readCTable (HUF_CElt* CTable, unsigned maxSymbolValue, const void* src, size_t srcSize); + + +/* +HUF_decompress() does the following: +1. select the decompression algorithm (X2, X4) based on pre-computed heuristics +2. build Huffman table from save, using HUF_readDTableXn() +3. decode 1 or 4 segments in parallel using HUF_decompressSXn_usingDTable +*/ + +/** HUF_selectDecoder() : +* Tells which decoder is likely to decode faster, +* based on a set of pre-determined metrics. +* @return : 0==HUF_decompress4X2, 1==HUF_decompress4X4 . +* Assumption : 0 < cSrcSize < dstSize <= 128 KB */ +U32 HUF_selectDecoder (size_t dstSize, size_t cSrcSize); + +size_t HUF_readDTableX2 (HUF_DTable* DTable, const void* src, size_t srcSize); +size_t HUF_readDTableX4 (HUF_DTable* DTable, const void* src, size_t srcSize); + +size_t HUF_decompress4X_usingDTable(void* dst, size_t maxDstSize, const void* cSrc, size_t cSrcSize, const HUF_DTable* DTable); +size_t HUF_decompress4X2_usingDTable(void* dst, size_t maxDstSize, const void* cSrc, size_t cSrcSize, const HUF_DTable* DTable); +size_t HUF_decompress4X4_usingDTable(void* dst, size_t maxDstSize, const void* cSrc, size_t cSrcSize, const HUF_DTable* DTable); + + +/* single stream variants */ + +size_t HUF_compress1X (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog); +size_t HUF_compress1X_wksp (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize); /**< `workSpace` must be a table of at least HUF_WORKSPACE_SIZE_U32 unsigned */ +size_t HUF_compress1X_usingCTable(void* dst, size_t dstSize, const void* src, size_t srcSize, const HUF_CElt* CTable); +/** HUF_compress1X_repeat() : +* Same as HUF_compress1X_wksp(), but considers using hufTable if *repeat != HUF_repeat_none. +* If it uses hufTable it does not modify hufTable or repeat. +* If it doesn't, it sets *repeat = HUF_repeat_none, and it sets hufTable to the table used. +* If preferRepeat then the old table will always be used if valid. */ +size_t HUF_compress1X_repeat(void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize, HUF_CElt* hufTable, HUF_repeat* repeat, int preferRepeat); /**< `workSpace` must be a table of at least HUF_WORKSPACE_SIZE_U32 unsigned */ + +size_t HUF_decompress1X2 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /* single-symbol decoder */ +size_t HUF_decompress1X4 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /* double-symbol decoder */ + +size_t HUF_decompress1X_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); +size_t HUF_decompress1X2_DCtx(HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< single-symbol decoder */ +size_t HUF_decompress1X4_DCtx(HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< double-symbols decoder */ + +size_t HUF_decompress1X_usingDTable(void* dst, size_t maxDstSize, const void* cSrc, size_t cSrcSize, const HUF_DTable* DTable); /**< automatic selection of sing or double symbol decoder, based on DTable */ +size_t HUF_decompress1X2_usingDTable(void* dst, size_t maxDstSize, const void* cSrc, size_t cSrcSize, const HUF_DTable* DTable); +size_t HUF_decompress1X4_usingDTable(void* dst, size_t maxDstSize, const void* cSrc, size_t cSrcSize, const HUF_DTable* DTable); + +#endif /* HUF_STATIC_LINKING_ONLY */ + + +#if defined (__cplusplus) +} +#endif + +#endif /* HUF_H_298734234 */ diff --git a/contrib/linux-kernel/lib/huf_compress.c b/contrib/linux-kernel/lib/huf_compress.c new file mode 100644 index 000000000..fe11aafb8 --- /dev/null +++ b/contrib/linux-kernel/lib/huf_compress.c @@ -0,0 +1,684 @@ +/* ****************************************************************** + Huffman encoder, part of New Generation Entropy library + Copyright (C) 2013-2016, Yann Collet. + + BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php) + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are + met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the + distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + You can contact the author at : + - FSE+HUF source repository : https://github.com/Cyan4973/FiniteStateEntropy + - Public forum : https://groups.google.com/forum/#!forum/lz4c +****************************************************************** */ + +/* ************************************************************** +* Compiler specifics +****************************************************************/ +#ifdef _MSC_VER /* Visual Studio */ +# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ +#endif + + +/* ************************************************************** +* Includes +****************************************************************/ +#include /* memcpy, memset */ +#include /* printf (debug) */ +#include "bitstream.h" +#define FSE_STATIC_LINKING_ONLY /* FSE_optimalTableLog_internal */ +#include "fse.h" /* header compression */ +#define HUF_STATIC_LINKING_ONLY +#include "huf.h" + + +/* ************************************************************** +* Error Management +****************************************************************/ +#define HUF_STATIC_ASSERT(c) { enum { HUF_static_assert = 1/(int)(!!(c)) }; } /* use only *after* variable declarations */ +#define CHECK_V_F(e, f) size_t const e = f; if (ERR_isError(e)) return f +#define CHECK_F(f) { CHECK_V_F(_var_err__, f); } + + +/* ************************************************************** +* Utils +****************************************************************/ +unsigned HUF_optimalTableLog(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue) +{ + return FSE_optimalTableLog_internal(maxTableLog, srcSize, maxSymbolValue, 1); +} + + +/* ******************************************************* +* HUF : Huffman block compression +*********************************************************/ +/* HUF_compressWeights() : + * Same as FSE_compress(), but dedicated to huff0's weights compression. + * The use case needs much less stack memory. + * Note : all elements within weightTable are supposed to be <= HUF_TABLELOG_MAX. + */ +#define MAX_FSE_TABLELOG_FOR_HUFF_HEADER 6 +size_t HUF_compressWeights (void* dst, size_t dstSize, const void* weightTable, size_t wtSize) +{ + BYTE* const ostart = (BYTE*) dst; + BYTE* op = ostart; + BYTE* const oend = ostart + dstSize; + + U32 maxSymbolValue = HUF_TABLELOG_MAX; + U32 tableLog = MAX_FSE_TABLELOG_FOR_HUFF_HEADER; + + FSE_CTable CTable[FSE_CTABLE_SIZE_U32(MAX_FSE_TABLELOG_FOR_HUFF_HEADER, HUF_TABLELOG_MAX)]; + BYTE scratchBuffer[1< not compressible */ + } + + tableLog = FSE_optimalTableLog(tableLog, wtSize, maxSymbolValue); + CHECK_F( FSE_normalizeCount(norm, tableLog, count, wtSize, maxSymbolValue) ); + + /* Write table description header */ + { CHECK_V_F(hSize, FSE_writeNCount(op, oend-op, norm, maxSymbolValue, tableLog) ); + op += hSize; + } + + /* Compress */ + CHECK_F( FSE_buildCTable_wksp(CTable, norm, maxSymbolValue, tableLog, scratchBuffer, sizeof(scratchBuffer)) ); + { CHECK_V_F(cSize, FSE_compress_usingCTable(op, oend - op, weightTable, wtSize, CTable) ); + if (cSize == 0) return 0; /* not enough space for compressed data */ + op += cSize; + } + + return op-ostart; +} + + +struct HUF_CElt_s { + U16 val; + BYTE nbBits; +}; /* typedef'd to HUF_CElt within "huf.h" */ + +/*! HUF_writeCTable() : + `CTable` : Huffman tree to save, using huf representation. + @return : size of saved CTable */ +size_t HUF_writeCTable (void* dst, size_t maxDstSize, + const HUF_CElt* CTable, U32 maxSymbolValue, U32 huffLog) +{ + BYTE bitsToWeight[HUF_TABLELOG_MAX + 1]; /* precomputed conversion table */ + BYTE huffWeight[HUF_SYMBOLVALUE_MAX]; + BYTE* op = (BYTE*)dst; + U32 n; + + /* check conditions */ + if (maxSymbolValue > HUF_SYMBOLVALUE_MAX) return ERROR(maxSymbolValue_tooLarge); + + /* convert to weight */ + bitsToWeight[0] = 0; + for (n=1; n1) & (hSize < maxSymbolValue/2)) { /* FSE compressed */ + op[0] = (BYTE)hSize; + return hSize+1; + } } + + /* write raw values as 4-bits (max : 15) */ + if (maxSymbolValue > (256-128)) return ERROR(GENERIC); /* should not happen : likely means source cannot be compressed */ + if (((maxSymbolValue+1)/2) + 1 > maxDstSize) return ERROR(dstSize_tooSmall); /* not enough space within dst buffer */ + op[0] = (BYTE)(128 /*special case*/ + (maxSymbolValue-1)); + huffWeight[maxSymbolValue] = 0; /* to be sure it doesn't cause msan issue in final combination */ + for (n=0; n HUF_TABLELOG_MAX) return ERROR(tableLog_tooLarge); + if (nbSymbols > maxSymbolValue+1) return ERROR(maxSymbolValue_tooSmall); + + /* Prepare base value per rank */ + { U32 n, nextRankStart = 0; + for (n=1; n<=tableLog; n++) { + U32 current = nextRankStart; + nextRankStart += (rankVal[n] << (n-1)); + rankVal[n] = current; + } } + + /* fill nbBits */ + { U32 n; for (n=0; nn=tableLog+1 */ + U16 valPerRank[HUF_TABLELOG_MAX+2] = {0}; + { U32 n; for (n=0; n0; n--) { /* start at n=tablelog <-> w=1 */ + valPerRank[n] = min; /* get starting value within each rank */ + min += nbPerRank[n]; + min >>= 1; + } } + /* assign value within rank, symbol order */ + { U32 n; for (n=0; n<=maxSymbolValue; n++) CTable[n].val = valPerRank[CTable[n].nbBits]++; } + } + + return readSize; +} + + +typedef struct nodeElt_s { + U32 count; + U16 parent; + BYTE byte; + BYTE nbBits; +} nodeElt; + +static U32 HUF_setMaxHeight(nodeElt* huffNode, U32 lastNonNull, U32 maxNbBits) +{ + const U32 largestBits = huffNode[lastNonNull].nbBits; + if (largestBits <= maxNbBits) return largestBits; /* early exit : no elt > maxNbBits */ + + /* there are several too large elements (at least >= 2) */ + { int totalCost = 0; + const U32 baseCost = 1 << (largestBits - maxNbBits); + U32 n = lastNonNull; + + while (huffNode[n].nbBits > maxNbBits) { + totalCost += baseCost - (1 << (largestBits - huffNode[n].nbBits)); + huffNode[n].nbBits = (BYTE)maxNbBits; + n --; + } /* n stops at huffNode[n].nbBits <= maxNbBits */ + while (huffNode[n].nbBits == maxNbBits) n--; /* n end at index of smallest symbol using < maxNbBits */ + + /* renorm totalCost */ + totalCost >>= (largestBits - maxNbBits); /* note : totalCost is necessarily a multiple of baseCost */ + + /* repay normalized cost */ + { U32 const noSymbol = 0xF0F0F0F0; + U32 rankLast[HUF_TABLELOG_MAX+2]; + int pos; + + /* Get pos of last (smallest) symbol per rank */ + memset(rankLast, 0xF0, sizeof(rankLast)); + { U32 currentNbBits = maxNbBits; + for (pos=n ; pos >= 0; pos--) { + if (huffNode[pos].nbBits >= currentNbBits) continue; + currentNbBits = huffNode[pos].nbBits; /* < maxNbBits */ + rankLast[maxNbBits-currentNbBits] = pos; + } } + + while (totalCost > 0) { + U32 nBitsToDecrease = BIT_highbit32(totalCost) + 1; + for ( ; nBitsToDecrease > 1; nBitsToDecrease--) { + U32 highPos = rankLast[nBitsToDecrease]; + U32 lowPos = rankLast[nBitsToDecrease-1]; + if (highPos == noSymbol) continue; + if (lowPos == noSymbol) break; + { U32 const highTotal = huffNode[highPos].count; + U32 const lowTotal = 2 * huffNode[lowPos].count; + if (highTotal <= lowTotal) break; + } } + /* only triggered when no more rank 1 symbol left => find closest one (note : there is necessarily at least one !) */ + while ((nBitsToDecrease<=HUF_TABLELOG_MAX) && (rankLast[nBitsToDecrease] == noSymbol)) /* HUF_MAX_TABLELOG test just to please gcc 5+; but it should not be necessary */ + nBitsToDecrease ++; + totalCost -= 1 << (nBitsToDecrease-1); + if (rankLast[nBitsToDecrease-1] == noSymbol) + rankLast[nBitsToDecrease-1] = rankLast[nBitsToDecrease]; /* this rank is no longer empty */ + huffNode[rankLast[nBitsToDecrease]].nbBits ++; + if (rankLast[nBitsToDecrease] == 0) /* special case, reached largest symbol */ + rankLast[nBitsToDecrease] = noSymbol; + else { + rankLast[nBitsToDecrease]--; + if (huffNode[rankLast[nBitsToDecrease]].nbBits != maxNbBits-nBitsToDecrease) + rankLast[nBitsToDecrease] = noSymbol; /* this rank is now empty */ + } } /* while (totalCost > 0) */ + + while (totalCost < 0) { /* Sometimes, cost correction overshoot */ + if (rankLast[1] == noSymbol) { /* special case : no rank 1 symbol (using maxNbBits-1); let's create one from largest rank 0 (using maxNbBits) */ + while (huffNode[n].nbBits == maxNbBits) n--; + huffNode[n+1].nbBits--; + rankLast[1] = n+1; + totalCost++; + continue; + } + huffNode[ rankLast[1] + 1 ].nbBits--; + rankLast[1]++; + totalCost ++; + } } } /* there are several too large elements (at least >= 2) */ + + return maxNbBits; +} + + +typedef struct { + U32 base; + U32 current; +} rankPos; + +static void HUF_sort(nodeElt* huffNode, const U32* count, U32 maxSymbolValue) +{ + rankPos rank[32]; + U32 n; + + memset(rank, 0, sizeof(rank)); + for (n=0; n<=maxSymbolValue; n++) { + U32 r = BIT_highbit32(count[n] + 1); + rank[r].base ++; + } + for (n=30; n>0; n--) rank[n-1].base += rank[n].base; + for (n=0; n<32; n++) rank[n].current = rank[n].base; + for (n=0; n<=maxSymbolValue; n++) { + U32 const c = count[n]; + U32 const r = BIT_highbit32(c+1) + 1; + U32 pos = rank[r].current++; + while ((pos > rank[r].base) && (c > huffNode[pos-1].count)) huffNode[pos]=huffNode[pos-1], pos--; + huffNode[pos].count = c; + huffNode[pos].byte = (BYTE)n; + } +} + + +/** HUF_buildCTable_wksp() : + * Same as HUF_buildCTable(), but using externally allocated scratch buffer. + * `workSpace` must be aligned on 4-bytes boundaries, and be at least as large as a table of 1024 unsigned. + */ +#define STARTNODE (HUF_SYMBOLVALUE_MAX+1) +typedef nodeElt huffNodeTable[2*HUF_SYMBOLVALUE_MAX+1 +1]; +size_t HUF_buildCTable_wksp (HUF_CElt* tree, const U32* count, U32 maxSymbolValue, U32 maxNbBits, void* workSpace, size_t wkspSize) +{ + nodeElt* const huffNode0 = (nodeElt*)workSpace; + nodeElt* const huffNode = huffNode0+1; + U32 n, nonNullRank; + int lowS, lowN; + U16 nodeNb = STARTNODE; + U32 nodeRoot; + + /* safety checks */ + if (wkspSize < sizeof(huffNodeTable)) return ERROR(GENERIC); /* workSpace is not large enough */ + if (maxNbBits == 0) maxNbBits = HUF_TABLELOG_DEFAULT; + if (maxSymbolValue > HUF_SYMBOLVALUE_MAX) return ERROR(GENERIC); + memset(huffNode0, 0, sizeof(huffNodeTable)); + + /* sort, decreasing order */ + HUF_sort(huffNode, count, maxSymbolValue); + + /* init for parents */ + nonNullRank = maxSymbolValue; + while(huffNode[nonNullRank].count == 0) nonNullRank--; + lowS = nonNullRank; nodeRoot = nodeNb + lowS - 1; lowN = nodeNb; + huffNode[nodeNb].count = huffNode[lowS].count + huffNode[lowS-1].count; + huffNode[lowS].parent = huffNode[lowS-1].parent = nodeNb; + nodeNb++; lowS-=2; + for (n=nodeNb; n<=nodeRoot; n++) huffNode[n].count = (U32)(1U<<30); + huffNode0[0].count = (U32)(1U<<31); /* fake entry, strong barrier */ + + /* create parents */ + while (nodeNb <= nodeRoot) { + U32 n1 = (huffNode[lowS].count < huffNode[lowN].count) ? lowS-- : lowN++; + U32 n2 = (huffNode[lowS].count < huffNode[lowN].count) ? lowS-- : lowN++; + huffNode[nodeNb].count = huffNode[n1].count + huffNode[n2].count; + huffNode[n1].parent = huffNode[n2].parent = nodeNb; + nodeNb++; + } + + /* distribute weights (unlimited tree height) */ + huffNode[nodeRoot].nbBits = 0; + for (n=nodeRoot-1; n>=STARTNODE; n--) + huffNode[n].nbBits = huffNode[ huffNode[n].parent ].nbBits + 1; + for (n=0; n<=nonNullRank; n++) + huffNode[n].nbBits = huffNode[ huffNode[n].parent ].nbBits + 1; + + /* enforce maxTableLog */ + maxNbBits = HUF_setMaxHeight(huffNode, nonNullRank, maxNbBits); + + /* fill result into tree (val, nbBits) */ + { U16 nbPerRank[HUF_TABLELOG_MAX+1] = {0}; + U16 valPerRank[HUF_TABLELOG_MAX+1] = {0}; + if (maxNbBits > HUF_TABLELOG_MAX) return ERROR(GENERIC); /* check fit into table */ + for (n=0; n<=nonNullRank; n++) + nbPerRank[huffNode[n].nbBits]++; + /* determine stating value per rank */ + { U16 min = 0; + for (n=maxNbBits; n>0; n--) { + valPerRank[n] = min; /* get starting value within each rank */ + min += nbPerRank[n]; + min >>= 1; + } } + for (n=0; n<=maxSymbolValue; n++) + tree[huffNode[n].byte].nbBits = huffNode[n].nbBits; /* push nbBits per symbol, symbol order */ + for (n=0; n<=maxSymbolValue; n++) + tree[n].val = valPerRank[tree[n].nbBits]++; /* assign value within rank, symbol order */ + } + + return maxNbBits; +} + +/** HUF_buildCTable() : + * Note : count is used before tree is written, so they can safely overlap + */ +size_t HUF_buildCTable (HUF_CElt* tree, const U32* count, U32 maxSymbolValue, U32 maxNbBits) +{ + huffNodeTable nodeTable; + return HUF_buildCTable_wksp(tree, count, maxSymbolValue, maxNbBits, nodeTable, sizeof(nodeTable)); +} + +static size_t HUF_estimateCompressedSize(HUF_CElt* CTable, const unsigned* count, unsigned maxSymbolValue) +{ + size_t nbBits = 0; + int s; + for (s = 0; s <= (int)maxSymbolValue; ++s) { + nbBits += CTable[s].nbBits * count[s]; + } + return nbBits >> 3; +} + +static int HUF_validateCTable(const HUF_CElt* CTable, const unsigned* count, unsigned maxSymbolValue) { + int bad = 0; + int s; + for (s = 0; s <= (int)maxSymbolValue; ++s) { + bad |= (count[s] != 0) & (CTable[s].nbBits == 0); + } + return !bad; +} + +static void HUF_encodeSymbol(BIT_CStream_t* bitCPtr, U32 symbol, const HUF_CElt* CTable) +{ + BIT_addBitsFast(bitCPtr, CTable[symbol].val, CTable[symbol].nbBits); +} + +size_t HUF_compressBound(size_t size) { return HUF_COMPRESSBOUND(size); } + +#define HUF_FLUSHBITS(s) (fast ? BIT_flushBitsFast(s) : BIT_flushBits(s)) + +#define HUF_FLUSHBITS_1(stream) \ + if (sizeof((stream)->bitContainer)*8 < HUF_TABLELOG_MAX*2+7) HUF_FLUSHBITS(stream) + +#define HUF_FLUSHBITS_2(stream) \ + if (sizeof((stream)->bitContainer)*8 < HUF_TABLELOG_MAX*4+7) HUF_FLUSHBITS(stream) + +size_t HUF_compress1X_usingCTable(void* dst, size_t dstSize, const void* src, size_t srcSize, const HUF_CElt* CTable) +{ + const BYTE* ip = (const BYTE*) src; + BYTE* const ostart = (BYTE*)dst; + BYTE* const oend = ostart + dstSize; + BYTE* op = ostart; + size_t n; + const unsigned fast = (dstSize >= HUF_BLOCKBOUND(srcSize)); + BIT_CStream_t bitC; + + /* init */ + if (dstSize < 8) return 0; /* not enough space to compress */ + { size_t const initErr = BIT_initCStream(&bitC, op, oend-op); + if (HUF_isError(initErr)) return 0; } + + n = srcSize & ~3; /* join to mod 4 */ + switch (srcSize & 3) + { + case 3 : HUF_encodeSymbol(&bitC, ip[n+ 2], CTable); + HUF_FLUSHBITS_2(&bitC); + case 2 : HUF_encodeSymbol(&bitC, ip[n+ 1], CTable); + HUF_FLUSHBITS_1(&bitC); + case 1 : HUF_encodeSymbol(&bitC, ip[n+ 0], CTable); + HUF_FLUSHBITS(&bitC); + case 0 : + default: ; + } + + for (; n>0; n-=4) { /* note : n&3==0 at this stage */ + HUF_encodeSymbol(&bitC, ip[n- 1], CTable); + HUF_FLUSHBITS_1(&bitC); + HUF_encodeSymbol(&bitC, ip[n- 2], CTable); + HUF_FLUSHBITS_2(&bitC); + HUF_encodeSymbol(&bitC, ip[n- 3], CTable); + HUF_FLUSHBITS_1(&bitC); + HUF_encodeSymbol(&bitC, ip[n- 4], CTable); + HUF_FLUSHBITS(&bitC); + } + + return BIT_closeCStream(&bitC); +} + + +size_t HUF_compress4X_usingCTable(void* dst, size_t dstSize, const void* src, size_t srcSize, const HUF_CElt* CTable) +{ + size_t const segmentSize = (srcSize+3)/4; /* first 3 segments */ + const BYTE* ip = (const BYTE*) src; + const BYTE* const iend = ip + srcSize; + BYTE* const ostart = (BYTE*) dst; + BYTE* const oend = ostart + dstSize; + BYTE* op = ostart; + + if (dstSize < 6 + 1 + 1 + 1 + 8) return 0; /* minimum space to compress successfully */ + if (srcSize < 12) return 0; /* no saving possible : too small input */ + op += 6; /* jumpTable */ + + { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, segmentSize, CTable) ); + if (cSize==0) return 0; + MEM_writeLE16(ostart, (U16)cSize); + op += cSize; + } + + ip += segmentSize; + { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, segmentSize, CTable) ); + if (cSize==0) return 0; + MEM_writeLE16(ostart+2, (U16)cSize); + op += cSize; + } + + ip += segmentSize; + { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, segmentSize, CTable) ); + if (cSize==0) return 0; + MEM_writeLE16(ostart+4, (U16)cSize); + op += cSize; + } + + ip += segmentSize; + { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, iend-ip, CTable) ); + if (cSize==0) return 0; + op += cSize; + } + + return op-ostart; +} + + +static size_t HUF_compressCTable_internal( + BYTE* const ostart, BYTE* op, BYTE* const oend, + const void* src, size_t srcSize, + unsigned singleStream, const HUF_CElt* CTable) +{ + size_t const cSize = singleStream ? + HUF_compress1X_usingCTable(op, oend - op, src, srcSize, CTable) : + HUF_compress4X_usingCTable(op, oend - op, src, srcSize, CTable); + if (HUF_isError(cSize)) { return cSize; } + if (cSize==0) { return 0; } /* uncompressible */ + op += cSize; + /* check compressibility */ + if ((size_t)(op-ostart) >= srcSize-1) { return 0; } + return op-ostart; +} + + +/* `workSpace` must a table of at least 1024 unsigned */ +static size_t HUF_compress_internal ( + void* dst, size_t dstSize, + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog, + unsigned singleStream, + void* workSpace, size_t wkspSize, + HUF_CElt* oldHufTable, HUF_repeat* repeat, int preferRepeat) +{ + BYTE* const ostart = (BYTE*)dst; + BYTE* const oend = ostart + dstSize; + BYTE* op = ostart; + + U32* count; + size_t const countSize = sizeof(U32) * (HUF_SYMBOLVALUE_MAX + 1); + HUF_CElt* CTable; + size_t const CTableSize = sizeof(HUF_CElt) * (HUF_SYMBOLVALUE_MAX + 1); + + /* checks & inits */ + if (wkspSize < sizeof(huffNodeTable) + countSize + CTableSize) return ERROR(GENERIC); + if (!srcSize) return 0; /* Uncompressed (note : 1 means rle, so first byte must be correct) */ + if (!dstSize) return 0; /* cannot fit within dst budget */ + if (srcSize > HUF_BLOCKSIZE_MAX) return ERROR(srcSize_wrong); /* current block size limit */ + if (huffLog > HUF_TABLELOG_MAX) return ERROR(tableLog_tooLarge); + if (!maxSymbolValue) maxSymbolValue = HUF_SYMBOLVALUE_MAX; + if (!huffLog) huffLog = HUF_TABLELOG_DEFAULT; + + count = (U32*)workSpace; + workSpace = (BYTE*)workSpace + countSize; + wkspSize -= countSize; + CTable = (HUF_CElt*)workSpace; + workSpace = (BYTE*)workSpace + CTableSize; + wkspSize -= CTableSize; + + /* Heuristic : If we don't need to check the validity of the old table use the old table for small inputs */ + if (preferRepeat && repeat && *repeat == HUF_repeat_valid) { + return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, oldHufTable); + } + + /* Scan input and build symbol stats */ + { CHECK_V_F(largest, FSE_count_wksp (count, &maxSymbolValue, (const BYTE*)src, srcSize, (U32*)workSpace) ); + if (largest == srcSize) { *ostart = ((const BYTE*)src)[0]; return 1; } /* single symbol, rle */ + if (largest <= (srcSize >> 7)+1) return 0; /* Fast heuristic : not compressible enough */ + } + + /* Check validity of previous table */ + if (repeat && *repeat == HUF_repeat_check && !HUF_validateCTable(oldHufTable, count, maxSymbolValue)) { + *repeat = HUF_repeat_none; + } + /* Heuristic : use existing table for small inputs */ + if (preferRepeat && repeat && *repeat != HUF_repeat_none) { + return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, oldHufTable); + } + + /* Build Huffman Tree */ + huffLog = HUF_optimalTableLog(huffLog, srcSize, maxSymbolValue); + { CHECK_V_F(maxBits, HUF_buildCTable_wksp (CTable, count, maxSymbolValue, huffLog, workSpace, wkspSize) ); + huffLog = (U32)maxBits; + /* Zero the unused symbols so we can check it for validity */ + memset(CTable + maxSymbolValue + 1, 0, CTableSize - (maxSymbolValue + 1) * sizeof(HUF_CElt)); + } + + /* Write table description header */ + { CHECK_V_F(hSize, HUF_writeCTable (op, dstSize, CTable, maxSymbolValue, huffLog) ); + /* Check if using the previous table will be beneficial */ + if (repeat && *repeat != HUF_repeat_none) { + size_t const oldSize = HUF_estimateCompressedSize(oldHufTable, count, maxSymbolValue); + size_t const newSize = HUF_estimateCompressedSize(CTable, count, maxSymbolValue); + if (oldSize <= hSize + newSize || hSize + 12 >= srcSize) { + return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, oldHufTable); + } + } + /* Use the new table */ + if (hSize + 12ul >= srcSize) { return 0; } + op += hSize; + if (repeat) { *repeat = HUF_repeat_none; } + if (oldHufTable) { memcpy(oldHufTable, CTable, CTableSize); } /* Save the new table */ + } + return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, CTable); +} + + +size_t HUF_compress1X_wksp (void* dst, size_t dstSize, + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog, + void* workSpace, size_t wkspSize) +{ + return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 1 /* single stream */, workSpace, wkspSize, NULL, NULL, 0); +} + +size_t HUF_compress1X_repeat (void* dst, size_t dstSize, + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog, + void* workSpace, size_t wkspSize, + HUF_CElt* hufTable, HUF_repeat* repeat, int preferRepeat) +{ + return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 1 /* single stream */, workSpace, wkspSize, hufTable, repeat, preferRepeat); +} + +size_t HUF_compress1X (void* dst, size_t dstSize, + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog) +{ + unsigned workSpace[1024]; + return HUF_compress1X_wksp(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, workSpace, sizeof(workSpace)); +} + +size_t HUF_compress4X_wksp (void* dst, size_t dstSize, + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog, + void* workSpace, size_t wkspSize) +{ + return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 0 /* 4 streams */, workSpace, wkspSize, NULL, NULL, 0); +} + +size_t HUF_compress4X_repeat (void* dst, size_t dstSize, + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog, + void* workSpace, size_t wkspSize, + HUF_CElt* hufTable, HUF_repeat* repeat, int preferRepeat) +{ + return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 0 /* 4 streams */, workSpace, wkspSize, hufTable, repeat, preferRepeat); +} + +size_t HUF_compress2 (void* dst, size_t dstSize, + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog) +{ + unsigned workSpace[1024]; + return HUF_compress4X_wksp(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, workSpace, sizeof(workSpace)); +} + +size_t HUF_compress (void* dst, size_t maxDstSize, const void* src, size_t srcSize) +{ + return HUF_compress2(dst, maxDstSize, src, (U32)srcSize, 255, HUF_TABLELOG_DEFAULT); +} diff --git a/contrib/linux-kernel/lib/huf_decompress.c b/contrib/linux-kernel/lib/huf_decompress.c new file mode 100644 index 000000000..ea35c3620 --- /dev/null +++ b/contrib/linux-kernel/lib/huf_decompress.c @@ -0,0 +1,888 @@ +/* ****************************************************************** + Huffman decoder, part of New Generation Entropy library + Copyright (C) 2013-2016, Yann Collet. + + BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php) + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are + met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the + distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + You can contact the author at : + - FSE+HUF source repository : https://github.com/Cyan4973/FiniteStateEntropy + - Public forum : https://groups.google.com/forum/#!forum/lz4c +****************************************************************** */ + +/* ************************************************************** +* Compiler specifics +****************************************************************/ +#ifdef _MSC_VER /* Visual Studio */ +# define FORCE_INLINE static __forceinline +# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ +#else +# if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ +# ifdef __GNUC__ +# define FORCE_INLINE static inline __attribute__((always_inline)) +# else +# define FORCE_INLINE static inline +# endif +# else +# define FORCE_INLINE static +# endif /* __STDC_VERSION__ */ +#endif + + +/* ************************************************************** +* Dependencies +****************************************************************/ +#include /* memcpy, memset */ +#include "bitstream.h" /* BIT_* */ +#include "fse.h" /* header compression */ +#define HUF_STATIC_LINKING_ONLY +#include "huf.h" + + +/* ************************************************************** +* Error Management +****************************************************************/ +#define HUF_STATIC_ASSERT(c) { enum { HUF_static_assert = 1/(int)(!!(c)) }; } /* use only *after* variable declarations */ + + +/*-***************************/ +/* generic DTableDesc */ +/*-***************************/ + +typedef struct { BYTE maxTableLog; BYTE tableType; BYTE tableLog; BYTE reserved; } DTableDesc; + +static DTableDesc HUF_getDTableDesc(const HUF_DTable* table) +{ + DTableDesc dtd; + memcpy(&dtd, table, sizeof(dtd)); + return dtd; +} + + +/*-***************************/ +/* single-symbol decoding */ +/*-***************************/ + +typedef struct { BYTE byte; BYTE nbBits; } HUF_DEltX2; /* single-symbol decoding */ + +size_t HUF_readDTableX2 (HUF_DTable* DTable, const void* src, size_t srcSize) +{ + BYTE huffWeight[HUF_SYMBOLVALUE_MAX + 1]; + U32 rankVal[HUF_TABLELOG_ABSOLUTEMAX + 1]; /* large enough for values from 0 to 16 */ + U32 tableLog = 0; + U32 nbSymbols = 0; + size_t iSize; + void* const dtPtr = DTable + 1; + HUF_DEltX2* const dt = (HUF_DEltX2*)dtPtr; + + HUF_STATIC_ASSERT(sizeof(DTableDesc) == sizeof(HUF_DTable)); + /* memset(huffWeight, 0, sizeof(huffWeight)); */ /* is not necessary, even though some analyzer complain ... */ + + iSize = HUF_readStats(huffWeight, HUF_SYMBOLVALUE_MAX + 1, rankVal, &nbSymbols, &tableLog, src, srcSize); + if (HUF_isError(iSize)) return iSize; + + /* Table header */ + { DTableDesc dtd = HUF_getDTableDesc(DTable); + if (tableLog > (U32)(dtd.maxTableLog+1)) return ERROR(tableLog_tooLarge); /* DTable too small, Huffman tree cannot fit in */ + dtd.tableType = 0; + dtd.tableLog = (BYTE)tableLog; + memcpy(DTable, &dtd, sizeof(dtd)); + } + + /* Calculate starting value for each rank */ + { U32 n, nextRankStart = 0; + for (n=1; n> 1; + U32 u; + HUF_DEltX2 D; + D.byte = (BYTE)n; D.nbBits = (BYTE)(tableLog + 1 - w); + for (u = rankVal[w]; u < rankVal[w] + length; u++) + dt[u] = D; + rankVal[w] += length; + } } + + return iSize; +} + + +static BYTE HUF_decodeSymbolX2(BIT_DStream_t* Dstream, const HUF_DEltX2* dt, const U32 dtLog) +{ + size_t const val = BIT_lookBitsFast(Dstream, dtLog); /* note : dtLog >= 1 */ + BYTE const c = dt[val].byte; + BIT_skipBits(Dstream, dt[val].nbBits); + return c; +} + +#define HUF_DECODE_SYMBOLX2_0(ptr, DStreamPtr) \ + *ptr++ = HUF_decodeSymbolX2(DStreamPtr, dt, dtLog) + +#define HUF_DECODE_SYMBOLX2_1(ptr, DStreamPtr) \ + if (MEM_64bits() || (HUF_TABLELOG_MAX<=12)) \ + HUF_DECODE_SYMBOLX2_0(ptr, DStreamPtr) + +#define HUF_DECODE_SYMBOLX2_2(ptr, DStreamPtr) \ + if (MEM_64bits()) \ + HUF_DECODE_SYMBOLX2_0(ptr, DStreamPtr) + +FORCE_INLINE size_t HUF_decodeStreamX2(BYTE* p, BIT_DStream_t* const bitDPtr, BYTE* const pEnd, const HUF_DEltX2* const dt, const U32 dtLog) +{ + BYTE* const pStart = p; + + /* up to 4 symbols at a time */ + while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) && (p <= pEnd-4)) { + HUF_DECODE_SYMBOLX2_2(p, bitDPtr); + HUF_DECODE_SYMBOLX2_1(p, bitDPtr); + HUF_DECODE_SYMBOLX2_2(p, bitDPtr); + HUF_DECODE_SYMBOLX2_0(p, bitDPtr); + } + + /* closer to the end */ + while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) && (p < pEnd)) + HUF_DECODE_SYMBOLX2_0(p, bitDPtr); + + /* no more data to retrieve from bitstream, hence no need to reload */ + while (p < pEnd) + HUF_DECODE_SYMBOLX2_0(p, bitDPtr); + + return pEnd-pStart; +} + +static size_t HUF_decompress1X2_usingDTable_internal( + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) +{ + BYTE* op = (BYTE*)dst; + BYTE* const oend = op + dstSize; + const void* dtPtr = DTable + 1; + const HUF_DEltX2* const dt = (const HUF_DEltX2*)dtPtr; + BIT_DStream_t bitD; + DTableDesc const dtd = HUF_getDTableDesc(DTable); + U32 const dtLog = dtd.tableLog; + + { size_t const errorCode = BIT_initDStream(&bitD, cSrc, cSrcSize); + if (HUF_isError(errorCode)) return errorCode; } + + HUF_decodeStreamX2(op, &bitD, oend, dt, dtLog); + + /* check */ + if (!BIT_endOfDStream(&bitD)) return ERROR(corruption_detected); + + return dstSize; +} + +size_t HUF_decompress1X2_usingDTable( + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) +{ + DTableDesc dtd = HUF_getDTableDesc(DTable); + if (dtd.tableType != 0) return ERROR(GENERIC); + return HUF_decompress1X2_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); +} + +size_t HUF_decompress1X2_DCtx (HUF_DTable* DCtx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + const BYTE* ip = (const BYTE*) cSrc; + + size_t const hSize = HUF_readDTableX2 (DCtx, cSrc, cSrcSize); + if (HUF_isError(hSize)) return hSize; + if (hSize >= cSrcSize) return ERROR(srcSize_wrong); + ip += hSize; cSrcSize -= hSize; + + return HUF_decompress1X2_usingDTable_internal (dst, dstSize, ip, cSrcSize, DCtx); +} + +size_t HUF_decompress1X2 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + HUF_CREATE_STATIC_DTABLEX2(DTable, HUF_TABLELOG_MAX); + return HUF_decompress1X2_DCtx (DTable, dst, dstSize, cSrc, cSrcSize); +} + + +static size_t HUF_decompress4X2_usingDTable_internal( + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) +{ + /* Check */ + if (cSrcSize < 10) return ERROR(corruption_detected); /* strict minimum : jump table + 1 byte per stream */ + + { const BYTE* const istart = (const BYTE*) cSrc; + BYTE* const ostart = (BYTE*) dst; + BYTE* const oend = ostart + dstSize; + const void* const dtPtr = DTable + 1; + const HUF_DEltX2* const dt = (const HUF_DEltX2*)dtPtr; + + /* Init */ + BIT_DStream_t bitD1; + BIT_DStream_t bitD2; + BIT_DStream_t bitD3; + BIT_DStream_t bitD4; + size_t const length1 = MEM_readLE16(istart); + size_t const length2 = MEM_readLE16(istart+2); + size_t const length3 = MEM_readLE16(istart+4); + size_t const length4 = cSrcSize - (length1 + length2 + length3 + 6); + const BYTE* const istart1 = istart + 6; /* jumpTable */ + const BYTE* const istart2 = istart1 + length1; + const BYTE* const istart3 = istart2 + length2; + const BYTE* const istart4 = istart3 + length3; + const size_t segmentSize = (dstSize+3) / 4; + BYTE* const opStart2 = ostart + segmentSize; + BYTE* const opStart3 = opStart2 + segmentSize; + BYTE* const opStart4 = opStart3 + segmentSize; + BYTE* op1 = ostart; + BYTE* op2 = opStart2; + BYTE* op3 = opStart3; + BYTE* op4 = opStart4; + U32 endSignal; + DTableDesc const dtd = HUF_getDTableDesc(DTable); + U32 const dtLog = dtd.tableLog; + + if (length4 > cSrcSize) return ERROR(corruption_detected); /* overflow */ + { size_t const errorCode = BIT_initDStream(&bitD1, istart1, length1); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD2, istart2, length2); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD3, istart3, length3); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD4, istart4, length4); + if (HUF_isError(errorCode)) return errorCode; } + + /* 16-32 symbols per loop (4-8 symbols per stream) */ + endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); + for ( ; (endSignal==BIT_DStream_unfinished) && (op4<(oend-7)) ; ) { + HUF_DECODE_SYMBOLX2_2(op1, &bitD1); + HUF_DECODE_SYMBOLX2_2(op2, &bitD2); + HUF_DECODE_SYMBOLX2_2(op3, &bitD3); + HUF_DECODE_SYMBOLX2_2(op4, &bitD4); + HUF_DECODE_SYMBOLX2_1(op1, &bitD1); + HUF_DECODE_SYMBOLX2_1(op2, &bitD2); + HUF_DECODE_SYMBOLX2_1(op3, &bitD3); + HUF_DECODE_SYMBOLX2_1(op4, &bitD4); + HUF_DECODE_SYMBOLX2_2(op1, &bitD1); + HUF_DECODE_SYMBOLX2_2(op2, &bitD2); + HUF_DECODE_SYMBOLX2_2(op3, &bitD3); + HUF_DECODE_SYMBOLX2_2(op4, &bitD4); + HUF_DECODE_SYMBOLX2_0(op1, &bitD1); + HUF_DECODE_SYMBOLX2_0(op2, &bitD2); + HUF_DECODE_SYMBOLX2_0(op3, &bitD3); + HUF_DECODE_SYMBOLX2_0(op4, &bitD4); + endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); + } + + /* check corruption */ + if (op1 > opStart2) return ERROR(corruption_detected); + if (op2 > opStart3) return ERROR(corruption_detected); + if (op3 > opStart4) return ERROR(corruption_detected); + /* note : op4 supposed already verified within main loop */ + + /* finish bitStreams one by one */ + HUF_decodeStreamX2(op1, &bitD1, opStart2, dt, dtLog); + HUF_decodeStreamX2(op2, &bitD2, opStart3, dt, dtLog); + HUF_decodeStreamX2(op3, &bitD3, opStart4, dt, dtLog); + HUF_decodeStreamX2(op4, &bitD4, oend, dt, dtLog); + + /* check */ + endSignal = BIT_endOfDStream(&bitD1) & BIT_endOfDStream(&bitD2) & BIT_endOfDStream(&bitD3) & BIT_endOfDStream(&bitD4); + if (!endSignal) return ERROR(corruption_detected); + + /* decoded size */ + return dstSize; + } +} + + +size_t HUF_decompress4X2_usingDTable( + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) +{ + DTableDesc dtd = HUF_getDTableDesc(DTable); + if (dtd.tableType != 0) return ERROR(GENERIC); + return HUF_decompress4X2_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); +} + + +size_t HUF_decompress4X2_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + const BYTE* ip = (const BYTE*) cSrc; + + size_t const hSize = HUF_readDTableX2 (dctx, cSrc, cSrcSize); + if (HUF_isError(hSize)) return hSize; + if (hSize >= cSrcSize) return ERROR(srcSize_wrong); + ip += hSize; cSrcSize -= hSize; + + return HUF_decompress4X2_usingDTable_internal (dst, dstSize, ip, cSrcSize, dctx); +} + +size_t HUF_decompress4X2 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + HUF_CREATE_STATIC_DTABLEX2(DTable, HUF_TABLELOG_MAX); + return HUF_decompress4X2_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); +} + + +/* *************************/ +/* double-symbols decoding */ +/* *************************/ +typedef struct { U16 sequence; BYTE nbBits; BYTE length; } HUF_DEltX4; /* double-symbols decoding */ + +typedef struct { BYTE symbol; BYTE weight; } sortedSymbol_t; + +/* HUF_fillDTableX4Level2() : + * `rankValOrigin` must be a table of at least (HUF_TABLELOG_MAX + 1) U32 */ +static void HUF_fillDTableX4Level2(HUF_DEltX4* DTable, U32 sizeLog, const U32 consumed, + const U32* rankValOrigin, const int minWeight, + const sortedSymbol_t* sortedSymbols, const U32 sortedListSize, + U32 nbBitsBaseline, U16 baseSeq) +{ + HUF_DEltX4 DElt; + U32 rankVal[HUF_TABLELOG_MAX + 1]; + + /* get pre-calculated rankVal */ + memcpy(rankVal, rankValOrigin, sizeof(rankVal)); + + /* fill skipped values */ + if (minWeight>1) { + U32 i, skipSize = rankVal[minWeight]; + MEM_writeLE16(&(DElt.sequence), baseSeq); + DElt.nbBits = (BYTE)(consumed); + DElt.length = 1; + for (i = 0; i < skipSize; i++) + DTable[i] = DElt; + } + + /* fill DTable */ + { U32 s; for (s=0; s= 1 */ + + rankVal[weight] += length; + } } +} + +typedef U32 rankVal_t[HUF_TABLELOG_MAX][HUF_TABLELOG_MAX + 1]; + +static void HUF_fillDTableX4(HUF_DEltX4* DTable, const U32 targetLog, + const sortedSymbol_t* sortedList, const U32 sortedListSize, + const U32* rankStart, rankVal_t rankValOrigin, const U32 maxWeight, + const U32 nbBitsBaseline) +{ + U32 rankVal[HUF_TABLELOG_MAX + 1]; + const int scaleLog = nbBitsBaseline - targetLog; /* note : targetLog >= srcLog, hence scaleLog <= 1 */ + const U32 minBits = nbBitsBaseline - maxWeight; + U32 s; + + memcpy(rankVal, rankValOrigin, sizeof(rankVal)); + + /* fill DTable */ + for (s=0; s= minBits) { /* enough room for a second symbol */ + U32 sortedRank; + int minWeight = nbBits + scaleLog; + if (minWeight < 1) minWeight = 1; + sortedRank = rankStart[minWeight]; + HUF_fillDTableX4Level2(DTable+start, targetLog-nbBits, nbBits, + rankValOrigin[nbBits], minWeight, + sortedList+sortedRank, sortedListSize-sortedRank, + nbBitsBaseline, symbol); + } else { + HUF_DEltX4 DElt; + MEM_writeLE16(&(DElt.sequence), symbol); + DElt.nbBits = (BYTE)(nbBits); + DElt.length = 1; + { U32 const end = start + length; + U32 u; + for (u = start; u < end; u++) DTable[u] = DElt; + } } + rankVal[weight] += length; + } +} + +size_t HUF_readDTableX4 (HUF_DTable* DTable, const void* src, size_t srcSize) +{ + BYTE weightList[HUF_SYMBOLVALUE_MAX + 1]; + sortedSymbol_t sortedSymbol[HUF_SYMBOLVALUE_MAX + 1]; + U32 rankStats[HUF_TABLELOG_MAX + 1] = { 0 }; + U32 rankStart0[HUF_TABLELOG_MAX + 2] = { 0 }; + U32* const rankStart = rankStart0+1; + rankVal_t rankVal; + U32 tableLog, maxW, sizeOfSort, nbSymbols; + DTableDesc dtd = HUF_getDTableDesc(DTable); + U32 const maxTableLog = dtd.maxTableLog; + size_t iSize; + void* dtPtr = DTable+1; /* force compiler to avoid strict-aliasing */ + HUF_DEltX4* const dt = (HUF_DEltX4*)dtPtr; + + HUF_STATIC_ASSERT(sizeof(HUF_DEltX4) == sizeof(HUF_DTable)); /* if compiler fails here, assertion is wrong */ + if (maxTableLog > HUF_TABLELOG_MAX) return ERROR(tableLog_tooLarge); + /* memset(weightList, 0, sizeof(weightList)); */ /* is not necessary, even though some analyzer complain ... */ + + iSize = HUF_readStats(weightList, HUF_SYMBOLVALUE_MAX + 1, rankStats, &nbSymbols, &tableLog, src, srcSize); + if (HUF_isError(iSize)) return iSize; + + /* check result */ + if (tableLog > maxTableLog) return ERROR(tableLog_tooLarge); /* DTable can't fit code depth */ + + /* find maxWeight */ + for (maxW = tableLog; rankStats[maxW]==0; maxW--) {} /* necessarily finds a solution before 0 */ + + /* Get start index of each weight */ + { U32 w, nextRankStart = 0; + for (w=1; w> consumed; + } } } } + + HUF_fillDTableX4(dt, maxTableLog, + sortedSymbol, sizeOfSort, + rankStart0, rankVal, maxW, + tableLog+1); + + dtd.tableLog = (BYTE)maxTableLog; + dtd.tableType = 1; + memcpy(DTable, &dtd, sizeof(dtd)); + return iSize; +} + + +static U32 HUF_decodeSymbolX4(void* op, BIT_DStream_t* DStream, const HUF_DEltX4* dt, const U32 dtLog) +{ + size_t const val = BIT_lookBitsFast(DStream, dtLog); /* note : dtLog >= 1 */ + memcpy(op, dt+val, 2); + BIT_skipBits(DStream, dt[val].nbBits); + return dt[val].length; +} + +static U32 HUF_decodeLastSymbolX4(void* op, BIT_DStream_t* DStream, const HUF_DEltX4* dt, const U32 dtLog) +{ + size_t const val = BIT_lookBitsFast(DStream, dtLog); /* note : dtLog >= 1 */ + memcpy(op, dt+val, 1); + if (dt[val].length==1) BIT_skipBits(DStream, dt[val].nbBits); + else { + if (DStream->bitsConsumed < (sizeof(DStream->bitContainer)*8)) { + BIT_skipBits(DStream, dt[val].nbBits); + if (DStream->bitsConsumed > (sizeof(DStream->bitContainer)*8)) + DStream->bitsConsumed = (sizeof(DStream->bitContainer)*8); /* ugly hack; works only because it's the last symbol. Note : can't easily extract nbBits from just this symbol */ + } } + return 1; +} + + +#define HUF_DECODE_SYMBOLX4_0(ptr, DStreamPtr) \ + ptr += HUF_decodeSymbolX4(ptr, DStreamPtr, dt, dtLog) + +#define HUF_DECODE_SYMBOLX4_1(ptr, DStreamPtr) \ + if (MEM_64bits() || (HUF_TABLELOG_MAX<=12)) \ + ptr += HUF_decodeSymbolX4(ptr, DStreamPtr, dt, dtLog) + +#define HUF_DECODE_SYMBOLX4_2(ptr, DStreamPtr) \ + if (MEM_64bits()) \ + ptr += HUF_decodeSymbolX4(ptr, DStreamPtr, dt, dtLog) + +FORCE_INLINE size_t HUF_decodeStreamX4(BYTE* p, BIT_DStream_t* bitDPtr, BYTE* const pEnd, const HUF_DEltX4* const dt, const U32 dtLog) +{ + BYTE* const pStart = p; + + /* up to 8 symbols at a time */ + while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) & (p < pEnd-(sizeof(bitDPtr->bitContainer)-1))) { + HUF_DECODE_SYMBOLX4_2(p, bitDPtr); + HUF_DECODE_SYMBOLX4_1(p, bitDPtr); + HUF_DECODE_SYMBOLX4_2(p, bitDPtr); + HUF_DECODE_SYMBOLX4_0(p, bitDPtr); + } + + /* closer to end : up to 2 symbols at a time */ + while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) & (p <= pEnd-2)) + HUF_DECODE_SYMBOLX4_0(p, bitDPtr); + + while (p <= pEnd-2) + HUF_DECODE_SYMBOLX4_0(p, bitDPtr); /* no need to reload : reached the end of DStream */ + + if (p < pEnd) + p += HUF_decodeLastSymbolX4(p, bitDPtr, dt, dtLog); + + return p-pStart; +} + + +static size_t HUF_decompress1X4_usingDTable_internal( + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) +{ + BIT_DStream_t bitD; + + /* Init */ + { size_t const errorCode = BIT_initDStream(&bitD, cSrc, cSrcSize); + if (HUF_isError(errorCode)) return errorCode; + } + + /* decode */ + { BYTE* const ostart = (BYTE*) dst; + BYTE* const oend = ostart + dstSize; + const void* const dtPtr = DTable+1; /* force compiler to not use strict-aliasing */ + const HUF_DEltX4* const dt = (const HUF_DEltX4*)dtPtr; + DTableDesc const dtd = HUF_getDTableDesc(DTable); + HUF_decodeStreamX4(ostart, &bitD, oend, dt, dtd.tableLog); + } + + /* check */ + if (!BIT_endOfDStream(&bitD)) return ERROR(corruption_detected); + + /* decoded size */ + return dstSize; +} + +size_t HUF_decompress1X4_usingDTable( + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) +{ + DTableDesc dtd = HUF_getDTableDesc(DTable); + if (dtd.tableType != 1) return ERROR(GENERIC); + return HUF_decompress1X4_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); +} + +size_t HUF_decompress1X4_DCtx (HUF_DTable* DCtx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + const BYTE* ip = (const BYTE*) cSrc; + + size_t const hSize = HUF_readDTableX4 (DCtx, cSrc, cSrcSize); + if (HUF_isError(hSize)) return hSize; + if (hSize >= cSrcSize) return ERROR(srcSize_wrong); + ip += hSize; cSrcSize -= hSize; + + return HUF_decompress1X4_usingDTable_internal (dst, dstSize, ip, cSrcSize, DCtx); +} + +size_t HUF_decompress1X4 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + HUF_CREATE_STATIC_DTABLEX4(DTable, HUF_TABLELOG_MAX); + return HUF_decompress1X4_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); +} + +static size_t HUF_decompress4X4_usingDTable_internal( + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) +{ + if (cSrcSize < 10) return ERROR(corruption_detected); /* strict minimum : jump table + 1 byte per stream */ + + { const BYTE* const istart = (const BYTE*) cSrc; + BYTE* const ostart = (BYTE*) dst; + BYTE* const oend = ostart + dstSize; + const void* const dtPtr = DTable+1; + const HUF_DEltX4* const dt = (const HUF_DEltX4*)dtPtr; + + /* Init */ + BIT_DStream_t bitD1; + BIT_DStream_t bitD2; + BIT_DStream_t bitD3; + BIT_DStream_t bitD4; + size_t const length1 = MEM_readLE16(istart); + size_t const length2 = MEM_readLE16(istart+2); + size_t const length3 = MEM_readLE16(istart+4); + size_t const length4 = cSrcSize - (length1 + length2 + length3 + 6); + const BYTE* const istart1 = istart + 6; /* jumpTable */ + const BYTE* const istart2 = istart1 + length1; + const BYTE* const istart3 = istart2 + length2; + const BYTE* const istart4 = istart3 + length3; + size_t const segmentSize = (dstSize+3) / 4; + BYTE* const opStart2 = ostart + segmentSize; + BYTE* const opStart3 = opStart2 + segmentSize; + BYTE* const opStart4 = opStart3 + segmentSize; + BYTE* op1 = ostart; + BYTE* op2 = opStart2; + BYTE* op3 = opStart3; + BYTE* op4 = opStart4; + U32 endSignal; + DTableDesc const dtd = HUF_getDTableDesc(DTable); + U32 const dtLog = dtd.tableLog; + + if (length4 > cSrcSize) return ERROR(corruption_detected); /* overflow */ + { size_t const errorCode = BIT_initDStream(&bitD1, istart1, length1); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD2, istart2, length2); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD3, istart3, length3); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD4, istart4, length4); + if (HUF_isError(errorCode)) return errorCode; } + + /* 16-32 symbols per loop (4-8 symbols per stream) */ + endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); + for ( ; (endSignal==BIT_DStream_unfinished) & (op4<(oend-(sizeof(bitD4.bitContainer)-1))) ; ) { + HUF_DECODE_SYMBOLX4_2(op1, &bitD1); + HUF_DECODE_SYMBOLX4_2(op2, &bitD2); + HUF_DECODE_SYMBOLX4_2(op3, &bitD3); + HUF_DECODE_SYMBOLX4_2(op4, &bitD4); + HUF_DECODE_SYMBOLX4_1(op1, &bitD1); + HUF_DECODE_SYMBOLX4_1(op2, &bitD2); + HUF_DECODE_SYMBOLX4_1(op3, &bitD3); + HUF_DECODE_SYMBOLX4_1(op4, &bitD4); + HUF_DECODE_SYMBOLX4_2(op1, &bitD1); + HUF_DECODE_SYMBOLX4_2(op2, &bitD2); + HUF_DECODE_SYMBOLX4_2(op3, &bitD3); + HUF_DECODE_SYMBOLX4_2(op4, &bitD4); + HUF_DECODE_SYMBOLX4_0(op1, &bitD1); + HUF_DECODE_SYMBOLX4_0(op2, &bitD2); + HUF_DECODE_SYMBOLX4_0(op3, &bitD3); + HUF_DECODE_SYMBOLX4_0(op4, &bitD4); + + endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); + } + + /* check corruption */ + if (op1 > opStart2) return ERROR(corruption_detected); + if (op2 > opStart3) return ERROR(corruption_detected); + if (op3 > opStart4) return ERROR(corruption_detected); + /* note : op4 already verified within main loop */ + + /* finish bitStreams one by one */ + HUF_decodeStreamX4(op1, &bitD1, opStart2, dt, dtLog); + HUF_decodeStreamX4(op2, &bitD2, opStart3, dt, dtLog); + HUF_decodeStreamX4(op3, &bitD3, opStart4, dt, dtLog); + HUF_decodeStreamX4(op4, &bitD4, oend, dt, dtLog); + + /* check */ + { U32 const endCheck = BIT_endOfDStream(&bitD1) & BIT_endOfDStream(&bitD2) & BIT_endOfDStream(&bitD3) & BIT_endOfDStream(&bitD4); + if (!endCheck) return ERROR(corruption_detected); } + + /* decoded size */ + return dstSize; + } +} + + +size_t HUF_decompress4X4_usingDTable( + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) +{ + DTableDesc dtd = HUF_getDTableDesc(DTable); + if (dtd.tableType != 1) return ERROR(GENERIC); + return HUF_decompress4X4_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); +} + + +size_t HUF_decompress4X4_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + const BYTE* ip = (const BYTE*) cSrc; + + size_t hSize = HUF_readDTableX4 (dctx, cSrc, cSrcSize); + if (HUF_isError(hSize)) return hSize; + if (hSize >= cSrcSize) return ERROR(srcSize_wrong); + ip += hSize; cSrcSize -= hSize; + + return HUF_decompress4X4_usingDTable_internal(dst, dstSize, ip, cSrcSize, dctx); +} + +size_t HUF_decompress4X4 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + HUF_CREATE_STATIC_DTABLEX4(DTable, HUF_TABLELOG_MAX); + return HUF_decompress4X4_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); +} + + +/* ********************************/ +/* Generic decompression selector */ +/* ********************************/ + +size_t HUF_decompress1X_usingDTable(void* dst, size_t maxDstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) +{ + DTableDesc const dtd = HUF_getDTableDesc(DTable); + return dtd.tableType ? HUF_decompress1X4_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable) : + HUF_decompress1X2_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable); +} + +size_t HUF_decompress4X_usingDTable(void* dst, size_t maxDstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) +{ + DTableDesc const dtd = HUF_getDTableDesc(DTable); + return dtd.tableType ? HUF_decompress4X4_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable) : + HUF_decompress4X2_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable); +} + + +typedef struct { U32 tableTime; U32 decode256Time; } algo_time_t; +static const algo_time_t algoTime[16 /* Quantization */][3 /* single, double, quad */] = +{ + /* single, double, quad */ + {{0,0}, {1,1}, {2,2}}, /* Q==0 : impossible */ + {{0,0}, {1,1}, {2,2}}, /* Q==1 : impossible */ + {{ 38,130}, {1313, 74}, {2151, 38}}, /* Q == 2 : 12-18% */ + {{ 448,128}, {1353, 74}, {2238, 41}}, /* Q == 3 : 18-25% */ + {{ 556,128}, {1353, 74}, {2238, 47}}, /* Q == 4 : 25-32% */ + {{ 714,128}, {1418, 74}, {2436, 53}}, /* Q == 5 : 32-38% */ + {{ 883,128}, {1437, 74}, {2464, 61}}, /* Q == 6 : 38-44% */ + {{ 897,128}, {1515, 75}, {2622, 68}}, /* Q == 7 : 44-50% */ + {{ 926,128}, {1613, 75}, {2730, 75}}, /* Q == 8 : 50-56% */ + {{ 947,128}, {1729, 77}, {3359, 77}}, /* Q == 9 : 56-62% */ + {{1107,128}, {2083, 81}, {4006, 84}}, /* Q ==10 : 62-69% */ + {{1177,128}, {2379, 87}, {4785, 88}}, /* Q ==11 : 69-75% */ + {{1242,128}, {2415, 93}, {5155, 84}}, /* Q ==12 : 75-81% */ + {{1349,128}, {2644,106}, {5260,106}}, /* Q ==13 : 81-87% */ + {{1455,128}, {2422,124}, {4174,124}}, /* Q ==14 : 87-93% */ + {{ 722,128}, {1891,145}, {1936,146}}, /* Q ==15 : 93-99% */ +}; + +/** HUF_selectDecoder() : +* Tells which decoder is likely to decode faster, +* based on a set of pre-determined metrics. +* @return : 0==HUF_decompress4X2, 1==HUF_decompress4X4 . +* Assumption : 0 < cSrcSize < dstSize <= 128 KB */ +U32 HUF_selectDecoder (size_t dstSize, size_t cSrcSize) +{ + /* decoder timing evaluation */ + U32 const Q = (U32)(cSrcSize * 16 / dstSize); /* Q < 16 since dstSize > cSrcSize */ + U32 const D256 = (U32)(dstSize >> 8); + U32 const DTime0 = algoTime[Q][0].tableTime + (algoTime[Q][0].decode256Time * D256); + U32 DTime1 = algoTime[Q][1].tableTime + (algoTime[Q][1].decode256Time * D256); + DTime1 += DTime1 >> 3; /* advantage to algorithm using less memory, for cache eviction */ + + return DTime1 < DTime0; +} + + +typedef size_t (*decompressionAlgo)(void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); + +size_t HUF_decompress (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + static const decompressionAlgo decompress[2] = { HUF_decompress4X2, HUF_decompress4X4 }; + + /* validation checks */ + if (dstSize == 0) return ERROR(dstSize_tooSmall); + if (cSrcSize > dstSize) return ERROR(corruption_detected); /* invalid */ + if (cSrcSize == dstSize) { memcpy(dst, cSrc, dstSize); return dstSize; } /* not compressed */ + if (cSrcSize == 1) { memset(dst, *(const BYTE*)cSrc, dstSize); return dstSize; } /* RLE */ + + { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); + return decompress[algoNb](dst, dstSize, cSrc, cSrcSize); + } +} + +size_t HUF_decompress4X_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + /* validation checks */ + if (dstSize == 0) return ERROR(dstSize_tooSmall); + if (cSrcSize > dstSize) return ERROR(corruption_detected); /* invalid */ + if (cSrcSize == dstSize) { memcpy(dst, cSrc, dstSize); return dstSize; } /* not compressed */ + if (cSrcSize == 1) { memset(dst, *(const BYTE*)cSrc, dstSize); return dstSize; } /* RLE */ + + { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); + return algoNb ? HUF_decompress4X4_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) : + HUF_decompress4X2_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) ; + } +} + +size_t HUF_decompress4X_hufOnly (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + /* validation checks */ + if (dstSize == 0) return ERROR(dstSize_tooSmall); + if ((cSrcSize >= dstSize) || (cSrcSize <= 1)) return ERROR(corruption_detected); /* invalid */ + + { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); + return algoNb ? HUF_decompress4X4_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) : + HUF_decompress4X2_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) ; + } +} + +size_t HUF_decompress1X_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) +{ + /* validation checks */ + if (dstSize == 0) return ERROR(dstSize_tooSmall); + if (cSrcSize > dstSize) return ERROR(corruption_detected); /* invalid */ + if (cSrcSize == dstSize) { memcpy(dst, cSrc, dstSize); return dstSize; } /* not compressed */ + if (cSrcSize == 1) { memset(dst, *(const BYTE*)cSrc, dstSize); return dstSize; } /* RLE */ + + { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); + return algoNb ? HUF_decompress1X4_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) : + HUF_decompress1X2_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) ; + } +} diff --git a/contrib/linux-kernel/lib/mem.h b/contrib/linux-kernel/lib/mem.h new file mode 100644 index 000000000..f049d181b --- /dev/null +++ b/contrib/linux-kernel/lib/mem.h @@ -0,0 +1,374 @@ +/** + * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + +#ifndef MEM_H_MODULE +#define MEM_H_MODULE + +#if defined (__cplusplus) +extern "C" { +#endif + +/*-**************************************** +* Dependencies +******************************************/ +#include /* size_t, ptrdiff_t */ +#include /* memcpy */ + + +/*-**************************************** +* Compiler specifics +******************************************/ +#if defined(_MSC_VER) /* Visual Studio */ +# include /* _byteswap_ulong */ +# include /* _byteswap_* */ +#endif +#if defined(__GNUC__) +# define MEM_STATIC static __inline __attribute__((unused)) +#elif defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) +# define MEM_STATIC static inline +#elif defined(_MSC_VER) +# define MEM_STATIC static __inline +#else +# define MEM_STATIC static /* this version may generate warnings for unused static functions; disable the relevant warning */ +#endif + +/* code only tested on 32 and 64 bits systems */ +#define MEM_STATIC_ASSERT(c) { enum { MEM_static_assert = 1/(int)(!!(c)) }; } +MEM_STATIC void MEM_check(void) { MEM_STATIC_ASSERT((sizeof(size_t)==4) || (sizeof(size_t)==8)); } + + +/*-************************************************************** +* Basic Types +*****************************************************************/ +#if !defined (__VMS) && (defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) ) +# include + typedef uint8_t BYTE; + typedef uint16_t U16; + typedef int16_t S16; + typedef uint32_t U32; + typedef int32_t S32; + typedef uint64_t U64; + typedef int64_t S64; + typedef intptr_t iPtrDiff; + typedef uintptr_t uPtrDiff; +#else + typedef unsigned char BYTE; + typedef unsigned short U16; + typedef signed short S16; + typedef unsigned int U32; + typedef signed int S32; + typedef unsigned long long U64; + typedef signed long long S64; + typedef ptrdiff_t iPtrDiff; + typedef size_t uPtrDiff; +#endif + + +/*-************************************************************** +* Memory I/O +*****************************************************************/ +/* MEM_FORCE_MEMORY_ACCESS : + * By default, access to unaligned memory is controlled by `memcpy()`, which is safe and portable. + * Unfortunately, on some target/compiler combinations, the generated assembly is sub-optimal. + * The below switch allow to select different access method for improved performance. + * Method 0 (default) : use `memcpy()`. Safe and portable. + * Method 1 : `__packed` statement. It depends on compiler extension (i.e., not portable). + * This method is safe if your compiler supports it, and *generally* as fast or faster than `memcpy`. + * Method 2 : direct access. This method is portable but violate C standard. + * It can generate buggy code on targets depending on alignment. + * In some circumstances, it's the only known way to get the most performance (i.e. GCC + ARMv6) + * See http://fastcompression.blogspot.fr/2015/08/accessing-unaligned-memory.html for details. + * Prefer these methods in priority order (0 > 1 > 2) + */ +#ifndef MEM_FORCE_MEMORY_ACCESS /* can be defined externally, on command line for example */ +# if defined(__GNUC__) && ( defined(__ARM_ARCH_6__) || defined(__ARM_ARCH_6J__) || defined(__ARM_ARCH_6K__) || defined(__ARM_ARCH_6Z__) || defined(__ARM_ARCH_6ZK__) || defined(__ARM_ARCH_6T2__) ) +# define MEM_FORCE_MEMORY_ACCESS 2 +# elif defined(__INTEL_COMPILER) /*|| defined(_MSC_VER)*/ || \ + (defined(__GNUC__) && ( defined(__ARM_ARCH_7__) || defined(__ARM_ARCH_7A__) || defined(__ARM_ARCH_7R__) || defined(__ARM_ARCH_7M__) || defined(__ARM_ARCH_7S__) )) +# define MEM_FORCE_MEMORY_ACCESS 1 +# endif +#endif + +MEM_STATIC unsigned MEM_32bits(void) { return sizeof(size_t)==4; } +MEM_STATIC unsigned MEM_64bits(void) { return sizeof(size_t)==8; } + +MEM_STATIC unsigned MEM_isLittleEndian(void) +{ + const union { U32 u; BYTE c[4]; } one = { 1 }; /* don't use static : performance detrimental */ + return one.c[0]; +} + +#if defined(MEM_FORCE_MEMORY_ACCESS) && (MEM_FORCE_MEMORY_ACCESS==2) + +/* violates C standard, by lying on structure alignment. +Only use if no other choice to achieve best performance on target platform */ +MEM_STATIC U16 MEM_read16(const void* memPtr) { return *(const U16*) memPtr; } +MEM_STATIC U32 MEM_read32(const void* memPtr) { return *(const U32*) memPtr; } +MEM_STATIC U64 MEM_read64(const void* memPtr) { return *(const U64*) memPtr; } +MEM_STATIC U64 MEM_readST(const void* memPtr) { return *(const size_t*) memPtr; } + +MEM_STATIC void MEM_write16(void* memPtr, U16 value) { *(U16*)memPtr = value; } +MEM_STATIC void MEM_write32(void* memPtr, U32 value) { *(U32*)memPtr = value; } +MEM_STATIC void MEM_write64(void* memPtr, U64 value) { *(U64*)memPtr = value; } + +#elif defined(MEM_FORCE_MEMORY_ACCESS) && (MEM_FORCE_MEMORY_ACCESS==1) + +/* __pack instructions are safer, but compiler specific, hence potentially problematic for some compilers */ +/* currently only defined for gcc and icc */ +#if defined(_MSC_VER) || (defined(__INTEL_COMPILER) && defined(WIN32)) + __pragma( pack(push, 1) ) + typedef union { U16 u16; U32 u32; U64 u64; size_t st; } unalign; + __pragma( pack(pop) ) +#else + typedef union { U16 u16; U32 u32; U64 u64; size_t st; } __attribute__((packed)) unalign; +#endif + +MEM_STATIC U16 MEM_read16(const void* ptr) { return ((const unalign*)ptr)->u16; } +MEM_STATIC U32 MEM_read32(const void* ptr) { return ((const unalign*)ptr)->u32; } +MEM_STATIC U64 MEM_read64(const void* ptr) { return ((const unalign*)ptr)->u64; } +MEM_STATIC U64 MEM_readST(const void* ptr) { return ((const unalign*)ptr)->st; } + +MEM_STATIC void MEM_write16(void* memPtr, U16 value) { ((unalign*)memPtr)->u16 = value; } +MEM_STATIC void MEM_write32(void* memPtr, U32 value) { ((unalign*)memPtr)->u32 = value; } +MEM_STATIC void MEM_write64(void* memPtr, U64 value) { ((unalign*)memPtr)->u64 = value; } + +#else + +/* default method, safe and standard. + can sometimes prove slower */ + +MEM_STATIC U16 MEM_read16(const void* memPtr) +{ + U16 val; memcpy(&val, memPtr, sizeof(val)); return val; +} + +MEM_STATIC U32 MEM_read32(const void* memPtr) +{ + U32 val; memcpy(&val, memPtr, sizeof(val)); return val; +} + +MEM_STATIC U64 MEM_read64(const void* memPtr) +{ + U64 val; memcpy(&val, memPtr, sizeof(val)); return val; +} + +MEM_STATIC size_t MEM_readST(const void* memPtr) +{ + size_t val; memcpy(&val, memPtr, sizeof(val)); return val; +} + +MEM_STATIC void MEM_write16(void* memPtr, U16 value) +{ + memcpy(memPtr, &value, sizeof(value)); +} + +MEM_STATIC void MEM_write32(void* memPtr, U32 value) +{ + memcpy(memPtr, &value, sizeof(value)); +} + +MEM_STATIC void MEM_write64(void* memPtr, U64 value) +{ + memcpy(memPtr, &value, sizeof(value)); +} + +#endif /* MEM_FORCE_MEMORY_ACCESS */ + +MEM_STATIC U32 MEM_swap32(U32 in) +{ +#if defined(_MSC_VER) /* Visual Studio */ + return _byteswap_ulong(in); +#elif defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403) + return __builtin_bswap32(in); +#else + return ((in << 24) & 0xff000000 ) | + ((in << 8) & 0x00ff0000 ) | + ((in >> 8) & 0x0000ff00 ) | + ((in >> 24) & 0x000000ff ); +#endif +} + +MEM_STATIC U64 MEM_swap64(U64 in) +{ +#if defined(_MSC_VER) /* Visual Studio */ + return _byteswap_uint64(in); +#elif defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403) + return __builtin_bswap64(in); +#else + return ((in << 56) & 0xff00000000000000ULL) | + ((in << 40) & 0x00ff000000000000ULL) | + ((in << 24) & 0x0000ff0000000000ULL) | + ((in << 8) & 0x000000ff00000000ULL) | + ((in >> 8) & 0x00000000ff000000ULL) | + ((in >> 24) & 0x0000000000ff0000ULL) | + ((in >> 40) & 0x000000000000ff00ULL) | + ((in >> 56) & 0x00000000000000ffULL); +#endif +} + +MEM_STATIC size_t MEM_swapST(size_t in) +{ + if (MEM_32bits()) + return (size_t)MEM_swap32((U32)in); + else + return (size_t)MEM_swap64((U64)in); +} + +/*=== Little endian r/w ===*/ + +MEM_STATIC U16 MEM_readLE16(const void* memPtr) +{ + if (MEM_isLittleEndian()) + return MEM_read16(memPtr); + else { + const BYTE* p = (const BYTE*)memPtr; + return (U16)(p[0] + (p[1]<<8)); + } +} + +MEM_STATIC void MEM_writeLE16(void* memPtr, U16 val) +{ + if (MEM_isLittleEndian()) { + MEM_write16(memPtr, val); + } else { + BYTE* p = (BYTE*)memPtr; + p[0] = (BYTE)val; + p[1] = (BYTE)(val>>8); + } +} + +MEM_STATIC U32 MEM_readLE24(const void* memPtr) +{ + return MEM_readLE16(memPtr) + (((const BYTE*)memPtr)[2] << 16); +} + +MEM_STATIC void MEM_writeLE24(void* memPtr, U32 val) +{ + MEM_writeLE16(memPtr, (U16)val); + ((BYTE*)memPtr)[2] = (BYTE)(val>>16); +} + +MEM_STATIC U32 MEM_readLE32(const void* memPtr) +{ + if (MEM_isLittleEndian()) + return MEM_read32(memPtr); + else + return MEM_swap32(MEM_read32(memPtr)); +} + +MEM_STATIC void MEM_writeLE32(void* memPtr, U32 val32) +{ + if (MEM_isLittleEndian()) + MEM_write32(memPtr, val32); + else + MEM_write32(memPtr, MEM_swap32(val32)); +} + +MEM_STATIC U64 MEM_readLE64(const void* memPtr) +{ + if (MEM_isLittleEndian()) + return MEM_read64(memPtr); + else + return MEM_swap64(MEM_read64(memPtr)); +} + +MEM_STATIC void MEM_writeLE64(void* memPtr, U64 val64) +{ + if (MEM_isLittleEndian()) + MEM_write64(memPtr, val64); + else + MEM_write64(memPtr, MEM_swap64(val64)); +} + +MEM_STATIC size_t MEM_readLEST(const void* memPtr) +{ + if (MEM_32bits()) + return (size_t)MEM_readLE32(memPtr); + else + return (size_t)MEM_readLE64(memPtr); +} + +MEM_STATIC void MEM_writeLEST(void* memPtr, size_t val) +{ + if (MEM_32bits()) + MEM_writeLE32(memPtr, (U32)val); + else + MEM_writeLE64(memPtr, (U64)val); +} + +/*=== Big endian r/w ===*/ + +MEM_STATIC U32 MEM_readBE32(const void* memPtr) +{ + if (MEM_isLittleEndian()) + return MEM_swap32(MEM_read32(memPtr)); + else + return MEM_read32(memPtr); +} + +MEM_STATIC void MEM_writeBE32(void* memPtr, U32 val32) +{ + if (MEM_isLittleEndian()) + MEM_write32(memPtr, MEM_swap32(val32)); + else + MEM_write32(memPtr, val32); +} + +MEM_STATIC U64 MEM_readBE64(const void* memPtr) +{ + if (MEM_isLittleEndian()) + return MEM_swap64(MEM_read64(memPtr)); + else + return MEM_read64(memPtr); +} + +MEM_STATIC void MEM_writeBE64(void* memPtr, U64 val64) +{ + if (MEM_isLittleEndian()) + MEM_write64(memPtr, MEM_swap64(val64)); + else + MEM_write64(memPtr, val64); +} + +MEM_STATIC size_t MEM_readBEST(const void* memPtr) +{ + if (MEM_32bits()) + return (size_t)MEM_readBE32(memPtr); + else + return (size_t)MEM_readBE64(memPtr); +} + +MEM_STATIC void MEM_writeBEST(void* memPtr, size_t val) +{ + if (MEM_32bits()) + MEM_writeBE32(memPtr, (U32)val); + else + MEM_writeBE64(memPtr, (U64)val); +} + + +/* function safe only for comparisons */ +MEM_STATIC U32 MEM_readMINMATCH(const void* memPtr, U32 length) +{ + switch (length) + { + default : + case 4 : return MEM_read32(memPtr); + case 3 : if (MEM_isLittleEndian()) + return MEM_read32(memPtr)<<8; + else + return MEM_read32(memPtr)>>8; + } +} + +#if defined (__cplusplus) +} +#endif + +#endif /* MEM_H_MODULE */ diff --git a/contrib/linux-kernel/lib/xxhash.c b/contrib/linux-kernel/lib/xxhash.c new file mode 100644 index 000000000..eb44222c5 --- /dev/null +++ b/contrib/linux-kernel/lib/xxhash.c @@ -0,0 +1,869 @@ +/* +* xxHash - Fast Hash algorithm +* Copyright (C) 2012-2016, Yann Collet +* +* BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php) +* +* Redistribution and use in source and binary forms, with or without +* modification, are permitted provided that the following conditions are +* met: +* +* * Redistributions of source code must retain the above copyright +* notice, this list of conditions and the following disclaimer. +* * Redistributions in binary form must reproduce the above +* copyright notice, this list of conditions and the following disclaimer +* in the documentation and/or other materials provided with the +* distribution. +* +* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +* +* You can contact the author at : +* - xxHash homepage: http://www.xxhash.com +* - xxHash source repository : https://github.com/Cyan4973/xxHash +*/ + + +/* ************************************* +* Tuning parameters +***************************************/ +/*!XXH_FORCE_MEMORY_ACCESS : + * By default, access to unaligned memory is controlled by `memcpy()`, which is safe and portable. + * Unfortunately, on some target/compiler combinations, the generated assembly is sub-optimal. + * The below switch allow to select different access method for improved performance. + * Method 0 (default) : use `memcpy()`. Safe and portable. + * Method 1 : `__packed` statement. It depends on compiler extension (ie, not portable). + * This method is safe if your compiler supports it, and *generally* as fast or faster than `memcpy`. + * Method 2 : direct access. This method doesn't depend on compiler but violate C standard. + * It can generate buggy code on targets which do not support unaligned memory accesses. + * But in some circumstances, it's the only known way to get the most performance (ie GCC + ARMv6) + * See http://stackoverflow.com/a/32095106/646947 for details. + * Prefer these methods in priority order (0 > 1 > 2) + */ +#ifndef XXH_FORCE_MEMORY_ACCESS /* can be defined externally, on command line for example */ +# if defined(__GNUC__) && ( defined(__ARM_ARCH_6__) || defined(__ARM_ARCH_6J__) || defined(__ARM_ARCH_6K__) || defined(__ARM_ARCH_6Z__) || defined(__ARM_ARCH_6ZK__) || defined(__ARM_ARCH_6T2__) ) +# define XXH_FORCE_MEMORY_ACCESS 2 +# elif (defined(__INTEL_COMPILER) && !defined(WIN32)) || \ + (defined(__GNUC__) && ( defined(__ARM_ARCH_7__) || defined(__ARM_ARCH_7A__) || defined(__ARM_ARCH_7R__) || defined(__ARM_ARCH_7M__) || defined(__ARM_ARCH_7S__) )) +# define XXH_FORCE_MEMORY_ACCESS 1 +# endif +#endif + +/*!XXH_ACCEPT_NULL_INPUT_POINTER : + * If the input pointer is a null pointer, xxHash default behavior is to trigger a memory access error, since it is a bad pointer. + * When this option is enabled, xxHash output for null input pointers will be the same as a null-length input. + * By default, this option is disabled. To enable it, uncomment below define : + */ +/* #define XXH_ACCEPT_NULL_INPUT_POINTER 1 */ + +/*!XXH_FORCE_NATIVE_FORMAT : + * By default, xxHash library provides endian-independant Hash values, based on little-endian convention. + * Results are therefore identical for little-endian and big-endian CPU. + * This comes at a performance cost for big-endian CPU, since some swapping is required to emulate little-endian format. + * Should endian-independance be of no importance for your application, you may set the #define below to 1, + * to improve speed for Big-endian CPU. + * This option has no impact on Little_Endian CPU. + */ +#ifndef XXH_FORCE_NATIVE_FORMAT /* can be defined externally */ +# define XXH_FORCE_NATIVE_FORMAT 0 +#endif + +/*!XXH_FORCE_ALIGN_CHECK : + * This is a minor performance trick, only useful with lots of very small keys. + * It means : check for aligned/unaligned input. + * The check costs one initial branch per hash; set to 0 when the input data + * is guaranteed to be aligned. + */ +#ifndef XXH_FORCE_ALIGN_CHECK /* can be defined externally */ +# if defined(__i386) || defined(_M_IX86) || defined(__x86_64__) || defined(_M_X64) +# define XXH_FORCE_ALIGN_CHECK 0 +# else +# define XXH_FORCE_ALIGN_CHECK 1 +# endif +#endif + + +/* ************************************* +* Includes & Memory related functions +***************************************/ +/* Modify the local functions below should you wish to use some other memory routines */ +/* for malloc(), free() */ +#include +static void* XXH_malloc(size_t s) { return malloc(s); } +static void XXH_free (void* p) { free(p); } +/* for memcpy() */ +#include +static void* XXH_memcpy(void* dest, const void* src, size_t size) { return memcpy(dest,src,size); } + +#ifndef XXH_STATIC_LINKING_ONLY +# define XXH_STATIC_LINKING_ONLY +#endif +#include "xxhash.h" + + +/* ************************************* +* Compiler Specific Options +***************************************/ +#ifdef _MSC_VER /* Visual Studio */ +# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ +# define FORCE_INLINE static __forceinline +#else +# if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ +# ifdef __GNUC__ +# define FORCE_INLINE static inline __attribute__((always_inline)) +# else +# define FORCE_INLINE static inline +# endif +# else +# define FORCE_INLINE static +# endif /* __STDC_VERSION__ */ +#endif + + +/* ************************************* +* Basic Types +***************************************/ +#ifndef MEM_MODULE +# define MEM_MODULE +# if !defined (__VMS) && (defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) ) +# include + typedef uint8_t BYTE; + typedef uint16_t U16; + typedef uint32_t U32; + typedef int32_t S32; + typedef uint64_t U64; +# else + typedef unsigned char BYTE; + typedef unsigned short U16; + typedef unsigned int U32; + typedef signed int S32; + typedef unsigned long long U64; /* if your compiler doesn't support unsigned long long, replace by another 64-bit type here. Note that xxhash.h will also need to be updated. */ +# endif +#endif + + +#if (defined(XXH_FORCE_MEMORY_ACCESS) && (XXH_FORCE_MEMORY_ACCESS==2)) + +/* Force direct memory access. Only works on CPU which support unaligned memory access in hardware */ +static U32 XXH_read32(const void* memPtr) { return *(const U32*) memPtr; } +static U64 XXH_read64(const void* memPtr) { return *(const U64*) memPtr; } + +#elif (defined(XXH_FORCE_MEMORY_ACCESS) && (XXH_FORCE_MEMORY_ACCESS==1)) + +/* __pack instructions are safer, but compiler specific, hence potentially problematic for some compilers */ +/* currently only defined for gcc and icc */ +typedef union { U32 u32; U64 u64; } __attribute__((packed)) unalign; + +static U32 XXH_read32(const void* ptr) { return ((const unalign*)ptr)->u32; } +static U64 XXH_read64(const void* ptr) { return ((const unalign*)ptr)->u64; } + +#else + +/* portable and safe solution. Generally efficient. + * see : http://stackoverflow.com/a/32095106/646947 + */ + +static U32 XXH_read32(const void* memPtr) +{ + U32 val; + memcpy(&val, memPtr, sizeof(val)); + return val; +} + +static U64 XXH_read64(const void* memPtr) +{ + U64 val; + memcpy(&val, memPtr, sizeof(val)); + return val; +} + +#endif /* XXH_FORCE_DIRECT_MEMORY_ACCESS */ + + +/* **************************************** +* Compiler-specific Functions and Macros +******************************************/ +#define GCC_VERSION (__GNUC__ * 100 + __GNUC_MINOR__) + +/* Note : although _rotl exists for minGW (GCC under windows), performance seems poor */ +#if defined(_MSC_VER) +# define XXH_rotl32(x,r) _rotl(x,r) +# define XXH_rotl64(x,r) _rotl64(x,r) +#else +# define XXH_rotl32(x,r) ((x << r) | (x >> (32 - r))) +# define XXH_rotl64(x,r) ((x << r) | (x >> (64 - r))) +#endif + +#if defined(_MSC_VER) /* Visual Studio */ +# define XXH_swap32 _byteswap_ulong +# define XXH_swap64 _byteswap_uint64 +#elif GCC_VERSION >= 403 +# define XXH_swap32 __builtin_bswap32 +# define XXH_swap64 __builtin_bswap64 +#else +static U32 XXH_swap32 (U32 x) +{ + return ((x << 24) & 0xff000000 ) | + ((x << 8) & 0x00ff0000 ) | + ((x >> 8) & 0x0000ff00 ) | + ((x >> 24) & 0x000000ff ); +} +static U64 XXH_swap64 (U64 x) +{ + return ((x << 56) & 0xff00000000000000ULL) | + ((x << 40) & 0x00ff000000000000ULL) | + ((x << 24) & 0x0000ff0000000000ULL) | + ((x << 8) & 0x000000ff00000000ULL) | + ((x >> 8) & 0x00000000ff000000ULL) | + ((x >> 24) & 0x0000000000ff0000ULL) | + ((x >> 40) & 0x000000000000ff00ULL) | + ((x >> 56) & 0x00000000000000ffULL); +} +#endif + + +/* ************************************* +* Architecture Macros +***************************************/ +typedef enum { XXH_bigEndian=0, XXH_littleEndian=1 } XXH_endianess; + +/* XXH_CPU_LITTLE_ENDIAN can be defined externally, for example on the compiler command line */ +#ifndef XXH_CPU_LITTLE_ENDIAN + static const int g_one = 1; +# define XXH_CPU_LITTLE_ENDIAN (*(const char*)(&g_one)) +#endif + + +/* *************************** +* Memory reads +*****************************/ +typedef enum { XXH_aligned, XXH_unaligned } XXH_alignment; + +FORCE_INLINE U32 XXH_readLE32_align(const void* ptr, XXH_endianess endian, XXH_alignment align) +{ + if (align==XXH_unaligned) + return endian==XXH_littleEndian ? XXH_read32(ptr) : XXH_swap32(XXH_read32(ptr)); + else + return endian==XXH_littleEndian ? *(const U32*)ptr : XXH_swap32(*(const U32*)ptr); +} + +FORCE_INLINE U32 XXH_readLE32(const void* ptr, XXH_endianess endian) +{ + return XXH_readLE32_align(ptr, endian, XXH_unaligned); +} + +static U32 XXH_readBE32(const void* ptr) +{ + return XXH_CPU_LITTLE_ENDIAN ? XXH_swap32(XXH_read32(ptr)) : XXH_read32(ptr); +} + +FORCE_INLINE U64 XXH_readLE64_align(const void* ptr, XXH_endianess endian, XXH_alignment align) +{ + if (align==XXH_unaligned) + return endian==XXH_littleEndian ? XXH_read64(ptr) : XXH_swap64(XXH_read64(ptr)); + else + return endian==XXH_littleEndian ? *(const U64*)ptr : XXH_swap64(*(const U64*)ptr); +} + +FORCE_INLINE U64 XXH_readLE64(const void* ptr, XXH_endianess endian) +{ + return XXH_readLE64_align(ptr, endian, XXH_unaligned); +} + +static U64 XXH_readBE64(const void* ptr) +{ + return XXH_CPU_LITTLE_ENDIAN ? XXH_swap64(XXH_read64(ptr)) : XXH_read64(ptr); +} + + +/* ************************************* +* Macros +***************************************/ +#define XXH_STATIC_ASSERT(c) { enum { XXH_static_assert = 1/(int)(!!(c)) }; } /* use only *after* variable declarations */ + + +/* ************************************* +* Constants +***************************************/ +static const U32 PRIME32_1 = 2654435761U; +static const U32 PRIME32_2 = 2246822519U; +static const U32 PRIME32_3 = 3266489917U; +static const U32 PRIME32_4 = 668265263U; +static const U32 PRIME32_5 = 374761393U; + +static const U64 PRIME64_1 = 11400714785074694791ULL; +static const U64 PRIME64_2 = 14029467366897019727ULL; +static const U64 PRIME64_3 = 1609587929392839161ULL; +static const U64 PRIME64_4 = 9650029242287828579ULL; +static const U64 PRIME64_5 = 2870177450012600261ULL; + +XXH_PUBLIC_API unsigned XXH_versionNumber (void) { return XXH_VERSION_NUMBER; } + + +/* ************************** +* Utils +****************************/ +XXH_PUBLIC_API void XXH32_copyState(XXH32_state_t* restrict dstState, const XXH32_state_t* restrict srcState) +{ + memcpy(dstState, srcState, sizeof(*dstState)); +} + +XXH_PUBLIC_API void XXH64_copyState(XXH64_state_t* restrict dstState, const XXH64_state_t* restrict srcState) +{ + memcpy(dstState, srcState, sizeof(*dstState)); +} + + +/* *************************** +* Simple Hash Functions +*****************************/ + +static U32 XXH32_round(U32 seed, U32 input) +{ + seed += input * PRIME32_2; + seed = XXH_rotl32(seed, 13); + seed *= PRIME32_1; + return seed; +} + +FORCE_INLINE U32 XXH32_endian_align(const void* input, size_t len, U32 seed, XXH_endianess endian, XXH_alignment align) +{ + const BYTE* p = (const BYTE*)input; + const BYTE* bEnd = p + len; + U32 h32; +#define XXH_get32bits(p) XXH_readLE32_align(p, endian, align) + +#ifdef XXH_ACCEPT_NULL_INPUT_POINTER + if (p==NULL) { + len=0; + bEnd=p=(const BYTE*)(size_t)16; + } +#endif + + if (len>=16) { + const BYTE* const limit = bEnd - 16; + U32 v1 = seed + PRIME32_1 + PRIME32_2; + U32 v2 = seed + PRIME32_2; + U32 v3 = seed + 0; + U32 v4 = seed - PRIME32_1; + + do { + v1 = XXH32_round(v1, XXH_get32bits(p)); p+=4; + v2 = XXH32_round(v2, XXH_get32bits(p)); p+=4; + v3 = XXH32_round(v3, XXH_get32bits(p)); p+=4; + v4 = XXH32_round(v4, XXH_get32bits(p)); p+=4; + } while (p<=limit); + + h32 = XXH_rotl32(v1, 1) + XXH_rotl32(v2, 7) + XXH_rotl32(v3, 12) + XXH_rotl32(v4, 18); + } else { + h32 = seed + PRIME32_5; + } + + h32 += (U32) len; + + while (p+4<=bEnd) { + h32 += XXH_get32bits(p) * PRIME32_3; + h32 = XXH_rotl32(h32, 17) * PRIME32_4 ; + p+=4; + } + + while (p> 15; + h32 *= PRIME32_2; + h32 ^= h32 >> 13; + h32 *= PRIME32_3; + h32 ^= h32 >> 16; + + return h32; +} + + +XXH_PUBLIC_API unsigned int XXH32 (const void* input, size_t len, unsigned int seed) +{ +#if 0 + /* Simple version, good for code maintenance, but unfortunately slow for small inputs */ + XXH32_CREATESTATE_STATIC(state); + XXH32_reset(state, seed); + XXH32_update(state, input, len); + return XXH32_digest(state); +#else + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + + if (XXH_FORCE_ALIGN_CHECK) { + if ((((size_t)input) & 3) == 0) { /* Input is 4-bytes aligned, leverage the speed benefit */ + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH32_endian_align(input, len, seed, XXH_littleEndian, XXH_aligned); + else + return XXH32_endian_align(input, len, seed, XXH_bigEndian, XXH_aligned); + } } + + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH32_endian_align(input, len, seed, XXH_littleEndian, XXH_unaligned); + else + return XXH32_endian_align(input, len, seed, XXH_bigEndian, XXH_unaligned); +#endif +} + + +static U64 XXH64_round(U64 acc, U64 input) +{ + acc += input * PRIME64_2; + acc = XXH_rotl64(acc, 31); + acc *= PRIME64_1; + return acc; +} + +static U64 XXH64_mergeRound(U64 acc, U64 val) +{ + val = XXH64_round(0, val); + acc ^= val; + acc = acc * PRIME64_1 + PRIME64_4; + return acc; +} + +FORCE_INLINE U64 XXH64_endian_align(const void* input, size_t len, U64 seed, XXH_endianess endian, XXH_alignment align) +{ + const BYTE* p = (const BYTE*)input; + const BYTE* const bEnd = p + len; + U64 h64; +#define XXH_get64bits(p) XXH_readLE64_align(p, endian, align) + +#ifdef XXH_ACCEPT_NULL_INPUT_POINTER + if (p==NULL) { + len=0; + bEnd=p=(const BYTE*)(size_t)32; + } +#endif + + if (len>=32) { + const BYTE* const limit = bEnd - 32; + U64 v1 = seed + PRIME64_1 + PRIME64_2; + U64 v2 = seed + PRIME64_2; + U64 v3 = seed + 0; + U64 v4 = seed - PRIME64_1; + + do { + v1 = XXH64_round(v1, XXH_get64bits(p)); p+=8; + v2 = XXH64_round(v2, XXH_get64bits(p)); p+=8; + v3 = XXH64_round(v3, XXH_get64bits(p)); p+=8; + v4 = XXH64_round(v4, XXH_get64bits(p)); p+=8; + } while (p<=limit); + + h64 = XXH_rotl64(v1, 1) + XXH_rotl64(v2, 7) + XXH_rotl64(v3, 12) + XXH_rotl64(v4, 18); + h64 = XXH64_mergeRound(h64, v1); + h64 = XXH64_mergeRound(h64, v2); + h64 = XXH64_mergeRound(h64, v3); + h64 = XXH64_mergeRound(h64, v4); + + } else { + h64 = seed + PRIME64_5; + } + + h64 += (U64) len; + + while (p+8<=bEnd) { + U64 const k1 = XXH64_round(0, XXH_get64bits(p)); + h64 ^= k1; + h64 = XXH_rotl64(h64,27) * PRIME64_1 + PRIME64_4; + p+=8; + } + + if (p+4<=bEnd) { + h64 ^= (U64)(XXH_get32bits(p)) * PRIME64_1; + h64 = XXH_rotl64(h64, 23) * PRIME64_2 + PRIME64_3; + p+=4; + } + + while (p> 33; + h64 *= PRIME64_2; + h64 ^= h64 >> 29; + h64 *= PRIME64_3; + h64 ^= h64 >> 32; + + return h64; +} + + +XXH_PUBLIC_API unsigned long long XXH64 (const void* input, size_t len, unsigned long long seed) +{ +#if 0 + /* Simple version, good for code maintenance, but unfortunately slow for small inputs */ + XXH64_CREATESTATE_STATIC(state); + XXH64_reset(state, seed); + XXH64_update(state, input, len); + return XXH64_digest(state); +#else + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + + if (XXH_FORCE_ALIGN_CHECK) { + if ((((size_t)input) & 7)==0) { /* Input is aligned, let's leverage the speed advantage */ + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH64_endian_align(input, len, seed, XXH_littleEndian, XXH_aligned); + else + return XXH64_endian_align(input, len, seed, XXH_bigEndian, XXH_aligned); + } } + + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH64_endian_align(input, len, seed, XXH_littleEndian, XXH_unaligned); + else + return XXH64_endian_align(input, len, seed, XXH_bigEndian, XXH_unaligned); +#endif +} + + +/* ************************************************** +* Advanced Hash Functions +****************************************************/ + +XXH_PUBLIC_API XXH32_state_t* XXH32_createState(void) +{ + return (XXH32_state_t*)XXH_malloc(sizeof(XXH32_state_t)); +} +XXH_PUBLIC_API XXH_errorcode XXH32_freeState(XXH32_state_t* statePtr) +{ + XXH_free(statePtr); + return XXH_OK; +} + +XXH_PUBLIC_API XXH64_state_t* XXH64_createState(void) +{ + return (XXH64_state_t*)XXH_malloc(sizeof(XXH64_state_t)); +} +XXH_PUBLIC_API XXH_errorcode XXH64_freeState(XXH64_state_t* statePtr) +{ + XXH_free(statePtr); + return XXH_OK; +} + + +/*** Hash feed ***/ + +XXH_PUBLIC_API XXH_errorcode XXH32_reset(XXH32_state_t* statePtr, unsigned int seed) +{ + XXH32_state_t state; /* using a local state to memcpy() in order to avoid strict-aliasing warnings */ + memset(&state, 0, sizeof(state)-4); /* do not write into reserved, for future removal */ + state.v1 = seed + PRIME32_1 + PRIME32_2; + state.v2 = seed + PRIME32_2; + state.v3 = seed + 0; + state.v4 = seed - PRIME32_1; + memcpy(statePtr, &state, sizeof(state)); + return XXH_OK; +} + + +XXH_PUBLIC_API XXH_errorcode XXH64_reset(XXH64_state_t* statePtr, unsigned long long seed) +{ + XXH64_state_t state; /* using a local state to memcpy() in order to avoid strict-aliasing warnings */ + memset(&state, 0, sizeof(state)-8); /* do not write into reserved, for future removal */ + state.v1 = seed + PRIME64_1 + PRIME64_2; + state.v2 = seed + PRIME64_2; + state.v3 = seed + 0; + state.v4 = seed - PRIME64_1; + memcpy(statePtr, &state, sizeof(state)); + return XXH_OK; +} + + +FORCE_INLINE XXH_errorcode XXH32_update_endian (XXH32_state_t* state, const void* input, size_t len, XXH_endianess endian) +{ + const BYTE* p = (const BYTE*)input; + const BYTE* const bEnd = p + len; + +#ifdef XXH_ACCEPT_NULL_INPUT_POINTER + if (input==NULL) return XXH_ERROR; +#endif + + state->total_len_32 += (unsigned)len; + state->large_len |= (len>=16) | (state->total_len_32>=16); + + if (state->memsize + len < 16) { /* fill in tmp buffer */ + XXH_memcpy((BYTE*)(state->mem32) + state->memsize, input, len); + state->memsize += (unsigned)len; + return XXH_OK; + } + + if (state->memsize) { /* some data left from previous update */ + XXH_memcpy((BYTE*)(state->mem32) + state->memsize, input, 16-state->memsize); + { const U32* p32 = state->mem32; + state->v1 = XXH32_round(state->v1, XXH_readLE32(p32, endian)); p32++; + state->v2 = XXH32_round(state->v2, XXH_readLE32(p32, endian)); p32++; + state->v3 = XXH32_round(state->v3, XXH_readLE32(p32, endian)); p32++; + state->v4 = XXH32_round(state->v4, XXH_readLE32(p32, endian)); p32++; + } + p += 16-state->memsize; + state->memsize = 0; + } + + if (p <= bEnd-16) { + const BYTE* const limit = bEnd - 16; + U32 v1 = state->v1; + U32 v2 = state->v2; + U32 v3 = state->v3; + U32 v4 = state->v4; + + do { + v1 = XXH32_round(v1, XXH_readLE32(p, endian)); p+=4; + v2 = XXH32_round(v2, XXH_readLE32(p, endian)); p+=4; + v3 = XXH32_round(v3, XXH_readLE32(p, endian)); p+=4; + v4 = XXH32_round(v4, XXH_readLE32(p, endian)); p+=4; + } while (p<=limit); + + state->v1 = v1; + state->v2 = v2; + state->v3 = v3; + state->v4 = v4; + } + + if (p < bEnd) { + XXH_memcpy(state->mem32, p, (size_t)(bEnd-p)); + state->memsize = (unsigned)(bEnd-p); + } + + return XXH_OK; +} + +XXH_PUBLIC_API XXH_errorcode XXH32_update (XXH32_state_t* state_in, const void* input, size_t len) +{ + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH32_update_endian(state_in, input, len, XXH_littleEndian); + else + return XXH32_update_endian(state_in, input, len, XXH_bigEndian); +} + + + +FORCE_INLINE U32 XXH32_digest_endian (const XXH32_state_t* state, XXH_endianess endian) +{ + const BYTE * p = (const BYTE*)state->mem32; + const BYTE* const bEnd = (const BYTE*)(state->mem32) + state->memsize; + U32 h32; + + if (state->large_len) { + h32 = XXH_rotl32(state->v1, 1) + XXH_rotl32(state->v2, 7) + XXH_rotl32(state->v3, 12) + XXH_rotl32(state->v4, 18); + } else { + h32 = state->v3 /* == seed */ + PRIME32_5; + } + + h32 += state->total_len_32; + + while (p+4<=bEnd) { + h32 += XXH_readLE32(p, endian) * PRIME32_3; + h32 = XXH_rotl32(h32, 17) * PRIME32_4; + p+=4; + } + + while (p> 15; + h32 *= PRIME32_2; + h32 ^= h32 >> 13; + h32 *= PRIME32_3; + h32 ^= h32 >> 16; + + return h32; +} + + +XXH_PUBLIC_API unsigned int XXH32_digest (const XXH32_state_t* state_in) +{ + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH32_digest_endian(state_in, XXH_littleEndian); + else + return XXH32_digest_endian(state_in, XXH_bigEndian); +} + + + +/* **** XXH64 **** */ + +FORCE_INLINE XXH_errorcode XXH64_update_endian (XXH64_state_t* state, const void* input, size_t len, XXH_endianess endian) +{ + const BYTE* p = (const BYTE*)input; + const BYTE* const bEnd = p + len; + +#ifdef XXH_ACCEPT_NULL_INPUT_POINTER + if (input==NULL) return XXH_ERROR; +#endif + + state->total_len += len; + + if (state->memsize + len < 32) { /* fill in tmp buffer */ + XXH_memcpy(((BYTE*)state->mem64) + state->memsize, input, len); + state->memsize += (U32)len; + return XXH_OK; + } + + if (state->memsize) { /* tmp buffer is full */ + XXH_memcpy(((BYTE*)state->mem64) + state->memsize, input, 32-state->memsize); + state->v1 = XXH64_round(state->v1, XXH_readLE64(state->mem64+0, endian)); + state->v2 = XXH64_round(state->v2, XXH_readLE64(state->mem64+1, endian)); + state->v3 = XXH64_round(state->v3, XXH_readLE64(state->mem64+2, endian)); + state->v4 = XXH64_round(state->v4, XXH_readLE64(state->mem64+3, endian)); + p += 32-state->memsize; + state->memsize = 0; + } + + if (p+32 <= bEnd) { + const BYTE* const limit = bEnd - 32; + U64 v1 = state->v1; + U64 v2 = state->v2; + U64 v3 = state->v3; + U64 v4 = state->v4; + + do { + v1 = XXH64_round(v1, XXH_readLE64(p, endian)); p+=8; + v2 = XXH64_round(v2, XXH_readLE64(p, endian)); p+=8; + v3 = XXH64_round(v3, XXH_readLE64(p, endian)); p+=8; + v4 = XXH64_round(v4, XXH_readLE64(p, endian)); p+=8; + } while (p<=limit); + + state->v1 = v1; + state->v2 = v2; + state->v3 = v3; + state->v4 = v4; + } + + if (p < bEnd) { + XXH_memcpy(state->mem64, p, (size_t)(bEnd-p)); + state->memsize = (unsigned)(bEnd-p); + } + + return XXH_OK; +} + +XXH_PUBLIC_API XXH_errorcode XXH64_update (XXH64_state_t* state_in, const void* input, size_t len) +{ + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH64_update_endian(state_in, input, len, XXH_littleEndian); + else + return XXH64_update_endian(state_in, input, len, XXH_bigEndian); +} + + + +FORCE_INLINE U64 XXH64_digest_endian (const XXH64_state_t* state, XXH_endianess endian) +{ + const BYTE * p = (const BYTE*)state->mem64; + const BYTE* const bEnd = (const BYTE*)state->mem64 + state->memsize; + U64 h64; + + if (state->total_len >= 32) { + U64 const v1 = state->v1; + U64 const v2 = state->v2; + U64 const v3 = state->v3; + U64 const v4 = state->v4; + + h64 = XXH_rotl64(v1, 1) + XXH_rotl64(v2, 7) + XXH_rotl64(v3, 12) + XXH_rotl64(v4, 18); + h64 = XXH64_mergeRound(h64, v1); + h64 = XXH64_mergeRound(h64, v2); + h64 = XXH64_mergeRound(h64, v3); + h64 = XXH64_mergeRound(h64, v4); + } else { + h64 = state->v3 + PRIME64_5; + } + + h64 += (U64) state->total_len; + + while (p+8<=bEnd) { + U64 const k1 = XXH64_round(0, XXH_readLE64(p, endian)); + h64 ^= k1; + h64 = XXH_rotl64(h64,27) * PRIME64_1 + PRIME64_4; + p+=8; + } + + if (p+4<=bEnd) { + h64 ^= (U64)(XXH_readLE32(p, endian)) * PRIME64_1; + h64 = XXH_rotl64(h64, 23) * PRIME64_2 + PRIME64_3; + p+=4; + } + + while (p> 33; + h64 *= PRIME64_2; + h64 ^= h64 >> 29; + h64 *= PRIME64_3; + h64 ^= h64 >> 32; + + return h64; +} + + +XXH_PUBLIC_API unsigned long long XXH64_digest (const XXH64_state_t* state_in) +{ + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH64_digest_endian(state_in, XXH_littleEndian); + else + return XXH64_digest_endian(state_in, XXH_bigEndian); +} + + +/* ************************** +* Canonical representation +****************************/ + +/*! Default XXH result types are basic unsigned 32 and 64 bits. +* The canonical representation follows human-readable write convention, aka big-endian (large digits first). +* These functions allow transformation of hash result into and from its canonical format. +* This way, hash values can be written into a file or buffer, and remain comparable across different systems and programs. +*/ + +XXH_PUBLIC_API void XXH32_canonicalFromHash(XXH32_canonical_t* dst, XXH32_hash_t hash) +{ + XXH_STATIC_ASSERT(sizeof(XXH32_canonical_t) == sizeof(XXH32_hash_t)); + if (XXH_CPU_LITTLE_ENDIAN) hash = XXH_swap32(hash); + memcpy(dst, &hash, sizeof(*dst)); +} + +XXH_PUBLIC_API void XXH64_canonicalFromHash(XXH64_canonical_t* dst, XXH64_hash_t hash) +{ + XXH_STATIC_ASSERT(sizeof(XXH64_canonical_t) == sizeof(XXH64_hash_t)); + if (XXH_CPU_LITTLE_ENDIAN) hash = XXH_swap64(hash); + memcpy(dst, &hash, sizeof(*dst)); +} + +XXH_PUBLIC_API XXH32_hash_t XXH32_hashFromCanonical(const XXH32_canonical_t* src) +{ + return XXH_readBE32(src); +} + +XXH_PUBLIC_API XXH64_hash_t XXH64_hashFromCanonical(const XXH64_canonical_t* src) +{ + return XXH_readBE64(src); +} diff --git a/contrib/linux-kernel/lib/xxhash.h b/contrib/linux-kernel/lib/xxhash.h new file mode 100644 index 000000000..9bad1f59f --- /dev/null +++ b/contrib/linux-kernel/lib/xxhash.h @@ -0,0 +1,305 @@ +/* + xxHash - Extremely Fast Hash algorithm + Header File + Copyright (C) 2012-2016, Yann Collet. + + BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php) + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are + met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the + distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + You can contact the author at : + - xxHash source repository : https://github.com/Cyan4973/xxHash +*/ + +/* Notice extracted from xxHash homepage : + +xxHash is an extremely fast Hash algorithm, running at RAM speed limits. +It also successfully passes all tests from the SMHasher suite. + +Comparison (single thread, Windows Seven 32 bits, using SMHasher on a Core 2 Duo @3GHz) + +Name Speed Q.Score Author +xxHash 5.4 GB/s 10 +CrapWow 3.2 GB/s 2 Andrew +MumurHash 3a 2.7 GB/s 10 Austin Appleby +SpookyHash 2.0 GB/s 10 Bob Jenkins +SBox 1.4 GB/s 9 Bret Mulvey +Lookup3 1.2 GB/s 9 Bob Jenkins +SuperFastHash 1.2 GB/s 1 Paul Hsieh +CityHash64 1.05 GB/s 10 Pike & Alakuijala +FNV 0.55 GB/s 5 Fowler, Noll, Vo +CRC32 0.43 GB/s 9 +MD5-32 0.33 GB/s 10 Ronald L. Rivest +SHA1-32 0.28 GB/s 10 + +Q.Score is a measure of quality of the hash function. +It depends on successfully passing SMHasher test set. +10 is a perfect score. + +A 64-bits version, named XXH64, is available since r35. +It offers much better speed, but for 64-bits applications only. +Name Speed on 64 bits Speed on 32 bits +XXH64 13.8 GB/s 1.9 GB/s +XXH32 6.8 GB/s 6.0 GB/s +*/ + +#if defined (__cplusplus) +extern "C" { +#endif + +#ifndef XXHASH_H_5627135585666179 +#define XXHASH_H_5627135585666179 1 + + +/* **************************** +* Definitions +******************************/ +#include /* size_t */ +typedef enum { XXH_OK=0, XXH_ERROR } XXH_errorcode; + + +/* **************************** +* API modifier +******************************/ +/** XXH_PRIVATE_API +* This is useful if you want to include xxhash functions in `static` mode +* in order to inline them, and remove their symbol from the public list. +* Methodology : +* #define XXH_PRIVATE_API +* #include "xxhash.h" +* `xxhash.c` is automatically included. +* It's not useful to compile and link it as a separate module anymore. +*/ +#ifdef XXH_PRIVATE_API +# ifndef XXH_STATIC_LINKING_ONLY +# define XXH_STATIC_LINKING_ONLY +# endif +# if defined(__GNUC__) +# define XXH_PUBLIC_API static __inline __attribute__((unused)) +# elif defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) +# define XXH_PUBLIC_API static inline +# elif defined(_MSC_VER) +# define XXH_PUBLIC_API static __inline +# else +# define XXH_PUBLIC_API static /* this version may generate warnings for unused static functions; disable the relevant warning */ +# endif +#else +# define XXH_PUBLIC_API /* do nothing */ +#endif /* XXH_PRIVATE_API */ + +/*!XXH_NAMESPACE, aka Namespace Emulation : + +If you want to include _and expose_ xxHash functions from within your own library, +but also want to avoid symbol collisions with another library which also includes xxHash, + +you can use XXH_NAMESPACE, to automatically prefix any public symbol from xxhash library +with the value of XXH_NAMESPACE (so avoid to keep it NULL and avoid numeric values). + +Note that no change is required within the calling program as long as it includes `xxhash.h` : +regular symbol name will be automatically translated by this header. +*/ +#ifdef XXH_NAMESPACE +# define XXH_CAT(A,B) A##B +# define XXH_NAME2(A,B) XXH_CAT(A,B) +# define XXH32 XXH_NAME2(XXH_NAMESPACE, XXH32) +# define XXH64 XXH_NAME2(XXH_NAMESPACE, XXH64) +# define XXH_versionNumber XXH_NAME2(XXH_NAMESPACE, XXH_versionNumber) +# define XXH32_createState XXH_NAME2(XXH_NAMESPACE, XXH32_createState) +# define XXH64_createState XXH_NAME2(XXH_NAMESPACE, XXH64_createState) +# define XXH32_freeState XXH_NAME2(XXH_NAMESPACE, XXH32_freeState) +# define XXH64_freeState XXH_NAME2(XXH_NAMESPACE, XXH64_freeState) +# define XXH32_reset XXH_NAME2(XXH_NAMESPACE, XXH32_reset) +# define XXH64_reset XXH_NAME2(XXH_NAMESPACE, XXH64_reset) +# define XXH32_update XXH_NAME2(XXH_NAMESPACE, XXH32_update) +# define XXH64_update XXH_NAME2(XXH_NAMESPACE, XXH64_update) +# define XXH32_digest XXH_NAME2(XXH_NAMESPACE, XXH32_digest) +# define XXH64_digest XXH_NAME2(XXH_NAMESPACE, XXH64_digest) +# define XXH32_copyState XXH_NAME2(XXH_NAMESPACE, XXH32_copyState) +# define XXH64_copyState XXH_NAME2(XXH_NAMESPACE, XXH64_copyState) +# define XXH32_canonicalFromHash XXH_NAME2(XXH_NAMESPACE, XXH32_canonicalFromHash) +# define XXH64_canonicalFromHash XXH_NAME2(XXH_NAMESPACE, XXH64_canonicalFromHash) +# define XXH32_hashFromCanonical XXH_NAME2(XXH_NAMESPACE, XXH32_hashFromCanonical) +# define XXH64_hashFromCanonical XXH_NAME2(XXH_NAMESPACE, XXH64_hashFromCanonical) +#endif + + +/* ************************************* +* Version +***************************************/ +#define XXH_VERSION_MAJOR 0 +#define XXH_VERSION_MINOR 6 +#define XXH_VERSION_RELEASE 2 +#define XXH_VERSION_NUMBER (XXH_VERSION_MAJOR *100*100 + XXH_VERSION_MINOR *100 + XXH_VERSION_RELEASE) +XXH_PUBLIC_API unsigned XXH_versionNumber (void); + + +/* **************************** +* Simple Hash Functions +******************************/ +typedef unsigned int XXH32_hash_t; +typedef unsigned long long XXH64_hash_t; + +XXH_PUBLIC_API XXH32_hash_t XXH32 (const void* input, size_t length, unsigned int seed); +XXH_PUBLIC_API XXH64_hash_t XXH64 (const void* input, size_t length, unsigned long long seed); + +/*! +XXH32() : + Calculate the 32-bits hash of sequence "length" bytes stored at memory address "input". + The memory between input & input+length must be valid (allocated and read-accessible). + "seed" can be used to alter the result predictably. + Speed on Core 2 Duo @ 3 GHz (single thread, SMHasher benchmark) : 5.4 GB/s +XXH64() : + Calculate the 64-bits hash of sequence of length "len" stored at memory address "input". + "seed" can be used to alter the result predictably. + This function runs 2x faster on 64-bits systems, but slower on 32-bits systems (see benchmark). +*/ + + +/* **************************** +* Streaming Hash Functions +******************************/ +typedef struct XXH32_state_s XXH32_state_t; /* incomplete type */ +typedef struct XXH64_state_s XXH64_state_t; /* incomplete type */ + +/*! State allocation, compatible with dynamic libraries */ + +XXH_PUBLIC_API XXH32_state_t* XXH32_createState(void); +XXH_PUBLIC_API XXH_errorcode XXH32_freeState(XXH32_state_t* statePtr); + +XXH_PUBLIC_API XXH64_state_t* XXH64_createState(void); +XXH_PUBLIC_API XXH_errorcode XXH64_freeState(XXH64_state_t* statePtr); + + +/* hash streaming */ + +XXH_PUBLIC_API XXH_errorcode XXH32_reset (XXH32_state_t* statePtr, unsigned int seed); +XXH_PUBLIC_API XXH_errorcode XXH32_update (XXH32_state_t* statePtr, const void* input, size_t length); +XXH_PUBLIC_API XXH32_hash_t XXH32_digest (const XXH32_state_t* statePtr); + +XXH_PUBLIC_API XXH_errorcode XXH64_reset (XXH64_state_t* statePtr, unsigned long long seed); +XXH_PUBLIC_API XXH_errorcode XXH64_update (XXH64_state_t* statePtr, const void* input, size_t length); +XXH_PUBLIC_API XXH64_hash_t XXH64_digest (const XXH64_state_t* statePtr); + +/* +These functions generate the xxHash of an input provided in multiple segments. +Note that, for small input, they are slower than single-call functions, due to state management. +For small input, prefer `XXH32()` and `XXH64()` . + +XXH state must first be allocated, using XXH*_createState() . + +Start a new hash by initializing state with a seed, using XXH*_reset(). + +Then, feed the hash state by calling XXH*_update() as many times as necessary. +Obviously, input must be allocated and read accessible. +The function returns an error code, with 0 meaning OK, and any other value meaning there is an error. + +Finally, a hash value can be produced anytime, by using XXH*_digest(). +This function returns the nn-bits hash as an int or long long. + +It's still possible to continue inserting input into the hash state after a digest, +and generate some new hashes later on, by calling again XXH*_digest(). + +When done, free XXH state space if it was allocated dynamically. +*/ + + +/* ************************** +* Utils +****************************/ +#if !(defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L)) /* ! C99 */ +# define restrict /* disable restrict */ +#endif + +XXH_PUBLIC_API void XXH32_copyState(XXH32_state_t* restrict dst_state, const XXH32_state_t* restrict src_state); +XXH_PUBLIC_API void XXH64_copyState(XXH64_state_t* restrict dst_state, const XXH64_state_t* restrict src_state); + + +/* ************************** +* Canonical representation +****************************/ +/* Default result type for XXH functions are primitive unsigned 32 and 64 bits. +* The canonical representation uses human-readable write convention, aka big-endian (large digits first). +* These functions allow transformation of hash result into and from its canonical format. +* This way, hash values can be written into a file / memory, and remain comparable on different systems and programs. +*/ +typedef struct { unsigned char digest[4]; } XXH32_canonical_t; +typedef struct { unsigned char digest[8]; } XXH64_canonical_t; + +XXH_PUBLIC_API void XXH32_canonicalFromHash(XXH32_canonical_t* dst, XXH32_hash_t hash); +XXH_PUBLIC_API void XXH64_canonicalFromHash(XXH64_canonical_t* dst, XXH64_hash_t hash); + +XXH_PUBLIC_API XXH32_hash_t XXH32_hashFromCanonical(const XXH32_canonical_t* src); +XXH_PUBLIC_API XXH64_hash_t XXH64_hashFromCanonical(const XXH64_canonical_t* src); + +#endif /* XXHASH_H_5627135585666179 */ + + + +/* ================================================================================================ + This section contains definitions which are not guaranteed to remain stable. + They may change in future versions, becoming incompatible with a different version of the library. + They shall only be used with static linking. + Never use these definitions in association with dynamic linking ! +=================================================================================================== */ +#if defined(XXH_STATIC_LINKING_ONLY) && !defined(XXH_STATIC_H_3543687687345) +#define XXH_STATIC_H_3543687687345 + +/* These definitions are only meant to allow allocation of XXH state + statically, on stack, or in a struct for example. + Do not use members directly. */ + + struct XXH32_state_s { + unsigned total_len_32; + unsigned large_len; + unsigned v1; + unsigned v2; + unsigned v3; + unsigned v4; + unsigned mem32[4]; /* buffer defined as U32 for alignment */ + unsigned memsize; + unsigned reserved; /* never read nor write, will be removed in a future version */ + }; /* typedef'd to XXH32_state_t */ + + struct XXH64_state_s { + unsigned long long total_len; + unsigned long long v1; + unsigned long long v2; + unsigned long long v3; + unsigned long long v4; + unsigned long long mem64[4]; /* buffer defined as U64 for alignment */ + unsigned memsize; + unsigned reserved[2]; /* never read nor write, will be removed in a future version */ + }; /* typedef'd to XXH64_state_t */ + + +# ifdef XXH_PRIVATE_API +# include "xxhash.c" /* include xxhash functions as `static`, for inlining */ +# endif + +#endif /* XXH_STATIC_LINKING_ONLY && XXH_STATIC_H_3543687687345 */ + + +#if defined (__cplusplus) +} +#endif diff --git a/contrib/linux-kernel/lib/zstd_common.c b/contrib/linux-kernel/lib/zstd_common.c new file mode 100644 index 000000000..8408a589a --- /dev/null +++ b/contrib/linux-kernel/lib/zstd_common.c @@ -0,0 +1,73 @@ +/** + * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + + + +/*-************************************* +* Dependencies +***************************************/ +#include /* malloc */ +#include "error_private.h" +#define ZSTD_STATIC_LINKING_ONLY +#include "zstd.h" /* declaration of ZSTD_isError, ZSTD_getErrorName, ZSTD_getErrorCode, ZSTD_getErrorString, ZSTD_versionNumber */ + + +/*-**************************************** +* Version +******************************************/ +unsigned ZSTD_versionNumber (void) { return ZSTD_VERSION_NUMBER; } + + +/*-**************************************** +* ZSTD Error Management +******************************************/ +/*! ZSTD_isError() : +* tells if a return value is an error code */ +unsigned ZSTD_isError(size_t code) { return ERR_isError(code); } + +/*! ZSTD_getErrorName() : +* provides error code string from function result (useful for debugging) */ +const char* ZSTD_getErrorName(size_t code) { return ERR_getErrorName(code); } + +/*! ZSTD_getError() : +* convert a `size_t` function result into a proper ZSTD_errorCode enum */ +ZSTD_ErrorCode ZSTD_getErrorCode(size_t code) { return ERR_getErrorCode(code); } + +/*! ZSTD_getErrorString() : +* provides error code string from enum */ +const char* ZSTD_getErrorString(ZSTD_ErrorCode code) { return ERR_getErrorString(code); } + + +/*=************************************************************** +* Custom allocator +****************************************************************/ +/* default uses stdlib */ +void* ZSTD_defaultAllocFunction(void* opaque, size_t size) +{ + void* address = malloc(size); + (void)opaque; + return address; +} + +void ZSTD_defaultFreeFunction(void* opaque, void* address) +{ + (void)opaque; + free(address); +} + +void* ZSTD_malloc(size_t size, ZSTD_customMem customMem) +{ + return customMem.customAlloc(customMem.opaque, size); +} + +void ZSTD_free(void* ptr, ZSTD_customMem customMem) +{ + if (ptr!=NULL) + customMem.customFree(customMem.opaque, ptr); +} diff --git a/contrib/linux-kernel/lib/zstd_compress.c b/contrib/linux-kernel/lib/zstd_compress.c new file mode 100644 index 000000000..c31f8db91 --- /dev/null +++ b/contrib/linux-kernel/lib/zstd_compress.c @@ -0,0 +1,3400 @@ +/** + * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + + +/*-************************************* +* Dependencies +***************************************/ +#include /* memset */ +#include "mem.h" +#define FSE_STATIC_LINKING_ONLY /* FSE_encodeSymbol */ +#include "fse.h" +#define HUF_STATIC_LINKING_ONLY +#include "huf.h" +#include "zstd_internal.h" /* includes zstd.h */ + + +/*-************************************* +* Constants +***************************************/ +static const U32 g_searchStrength = 8; /* control skip over incompressible data */ +#define HASH_READ_SIZE 8 +typedef enum { ZSTDcs_created=0, ZSTDcs_init, ZSTDcs_ongoing, ZSTDcs_ending } ZSTD_compressionStage_e; + + +/*-************************************* +* Helper functions +***************************************/ +#define ZSTD_STATIC_ASSERT(c) { enum { ZSTD_static_assert = 1/(int)(!!(c)) }; } +size_t ZSTD_compressBound(size_t srcSize) { return FSE_compressBound(srcSize) + 12; } + + +/*-************************************* +* Sequence storage +***************************************/ +static void ZSTD_resetSeqStore(seqStore_t* ssPtr) +{ + ssPtr->lit = ssPtr->litStart; + ssPtr->sequences = ssPtr->sequencesStart; + ssPtr->longLengthID = 0; +} + + +/*-************************************* +* Context memory management +***************************************/ +struct ZSTD_CCtx_s { + const BYTE* nextSrc; /* next block here to continue on current prefix */ + const BYTE* base; /* All regular indexes relative to this position */ + const BYTE* dictBase; /* extDict indexes relative to this position */ + U32 dictLimit; /* below that point, need extDict */ + U32 lowLimit; /* below that point, no more data */ + U32 nextToUpdate; /* index from which to continue dictionary update */ + U32 nextToUpdate3; /* index from which to continue dictionary update */ + U32 hashLog3; /* dispatch table : larger == faster, more memory */ + U32 loadedDictEnd; /* index of end of dictionary */ + U32 forceWindow; /* force back-references to respect limit of 1<customMem = customMem; + return cctx; +} + +size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx) +{ + if (cctx==NULL) return 0; /* support free on NULL */ + ZSTD_free(cctx->workSpace, cctx->customMem); + ZSTD_free(cctx, cctx->customMem); + return 0; /* reserved as a potential error code in the future */ +} + +size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx) +{ + if (cctx==NULL) return 0; /* support sizeof on NULL */ + return sizeof(*cctx) + cctx->workSpaceSize; +} + +size_t ZSTD_setCCtxParameter(ZSTD_CCtx* cctx, ZSTD_CCtxParameter param, unsigned value) +{ + switch(param) + { + case ZSTD_p_forceWindow : cctx->forceWindow = value>0; cctx->loadedDictEnd = 0; return 0; + case ZSTD_p_forceRawDict : cctx->forceRawDict = value>0; return 0; + default: return ERROR(parameter_unknown); + } +} + +const seqStore_t* ZSTD_getSeqStore(const ZSTD_CCtx* ctx) /* hidden interface */ +{ + return &(ctx->seqStore); +} + +static ZSTD_parameters ZSTD_getParamsFromCCtx(const ZSTD_CCtx* cctx) +{ + return cctx->params; +} + + +/** ZSTD_checkParams() : + ensure param values remain within authorized range. + @return : 0, or an error code if one value is beyond authorized range */ +size_t ZSTD_checkCParams(ZSTD_compressionParameters cParams) +{ +# define CLAMPCHECK(val,min,max) { if ((valmax)) return ERROR(compressionParameter_unsupported); } + CLAMPCHECK(cParams.windowLog, ZSTD_WINDOWLOG_MIN, ZSTD_WINDOWLOG_MAX); + CLAMPCHECK(cParams.chainLog, ZSTD_CHAINLOG_MIN, ZSTD_CHAINLOG_MAX); + CLAMPCHECK(cParams.hashLog, ZSTD_HASHLOG_MIN, ZSTD_HASHLOG_MAX); + CLAMPCHECK(cParams.searchLog, ZSTD_SEARCHLOG_MIN, ZSTD_SEARCHLOG_MAX); + CLAMPCHECK(cParams.searchLength, ZSTD_SEARCHLENGTH_MIN, ZSTD_SEARCHLENGTH_MAX); + CLAMPCHECK(cParams.targetLength, ZSTD_TARGETLENGTH_MIN, ZSTD_TARGETLENGTH_MAX); + if ((U32)(cParams.strategy) > (U32)ZSTD_btopt2) return ERROR(compressionParameter_unsupported); + return 0; +} + + +/** ZSTD_cycleLog() : + * condition for correct operation : hashLog > 1 */ +static U32 ZSTD_cycleLog(U32 hashLog, ZSTD_strategy strat) +{ + U32 const btScale = ((U32)strat >= (U32)ZSTD_btlazy2); + return hashLog - btScale; +} + +/** ZSTD_adjustCParams() : + optimize `cPar` for a given input (`srcSize` and `dictSize`). + mostly downsizing to reduce memory consumption and initialization. + Both `srcSize` and `dictSize` are optional (use 0 if unknown), + but if both are 0, no optimization can be done. + Note : cPar is considered validated at this stage. Use ZSTD_checkParams() to ensure that. */ +ZSTD_compressionParameters ZSTD_adjustCParams(ZSTD_compressionParameters cPar, unsigned long long srcSize, size_t dictSize) +{ + if (srcSize+dictSize == 0) return cPar; /* no size information available : no adjustment */ + + /* resize params, to use less memory when necessary */ + { U32 const minSrcSize = (srcSize==0) ? 500 : 0; + U64 const rSize = srcSize + dictSize + minSrcSize; + if (rSize < ((U64)1< srcLog) cPar.windowLog = srcLog; + } } + if (cPar.hashLog > cPar.windowLog) cPar.hashLog = cPar.windowLog; + { U32 const cycleLog = ZSTD_cycleLog(cPar.chainLog, cPar.strategy); + if (cycleLog > cPar.windowLog) cPar.chainLog -= (cycleLog - cPar.windowLog); + } + + if (cPar.windowLog < ZSTD_WINDOWLOG_ABSOLUTEMIN) cPar.windowLog = ZSTD_WINDOWLOG_ABSOLUTEMIN; /* required for frame header */ + + return cPar; +} + + +size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams) +{ + size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << cParams.windowLog); + U32 const divider = (cParams.searchLength==3) ? 3 : 4; + size_t const maxNbSeq = blockSize / divider; + size_t const tokenSpace = blockSize + 11*maxNbSeq; + + size_t const chainSize = (cParams.strategy == ZSTD_fast) ? 0 : (1 << cParams.chainLog); + size_t const hSize = ((size_t)1) << cParams.hashLog; + U32 const hashLog3 = (cParams.searchLength>3) ? 0 : MIN(ZSTD_HASHLOG3_MAX, cParams.windowLog); + size_t const h3Size = ((size_t)1) << hashLog3; + size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); + + size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<nextSrc - cctx->base); + cctx->params = params; + cctx->frameContentSize = frameContentSize; + cctx->lowLimit = end; + cctx->dictLimit = end; + cctx->nextToUpdate = end+1; + cctx->stage = ZSTDcs_init; + cctx->dictID = 0; + cctx->loadedDictEnd = 0; + { int i; for (i=0; irep[i] = repStartValue[i]; } + cctx->seqStore.litLengthSum = 0; /* force reset of btopt stats */ + XXH64_reset(&cctx->xxhState, 0); + return 0; +} + +typedef enum { ZSTDcrp_continue, ZSTDcrp_noMemset, ZSTDcrp_fullReset } ZSTD_compResetPolicy_e; + +/*! ZSTD_resetCCtx_advanced() : + note : `params` must be validated */ +static size_t ZSTD_resetCCtx_advanced (ZSTD_CCtx* zc, + ZSTD_parameters params, U64 frameContentSize, + ZSTD_compResetPolicy_e const crp) +{ + if (crp == ZSTDcrp_continue) + if (ZSTD_equivalentParams(params, zc->params)) { + zc->flagStaticTables = 0; + zc->flagStaticHufTable = HUF_repeat_none; + return ZSTD_continueCCtx(zc, params, frameContentSize); + } + + { size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << params.cParams.windowLog); + U32 const divider = (params.cParams.searchLength==3) ? 3 : 4; + size_t const maxNbSeq = blockSize / divider; + size_t const tokenSpace = blockSize + 11*maxNbSeq; + size_t const chainSize = (params.cParams.strategy == ZSTD_fast) ? 0 : (1 << params.cParams.chainLog); + size_t const hSize = ((size_t)1) << params.cParams.hashLog; + U32 const hashLog3 = (params.cParams.searchLength>3) ? 0 : MIN(ZSTD_HASHLOG3_MAX, params.cParams.windowLog); + size_t const h3Size = ((size_t)1) << hashLog3; + size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); + void* ptr; + + /* Check if workSpace is large enough, alloc a new one if needed */ + { size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<workSpaceSize < neededSpace) { + ZSTD_free(zc->workSpace, zc->customMem); + zc->workSpace = ZSTD_malloc(neededSpace, zc->customMem); + if (zc->workSpace == NULL) return ERROR(memory_allocation); + zc->workSpaceSize = neededSpace; + } } + + if (crp!=ZSTDcrp_noMemset) memset(zc->workSpace, 0, tableSpace); /* reset tables only */ + XXH64_reset(&zc->xxhState, 0); + zc->hashLog3 = hashLog3; + zc->hashTable = (U32*)(zc->workSpace); + zc->chainTable = zc->hashTable + hSize; + zc->hashTable3 = zc->chainTable + chainSize; + ptr = zc->hashTable3 + h3Size; + zc->hufTable = (HUF_CElt*)ptr; + zc->flagStaticTables = 0; + zc->flagStaticHufTable = HUF_repeat_none; + ptr = ((U32*)ptr) + 256; /* note : HUF_CElt* is incomplete type, size is simulated using U32 */ + + zc->nextToUpdate = 1; + zc->nextSrc = NULL; + zc->base = NULL; + zc->dictBase = NULL; + zc->dictLimit = 0; + zc->lowLimit = 0; + zc->params = params; + zc->blockSize = blockSize; + zc->frameContentSize = frameContentSize; + { int i; for (i=0; irep[i] = repStartValue[i]; } + + if ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) { + zc->seqStore.litFreq = (U32*)ptr; + zc->seqStore.litLengthFreq = zc->seqStore.litFreq + (1<seqStore.matchLengthFreq = zc->seqStore.litLengthFreq + (MaxLL+1); + zc->seqStore.offCodeFreq = zc->seqStore.matchLengthFreq + (MaxML+1); + ptr = zc->seqStore.offCodeFreq + (MaxOff+1); + zc->seqStore.matchTable = (ZSTD_match_t*)ptr; + ptr = zc->seqStore.matchTable + ZSTD_OPT_NUM+1; + zc->seqStore.priceTable = (ZSTD_optimal_t*)ptr; + ptr = zc->seqStore.priceTable + ZSTD_OPT_NUM+1; + zc->seqStore.litLengthSum = 0; + } + zc->seqStore.sequencesStart = (seqDef*)ptr; + ptr = zc->seqStore.sequencesStart + maxNbSeq; + zc->seqStore.llCode = (BYTE*) ptr; + zc->seqStore.mlCode = zc->seqStore.llCode + maxNbSeq; + zc->seqStore.ofCode = zc->seqStore.mlCode + maxNbSeq; + zc->seqStore.litStart = zc->seqStore.ofCode + maxNbSeq; + + zc->stage = ZSTDcs_init; + zc->dictID = 0; + zc->loadedDictEnd = 0; + + return 0; + } +} + +/* ZSTD_invalidateRepCodes() : + * ensures next compression will not use repcodes from previous block. + * Note : only works with regular variant; + * do not use with extDict variant ! */ +void ZSTD_invalidateRepCodes(ZSTD_CCtx* cctx) { + int i; + for (i=0; irep[i] = 0; +} + +/*! ZSTD_copyCCtx() : +* Duplicate an existing context `srcCCtx` into another one `dstCCtx`. +* Only works during stage ZSTDcs_init (i.e. after creation, but before first call to ZSTD_compressContinue()). +* @return : 0, or an error code */ +size_t ZSTD_copyCCtx(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx, unsigned long long pledgedSrcSize) +{ + if (srcCCtx->stage!=ZSTDcs_init) return ERROR(stage_wrong); + + + memcpy(&dstCCtx->customMem, &srcCCtx->customMem, sizeof(ZSTD_customMem)); + { ZSTD_parameters params = srcCCtx->params; + params.fParams.contentSizeFlag = (pledgedSrcSize > 0); + ZSTD_resetCCtx_advanced(dstCCtx, params, pledgedSrcSize, ZSTDcrp_noMemset); + } + + /* copy tables */ + { size_t const chainSize = (srcCCtx->params.cParams.strategy == ZSTD_fast) ? 0 : (1 << srcCCtx->params.cParams.chainLog); + size_t const hSize = ((size_t)1) << srcCCtx->params.cParams.hashLog; + size_t const h3Size = (size_t)1 << srcCCtx->hashLog3; + size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); + memcpy(dstCCtx->workSpace, srcCCtx->workSpace, tableSpace); + } + + /* copy dictionary offsets */ + dstCCtx->nextToUpdate = srcCCtx->nextToUpdate; + dstCCtx->nextToUpdate3= srcCCtx->nextToUpdate3; + dstCCtx->nextSrc = srcCCtx->nextSrc; + dstCCtx->base = srcCCtx->base; + dstCCtx->dictBase = srcCCtx->dictBase; + dstCCtx->dictLimit = srcCCtx->dictLimit; + dstCCtx->lowLimit = srcCCtx->lowLimit; + dstCCtx->loadedDictEnd= srcCCtx->loadedDictEnd; + dstCCtx->dictID = srcCCtx->dictID; + + /* copy entropy tables */ + dstCCtx->flagStaticTables = srcCCtx->flagStaticTables; + dstCCtx->flagStaticHufTable = srcCCtx->flagStaticHufTable; + if (srcCCtx->flagStaticTables) { + memcpy(dstCCtx->litlengthCTable, srcCCtx->litlengthCTable, sizeof(dstCCtx->litlengthCTable)); + memcpy(dstCCtx->matchlengthCTable, srcCCtx->matchlengthCTable, sizeof(dstCCtx->matchlengthCTable)); + memcpy(dstCCtx->offcodeCTable, srcCCtx->offcodeCTable, sizeof(dstCCtx->offcodeCTable)); + } + if (srcCCtx->flagStaticHufTable) { + memcpy(dstCCtx->hufTable, srcCCtx->hufTable, 256*4); + } + + return 0; +} + + +/*! ZSTD_reduceTable() : +* reduce table indexes by `reducerValue` */ +static void ZSTD_reduceTable (U32* const table, U32 const size, U32 const reducerValue) +{ + U32 u; + for (u=0 ; u < size ; u++) { + if (table[u] < reducerValue) table[u] = 0; + else table[u] -= reducerValue; + } +} + +/*! ZSTD_reduceIndex() : +* rescale all indexes to avoid future overflow (indexes are U32) */ +static void ZSTD_reduceIndex (ZSTD_CCtx* zc, const U32 reducerValue) +{ + { U32 const hSize = 1 << zc->params.cParams.hashLog; + ZSTD_reduceTable(zc->hashTable, hSize, reducerValue); } + + { U32 const chainSize = (zc->params.cParams.strategy == ZSTD_fast) ? 0 : (1 << zc->params.cParams.chainLog); + ZSTD_reduceTable(zc->chainTable, chainSize, reducerValue); } + + { U32 const h3Size = (zc->hashLog3) ? 1 << zc->hashLog3 : 0; + ZSTD_reduceTable(zc->hashTable3, h3Size, reducerValue); } +} + + +/*-******************************************************* +* Block entropic compression +*********************************************************/ + +/* See doc/zstd_compression_format.md for detailed format description */ + +size_t ZSTD_noCompressBlock (void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + if (srcSize + ZSTD_blockHeaderSize > dstCapacity) return ERROR(dstSize_tooSmall); + memcpy((BYTE*)dst + ZSTD_blockHeaderSize, src, srcSize); + MEM_writeLE24(dst, (U32)(srcSize << 2) + (U32)bt_raw); + return ZSTD_blockHeaderSize+srcSize; +} + + +static size_t ZSTD_noCompressLiterals (void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + BYTE* const ostart = (BYTE* const)dst; + U32 const flSize = 1 + (srcSize>31) + (srcSize>4095); + + if (srcSize + flSize > dstCapacity) return ERROR(dstSize_tooSmall); + + switch(flSize) + { + case 1: /* 2 - 1 - 5 */ + ostart[0] = (BYTE)((U32)set_basic + (srcSize<<3)); + break; + case 2: /* 2 - 2 - 12 */ + MEM_writeLE16(ostart, (U16)((U32)set_basic + (1<<2) + (srcSize<<4))); + break; + default: /*note : should not be necessary : flSize is within {1,2,3} */ + case 3: /* 2 - 2 - 20 */ + MEM_writeLE32(ostart, (U32)((U32)set_basic + (3<<2) + (srcSize<<4))); + break; + } + + memcpy(ostart + flSize, src, srcSize); + return srcSize + flSize; +} + +static size_t ZSTD_compressRleLiteralsBlock (void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + BYTE* const ostart = (BYTE* const)dst; + U32 const flSize = 1 + (srcSize>31) + (srcSize>4095); + + (void)dstCapacity; /* dstCapacity already guaranteed to be >=4, hence large enough */ + + switch(flSize) + { + case 1: /* 2 - 1 - 5 */ + ostart[0] = (BYTE)((U32)set_rle + (srcSize<<3)); + break; + case 2: /* 2 - 2 - 12 */ + MEM_writeLE16(ostart, (U16)((U32)set_rle + (1<<2) + (srcSize<<4))); + break; + default: /*note : should not be necessary : flSize is necessarily within {1,2,3} */ + case 3: /* 2 - 2 - 20 */ + MEM_writeLE32(ostart, (U32)((U32)set_rle + (3<<2) + (srcSize<<4))); + break; + } + + ostart[flSize] = *(const BYTE*)src; + return flSize+1; +} + + +static size_t ZSTD_minGain(size_t srcSize) { return (srcSize >> 6) + 2; } + +static size_t ZSTD_compressLiterals (ZSTD_CCtx* zc, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize) +{ + size_t const minGain = ZSTD_minGain(srcSize); + size_t const lhSize = 3 + (srcSize >= 1 KB) + (srcSize >= 16 KB); + BYTE* const ostart = (BYTE*)dst; + U32 singleStream = srcSize < 256; + symbolEncodingType_e hType = set_compressed; + size_t cLitSize; + + + /* small ? don't even attempt compression (speed opt) */ +# define LITERAL_NOENTROPY 63 + { size_t const minLitSize = zc->flagStaticHufTable == HUF_repeat_valid ? 6 : LITERAL_NOENTROPY; + if (srcSize <= minLitSize) return ZSTD_noCompressLiterals(dst, dstCapacity, src, srcSize); + } + + if (dstCapacity < lhSize+1) return ERROR(dstSize_tooSmall); /* not enough space for compression */ + { HUF_repeat repeat = zc->flagStaticHufTable; + int const preferRepeat = zc->params.cParams.strategy < ZSTD_lazy ? srcSize <= 1024 : 0; + if (repeat == HUF_repeat_valid && lhSize == 3) singleStream = 1; + cLitSize = singleStream ? HUF_compress1X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11, zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat) + : HUF_compress4X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11, zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat); + if (repeat != HUF_repeat_none) { hType = set_repeat; } /* reused the existing table */ + else { zc->flagStaticHufTable = HUF_repeat_check; } /* now have a table to reuse */ + } + + if ((cLitSize==0) | (cLitSize >= srcSize - minGain)) { + zc->flagStaticHufTable = HUF_repeat_none; + return ZSTD_noCompressLiterals(dst, dstCapacity, src, srcSize); + } + if (cLitSize==1) { + zc->flagStaticHufTable = HUF_repeat_none; + return ZSTD_compressRleLiteralsBlock(dst, dstCapacity, src, srcSize); + } + + /* Build header */ + switch(lhSize) + { + case 3: /* 2 - 2 - 10 - 10 */ + { U32 const lhc = hType + ((!singleStream) << 2) + ((U32)srcSize<<4) + ((U32)cLitSize<<14); + MEM_writeLE24(ostart, lhc); + break; + } + case 4: /* 2 - 2 - 14 - 14 */ + { U32 const lhc = hType + (2 << 2) + ((U32)srcSize<<4) + ((U32)cLitSize<<18); + MEM_writeLE32(ostart, lhc); + break; + } + default: /* should not be necessary, lhSize is only {3,4,5} */ + case 5: /* 2 - 2 - 18 - 18 */ + { U32 const lhc = hType + (3 << 2) + ((U32)srcSize<<4) + ((U32)cLitSize<<22); + MEM_writeLE32(ostart, lhc); + ostart[4] = (BYTE)(cLitSize >> 10); + break; + } + } + return lhSize+cLitSize; +} + +static const BYTE LL_Code[64] = { 0, 1, 2, 3, 4, 5, 6, 7, + 8, 9, 10, 11, 12, 13, 14, 15, + 16, 16, 17, 17, 18, 18, 19, 19, + 20, 20, 20, 20, 21, 21, 21, 21, + 22, 22, 22, 22, 22, 22, 22, 22, + 23, 23, 23, 23, 23, 23, 23, 23, + 24, 24, 24, 24, 24, 24, 24, 24, + 24, 24, 24, 24, 24, 24, 24, 24 }; + +static const BYTE ML_Code[128] = { 0, 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, 32, 33, 33, 34, 34, 35, 35, 36, 36, 36, 36, 37, 37, 37, 37, + 38, 38, 38, 38, 38, 38, 38, 38, 39, 39, 39, 39, 39, 39, 39, 39, + 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, + 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, + 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, + 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42 }; + + +void ZSTD_seqToCodes(const seqStore_t* seqStorePtr) +{ + BYTE const LL_deltaCode = 19; + BYTE const ML_deltaCode = 36; + const seqDef* const sequences = seqStorePtr->sequencesStart; + BYTE* const llCodeTable = seqStorePtr->llCode; + BYTE* const ofCodeTable = seqStorePtr->ofCode; + BYTE* const mlCodeTable = seqStorePtr->mlCode; + U32 const nbSeq = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); + U32 u; + for (u=0; u 63) ? (BYTE)ZSTD_highbit32(llv) + LL_deltaCode : LL_Code[llv]; + ofCodeTable[u] = (BYTE)ZSTD_highbit32(sequences[u].offset); + mlCodeTable[u] = (mlv>127) ? (BYTE)ZSTD_highbit32(mlv) + ML_deltaCode : ML_Code[mlv]; + } + if (seqStorePtr->longLengthID==1) + llCodeTable[seqStorePtr->longLengthPos] = MaxLL; + if (seqStorePtr->longLengthID==2) + mlCodeTable[seqStorePtr->longLengthPos] = MaxML; +} + +MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc, + void* dst, size_t dstCapacity, + size_t srcSize) +{ + const int longOffsets = zc->params.cParams.windowLog > STREAM_ACCUMULATOR_MIN; + const seqStore_t* seqStorePtr = &(zc->seqStore); + U32 count[MaxSeq+1]; + S16 norm[MaxSeq+1]; + FSE_CTable* CTable_LitLength = zc->litlengthCTable; + FSE_CTable* CTable_OffsetBits = zc->offcodeCTable; + FSE_CTable* CTable_MatchLength = zc->matchlengthCTable; + U32 LLtype, Offtype, MLtype; /* compressed, raw or rle */ + const seqDef* const sequences = seqStorePtr->sequencesStart; + const BYTE* const ofCodeTable = seqStorePtr->ofCode; + const BYTE* const llCodeTable = seqStorePtr->llCode; + const BYTE* const mlCodeTable = seqStorePtr->mlCode; + BYTE* const ostart = (BYTE*)dst; + BYTE* const oend = ostart + dstCapacity; + BYTE* op = ostart; + size_t const nbSeq = seqStorePtr->sequences - seqStorePtr->sequencesStart; + BYTE* seqHead; + BYTE scratchBuffer[1<litStart; + size_t const litSize = seqStorePtr->lit - literals; + size_t const cSize = ZSTD_compressLiterals(zc, op, dstCapacity, literals, litSize); + if (ZSTD_isError(cSize)) return cSize; + op += cSize; + } + + /* Sequences Header */ + if ((oend-op) < 3 /*max nbSeq Size*/ + 1 /*seqHead */) return ERROR(dstSize_tooSmall); + if (nbSeq < 0x7F) *op++ = (BYTE)nbSeq; + else if (nbSeq < LONGNBSEQ) op[0] = (BYTE)((nbSeq>>8) + 0x80), op[1] = (BYTE)nbSeq, op+=2; + else op[0]=0xFF, MEM_writeLE16(op+1, (U16)(nbSeq - LONGNBSEQ)), op+=3; + if (nbSeq==0) goto _check_compressibility; + + /* seqHead : flags for FSE encoding type */ + seqHead = op++; + +#define MIN_SEQ_FOR_DYNAMIC_FSE 64 +#define MAX_SEQ_FOR_STATIC_FSE 1000 + + /* convert length/distances into codes */ + ZSTD_seqToCodes(seqStorePtr); + + /* CTable for Literal Lengths */ + { U32 max = MaxLL; + size_t const mostFrequent = FSE_countFast_wksp(count, &max, llCodeTable, nbSeq, zc->tmpCounters); + if ((mostFrequent == nbSeq) && (nbSeq > 2)) { + *op++ = llCodeTable[0]; + FSE_buildCTable_rle(CTable_LitLength, (BYTE)max); + LLtype = set_rle; + } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) { + LLtype = set_repeat; + } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (LL_defaultNormLog-1)))) { + FSE_buildCTable_wksp(CTable_LitLength, LL_defaultNorm, MaxLL, LL_defaultNormLog, scratchBuffer, sizeof(scratchBuffer)); + LLtype = set_basic; + } else { + size_t nbSeq_1 = nbSeq; + const U32 tableLog = FSE_optimalTableLog(LLFSELog, nbSeq, max); + if (count[llCodeTable[nbSeq-1]]>1) { count[llCodeTable[nbSeq-1]]--; nbSeq_1--; } + FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); + { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ + if (FSE_isError(NCountSize)) return NCountSize; + op += NCountSize; } + FSE_buildCTable_wksp(CTable_LitLength, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); + LLtype = set_compressed; + } } + + /* CTable for Offsets */ + { U32 max = MaxOff; + size_t const mostFrequent = FSE_countFast_wksp(count, &max, ofCodeTable, nbSeq, zc->tmpCounters); + if ((mostFrequent == nbSeq) && (nbSeq > 2)) { + *op++ = ofCodeTable[0]; + FSE_buildCTable_rle(CTable_OffsetBits, (BYTE)max); + Offtype = set_rle; + } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) { + Offtype = set_repeat; + } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (OF_defaultNormLog-1)))) { + FSE_buildCTable_wksp(CTable_OffsetBits, OF_defaultNorm, MaxOff, OF_defaultNormLog, scratchBuffer, sizeof(scratchBuffer)); + Offtype = set_basic; + } else { + size_t nbSeq_1 = nbSeq; + const U32 tableLog = FSE_optimalTableLog(OffFSELog, nbSeq, max); + if (count[ofCodeTable[nbSeq-1]]>1) { count[ofCodeTable[nbSeq-1]]--; nbSeq_1--; } + FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); + { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ + if (FSE_isError(NCountSize)) return NCountSize; + op += NCountSize; } + FSE_buildCTable_wksp(CTable_OffsetBits, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); + Offtype = set_compressed; + } } + + /* CTable for MatchLengths */ + { U32 max = MaxML; + size_t const mostFrequent = FSE_countFast_wksp(count, &max, mlCodeTable, nbSeq, zc->tmpCounters); + if ((mostFrequent == nbSeq) && (nbSeq > 2)) { + *op++ = *mlCodeTable; + FSE_buildCTable_rle(CTable_MatchLength, (BYTE)max); + MLtype = set_rle; + } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) { + MLtype = set_repeat; + } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (ML_defaultNormLog-1)))) { + FSE_buildCTable_wksp(CTable_MatchLength, ML_defaultNorm, MaxML, ML_defaultNormLog, scratchBuffer, sizeof(scratchBuffer)); + MLtype = set_basic; + } else { + size_t nbSeq_1 = nbSeq; + const U32 tableLog = FSE_optimalTableLog(MLFSELog, nbSeq, max); + if (count[mlCodeTable[nbSeq-1]]>1) { count[mlCodeTable[nbSeq-1]]--; nbSeq_1--; } + FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); + { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ + if (FSE_isError(NCountSize)) return NCountSize; + op += NCountSize; } + FSE_buildCTable_wksp(CTable_MatchLength, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); + MLtype = set_compressed; + } } + + *seqHead = (BYTE)((LLtype<<6) + (Offtype<<4) + (MLtype<<2)); + zc->flagStaticTables = 0; + + /* Encoding Sequences */ + { BIT_CStream_t blockStream; + FSE_CState_t stateMatchLength; + FSE_CState_t stateOffsetBits; + FSE_CState_t stateLitLength; + + CHECK_E(BIT_initCStream(&blockStream, op, oend-op), dstSize_tooSmall); /* not enough space remaining */ + + /* first symbols */ + FSE_initCState2(&stateMatchLength, CTable_MatchLength, mlCodeTable[nbSeq-1]); + FSE_initCState2(&stateOffsetBits, CTable_OffsetBits, ofCodeTable[nbSeq-1]); + FSE_initCState2(&stateLitLength, CTable_LitLength, llCodeTable[nbSeq-1]); + BIT_addBits(&blockStream, sequences[nbSeq-1].litLength, LL_bits[llCodeTable[nbSeq-1]]); + if (MEM_32bits()) BIT_flushBits(&blockStream); + BIT_addBits(&blockStream, sequences[nbSeq-1].matchLength, ML_bits[mlCodeTable[nbSeq-1]]); + if (MEM_32bits()) BIT_flushBits(&blockStream); + if (longOffsets) { + U32 const ofBits = ofCodeTable[nbSeq-1]; + int const extraBits = ofBits - MIN(ofBits, STREAM_ACCUMULATOR_MIN-1); + if (extraBits) { + BIT_addBits(&blockStream, sequences[nbSeq-1].offset, extraBits); + BIT_flushBits(&blockStream); + } + BIT_addBits(&blockStream, sequences[nbSeq-1].offset >> extraBits, + ofBits - extraBits); + } else { + BIT_addBits(&blockStream, sequences[nbSeq-1].offset, ofCodeTable[nbSeq-1]); + } + BIT_flushBits(&blockStream); + + { size_t n; + for (n=nbSeq-2 ; n= 64-7-(LLFSELog+MLFSELog+OffFSELog))) + BIT_flushBits(&blockStream); /* (7)*/ + BIT_addBits(&blockStream, sequences[n].litLength, llBits); + if (MEM_32bits() && ((llBits+mlBits)>24)) BIT_flushBits(&blockStream); + BIT_addBits(&blockStream, sequences[n].matchLength, mlBits); + if (MEM_32bits()) BIT_flushBits(&blockStream); /* (7)*/ + if (longOffsets) { + int const extraBits = ofBits - MIN(ofBits, STREAM_ACCUMULATOR_MIN-1); + if (extraBits) { + BIT_addBits(&blockStream, sequences[n].offset, extraBits); + BIT_flushBits(&blockStream); /* (7)*/ + } + BIT_addBits(&blockStream, sequences[n].offset >> extraBits, + ofBits - extraBits); /* 31 */ + } else { + BIT_addBits(&blockStream, sequences[n].offset, ofBits); /* 31 */ + } + BIT_flushBits(&blockStream); /* (7)*/ + } } + + FSE_flushCState(&blockStream, &stateMatchLength); + FSE_flushCState(&blockStream, &stateOffsetBits); + FSE_flushCState(&blockStream, &stateLitLength); + + { size_t const streamSize = BIT_closeCStream(&blockStream); + if (streamSize==0) return ERROR(dstSize_tooSmall); /* not enough space */ + op += streamSize; + } } + + /* check compressibility */ +_check_compressibility: + { size_t const minGain = ZSTD_minGain(srcSize); + size_t const maxCSize = srcSize - minGain; + if ((size_t)(op-ostart) >= maxCSize) { + zc->flagStaticHufTable = HUF_repeat_none; + return 0; + } } + + /* confirm repcodes */ + { int i; for (i=0; irep[i] = zc->repToConfirm[i]; } + + return op - ostart; +} + +#if 0 /* for debug */ +# define STORESEQ_DEBUG +#include /* fprintf */ +U32 g_startDebug = 0; +const BYTE* g_start = NULL; +#endif + +/*! ZSTD_storeSeq() : + Store a sequence (literal length, literals, offset code and match length code) into seqStore_t. + `offsetCode` : distance to match, or 0 == repCode. + `matchCode` : matchLength - MINMATCH +*/ +MEM_STATIC void ZSTD_storeSeq(seqStore_t* seqStorePtr, size_t litLength, const void* literals, U32 offsetCode, size_t matchCode) +{ +#ifdef STORESEQ_DEBUG + if (g_startDebug) { + const U32 pos = (U32)((const BYTE*)literals - g_start); + if (g_start==NULL) g_start = (const BYTE*)literals; + if ((pos > 1895000) && (pos < 1895300)) + fprintf(stderr, "Cpos %6u :%5u literals & match %3u bytes at distance %6u \n", + pos, (U32)litLength, (U32)matchCode+MINMATCH, (U32)offsetCode); + } +#endif + /* copy Literals */ + ZSTD_wildcopy(seqStorePtr->lit, literals, litLength); + seqStorePtr->lit += litLength; + + /* literal Length */ + if (litLength>0xFFFF) { seqStorePtr->longLengthID = 1; seqStorePtr->longLengthPos = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); } + seqStorePtr->sequences[0].litLength = (U16)litLength; + + /* match offset */ + seqStorePtr->sequences[0].offset = offsetCode + 1; + + /* match Length */ + if (matchCode>0xFFFF) { seqStorePtr->longLengthID = 2; seqStorePtr->longLengthPos = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); } + seqStorePtr->sequences[0].matchLength = (U16)matchCode; + + seqStorePtr->sequences++; +} + + +/*-************************************* +* Match length counter +***************************************/ +static unsigned ZSTD_NbCommonBytes (register size_t val) +{ + if (MEM_isLittleEndian()) { + if (MEM_64bits()) { +# if defined(_MSC_VER) && defined(_WIN64) + unsigned long r = 0; + _BitScanForward64( &r, (U64)val ); + return (unsigned)(r>>3); +# elif defined(__GNUC__) && (__GNUC__ >= 3) + return (__builtin_ctzll((U64)val) >> 3); +# else + static const int DeBruijnBytePos[64] = { 0, 0, 0, 0, 0, 1, 1, 2, 0, 3, 1, 3, 1, 4, 2, 7, 0, 2, 3, 6, 1, 5, 3, 5, 1, 3, 4, 4, 2, 5, 6, 7, 7, 0, 1, 2, 3, 3, 4, 6, 2, 6, 5, 5, 3, 4, 5, 6, 7, 1, 2, 4, 6, 4, 4, 5, 7, 2, 6, 5, 7, 6, 7, 7 }; + return DeBruijnBytePos[((U64)((val & -(long long)val) * 0x0218A392CDABBD3FULL)) >> 58]; +# endif + } else { /* 32 bits */ +# if defined(_MSC_VER) + unsigned long r=0; + _BitScanForward( &r, (U32)val ); + return (unsigned)(r>>3); +# elif defined(__GNUC__) && (__GNUC__ >= 3) + return (__builtin_ctz((U32)val) >> 3); +# else + static const int DeBruijnBytePos[32] = { 0, 0, 3, 0, 3, 1, 3, 0, 3, 2, 2, 1, 3, 2, 0, 1, 3, 3, 1, 2, 2, 2, 2, 0, 3, 1, 2, 0, 1, 0, 1, 1 }; + return DeBruijnBytePos[((U32)((val & -(S32)val) * 0x077CB531U)) >> 27]; +# endif + } + } else { /* Big Endian CPU */ + if (MEM_64bits()) { +# if defined(_MSC_VER) && defined(_WIN64) + unsigned long r = 0; + _BitScanReverse64( &r, val ); + return (unsigned)(r>>3); +# elif defined(__GNUC__) && (__GNUC__ >= 3) + return (__builtin_clzll(val) >> 3); +# else + unsigned r; + const unsigned n32 = sizeof(size_t)*4; /* calculate this way due to compiler complaining in 32-bits mode */ + if (!(val>>n32)) { r=4; } else { r=0; val>>=n32; } + if (!(val>>16)) { r+=2; val>>=8; } else { val>>=24; } + r += (!val); + return r; +# endif + } else { /* 32 bits */ +# if defined(_MSC_VER) + unsigned long r = 0; + _BitScanReverse( &r, (unsigned long)val ); + return (unsigned)(r>>3); +# elif defined(__GNUC__) && (__GNUC__ >= 3) + return (__builtin_clz((U32)val) >> 3); +# else + unsigned r; + if (!(val>>16)) { r=2; val>>=8; } else { r=0; val>>=24; } + r += (!val); + return r; +# endif + } } +} + + +static size_t ZSTD_count(const BYTE* pIn, const BYTE* pMatch, const BYTE* const pInLimit) +{ + const BYTE* const pStart = pIn; + const BYTE* const pInLoopLimit = pInLimit - (sizeof(size_t)-1); + + while (pIn < pInLoopLimit) { + size_t const diff = MEM_readST(pMatch) ^ MEM_readST(pIn); + if (!diff) { pIn+=sizeof(size_t); pMatch+=sizeof(size_t); continue; } + pIn += ZSTD_NbCommonBytes(diff); + return (size_t)(pIn - pStart); + } + if (MEM_64bits()) if ((pIn<(pInLimit-3)) && (MEM_read32(pMatch) == MEM_read32(pIn))) { pIn+=4; pMatch+=4; } + if ((pIn<(pInLimit-1)) && (MEM_read16(pMatch) == MEM_read16(pIn))) { pIn+=2; pMatch+=2; } + if ((pIn> (32-h) ; } +MEM_STATIC size_t ZSTD_hash3Ptr(const void* ptr, U32 h) { return ZSTD_hash3(MEM_readLE32(ptr), h); } /* only in zstd_opt.h */ + +static const U32 prime4bytes = 2654435761U; +static U32 ZSTD_hash4(U32 u, U32 h) { return (u * prime4bytes) >> (32-h) ; } +static size_t ZSTD_hash4Ptr(const void* ptr, U32 h) { return ZSTD_hash4(MEM_read32(ptr), h); } + +static const U64 prime5bytes = 889523592379ULL; +static size_t ZSTD_hash5(U64 u, U32 h) { return (size_t)(((u << (64-40)) * prime5bytes) >> (64-h)) ; } +static size_t ZSTD_hash5Ptr(const void* p, U32 h) { return ZSTD_hash5(MEM_readLE64(p), h); } + +static const U64 prime6bytes = 227718039650203ULL; +static size_t ZSTD_hash6(U64 u, U32 h) { return (size_t)(((u << (64-48)) * prime6bytes) >> (64-h)) ; } +static size_t ZSTD_hash6Ptr(const void* p, U32 h) { return ZSTD_hash6(MEM_readLE64(p), h); } + +static const U64 prime7bytes = 58295818150454627ULL; +static size_t ZSTD_hash7(U64 u, U32 h) { return (size_t)(((u << (64-56)) * prime7bytes) >> (64-h)) ; } +static size_t ZSTD_hash7Ptr(const void* p, U32 h) { return ZSTD_hash7(MEM_readLE64(p), h); } + +static const U64 prime8bytes = 0xCF1BBCDCB7A56463ULL; +static size_t ZSTD_hash8(U64 u, U32 h) { return (size_t)(((u) * prime8bytes) >> (64-h)) ; } +static size_t ZSTD_hash8Ptr(const void* p, U32 h) { return ZSTD_hash8(MEM_readLE64(p), h); } + +static size_t ZSTD_hashPtr(const void* p, U32 hBits, U32 mls) +{ + switch(mls) + { + //case 3: return ZSTD_hash3Ptr(p, hBits); + default: + case 4: return ZSTD_hash4Ptr(p, hBits); + case 5: return ZSTD_hash5Ptr(p, hBits); + case 6: return ZSTD_hash6Ptr(p, hBits); + case 7: return ZSTD_hash7Ptr(p, hBits); + case 8: return ZSTD_hash8Ptr(p, hBits); + } +} + + +/*-************************************* +* Fast Scan +***************************************/ +static void ZSTD_fillHashTable (ZSTD_CCtx* zc, const void* end, const U32 mls) +{ + U32* const hashTable = zc->hashTable; + U32 const hBits = zc->params.cParams.hashLog; + const BYTE* const base = zc->base; + const BYTE* ip = base + zc->nextToUpdate; + const BYTE* const iend = ((const BYTE*)end) - HASH_READ_SIZE; + const size_t fastHashFillStep = 3; + + while(ip <= iend) { + hashTable[ZSTD_hashPtr(ip, hBits, mls)] = (U32)(ip - base); + ip += fastHashFillStep; + } +} + + +FORCE_INLINE +void ZSTD_compressBlock_fast_generic(ZSTD_CCtx* cctx, + const void* src, size_t srcSize, + const U32 mls) +{ + U32* const hashTable = cctx->hashTable; + U32 const hBits = cctx->params.cParams.hashLog; + seqStore_t* seqStorePtr = &(cctx->seqStore); + const BYTE* const base = cctx->base; + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const U32 lowestIndex = cctx->dictLimit; + const BYTE* const lowest = base + lowestIndex; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - HASH_READ_SIZE; + U32 offset_1=cctx->rep[0], offset_2=cctx->rep[1]; + U32 offsetSaved = 0; + + /* init */ + ip += (ip==lowest); + { U32 const maxRep = (U32)(ip-lowest); + if (offset_2 > maxRep) offsetSaved = offset_2, offset_2 = 0; + if (offset_1 > maxRep) offsetSaved = offset_1, offset_1 = 0; + } + + /* Main Search Loop */ + while (ip < ilimit) { /* < instead of <=, because repcode check at (ip+1) */ + size_t mLength; + size_t const h = ZSTD_hashPtr(ip, hBits, mls); + U32 const current = (U32)(ip-base); + U32 const matchIndex = hashTable[h]; + const BYTE* match = base + matchIndex; + hashTable[h] = current; /* update hash table */ + + if ((offset_1 > 0) & (MEM_read32(ip+1-offset_1) == MEM_read32(ip+1))) { + mLength = ZSTD_count(ip+1+4, ip+1+4-offset_1, iend) + 4; + ip++; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); + } else { + U32 offset; + if ( (matchIndex <= lowestIndex) || (MEM_read32(match) != MEM_read32(ip)) ) { + ip += ((ip-anchor) >> g_searchStrength) + 1; + continue; + } + mLength = ZSTD_count(ip+4, match+4, iend) + 4; + offset = (U32)(ip-match); + while (((ip>anchor) & (match>lowest)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ + offset_2 = offset_1; + offset_1 = offset; + + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); + } + + /* match found */ + ip += mLength; + anchor = ip; + + if (ip <= ilimit) { + /* Fill Table */ + hashTable[ZSTD_hashPtr(base+current+2, hBits, mls)] = current+2; /* here because current+2 could be > iend-8 */ + hashTable[ZSTD_hashPtr(ip-2, hBits, mls)] = (U32)(ip-2-base); + /* check immediate repcode */ + while ( (ip <= ilimit) + && ( (offset_2>0) + & (MEM_read32(ip) == MEM_read32(ip - offset_2)) )) { + /* store sequence */ + size_t const rLength = ZSTD_count(ip+4, ip+4-offset_2, iend) + 4; + { U32 const tmpOff = offset_2; offset_2 = offset_1; offset_1 = tmpOff; } /* swap offset_2 <=> offset_1 */ + hashTable[ZSTD_hashPtr(ip, hBits, mls)] = (U32)(ip-base); + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, rLength-MINMATCH); + ip += rLength; + anchor = ip; + continue; /* faster when present ... (?) */ + } } } + + /* save reps for next block */ + cctx->repToConfirm[0] = offset_1 ? offset_1 : offsetSaved; + cctx->repToConfirm[1] = offset_2 ? offset_2 : offsetSaved; + + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } +} + + +static void ZSTD_compressBlock_fast(ZSTD_CCtx* ctx, + const void* src, size_t srcSize) +{ + const U32 mls = ctx->params.cParams.searchLength; + switch(mls) + { + default: /* includes case 3 */ + case 4 : + ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 4); return; + case 5 : + ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 5); return; + case 6 : + ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 6); return; + case 7 : + ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 7); return; + } +} + + +static void ZSTD_compressBlock_fast_extDict_generic(ZSTD_CCtx* ctx, + const void* src, size_t srcSize, + const U32 mls) +{ + U32* hashTable = ctx->hashTable; + const U32 hBits = ctx->params.cParams.hashLog; + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const base = ctx->base; + const BYTE* const dictBase = ctx->dictBase; + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const U32 lowestIndex = ctx->lowLimit; + const BYTE* const dictStart = dictBase + lowestIndex; + const U32 dictLimit = ctx->dictLimit; + const BYTE* const lowPrefixPtr = base + dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + U32 offset_1=ctx->rep[0], offset_2=ctx->rep[1]; + + /* Search Loop */ + while (ip < ilimit) { /* < instead of <=, because (ip+1) */ + const size_t h = ZSTD_hashPtr(ip, hBits, mls); + const U32 matchIndex = hashTable[h]; + const BYTE* matchBase = matchIndex < dictLimit ? dictBase : base; + const BYTE* match = matchBase + matchIndex; + const U32 current = (U32)(ip-base); + const U32 repIndex = current + 1 - offset_1; /* offset_1 expected <= current +1 */ + const BYTE* repBase = repIndex < dictLimit ? dictBase : base; + const BYTE* repMatch = repBase + repIndex; + size_t mLength; + hashTable[h] = current; /* update hash table */ + + if ( (((U32)((dictLimit-1) - repIndex) >= 3) /* intentional underflow */ & (repIndex > lowestIndex)) + && (MEM_read32(repMatch) == MEM_read32(ip+1)) ) { + const BYTE* repMatchEnd = repIndex < dictLimit ? dictEnd : iend; + mLength = ZSTD_count_2segments(ip+1+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repMatchEnd, lowPrefixPtr) + EQUAL_READ32; + ip++; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); + } else { + if ( (matchIndex < lowestIndex) || + (MEM_read32(match) != MEM_read32(ip)) ) { + ip += ((ip-anchor) >> g_searchStrength) + 1; + continue; + } + { const BYTE* matchEnd = matchIndex < dictLimit ? dictEnd : iend; + const BYTE* lowMatchPtr = matchIndex < dictLimit ? dictStart : lowPrefixPtr; + U32 offset; + mLength = ZSTD_count_2segments(ip+EQUAL_READ32, match+EQUAL_READ32, iend, matchEnd, lowPrefixPtr) + EQUAL_READ32; + while (((ip>anchor) & (match>lowMatchPtr)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ + offset = current - matchIndex; + offset_2 = offset_1; + offset_1 = offset; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); + } } + + /* found a match : store it */ + ip += mLength; + anchor = ip; + + if (ip <= ilimit) { + /* Fill Table */ + hashTable[ZSTD_hashPtr(base+current+2, hBits, mls)] = current+2; + hashTable[ZSTD_hashPtr(ip-2, hBits, mls)] = (U32)(ip-2-base); + /* check immediate repcode */ + while (ip <= ilimit) { + U32 const current2 = (U32)(ip-base); + U32 const repIndex2 = current2 - offset_2; + const BYTE* repMatch2 = repIndex2 < dictLimit ? dictBase + repIndex2 : base + repIndex2; + if ( (((U32)((dictLimit-1) - repIndex2) >= 3) & (repIndex2 > lowestIndex)) /* intentional overflow */ + && (MEM_read32(repMatch2) == MEM_read32(ip)) ) { + const BYTE* const repEnd2 = repIndex2 < dictLimit ? dictEnd : iend; + size_t repLength2 = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch2+EQUAL_READ32, iend, repEnd2, lowPrefixPtr) + EQUAL_READ32; + U32 tmpOffset = offset_2; offset_2 = offset_1; offset_1 = tmpOffset; /* swap offset_2 <=> offset_1 */ + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, repLength2-MINMATCH); + hashTable[ZSTD_hashPtr(ip, hBits, mls)] = current2; + ip += repLength2; + anchor = ip; + continue; + } + break; + } } } + + /* save reps for next block */ + ctx->repToConfirm[0] = offset_1; ctx->repToConfirm[1] = offset_2; + + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } +} + + +static void ZSTD_compressBlock_fast_extDict(ZSTD_CCtx* ctx, + const void* src, size_t srcSize) +{ + U32 const mls = ctx->params.cParams.searchLength; + switch(mls) + { + default: /* includes case 3 */ + case 4 : + ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 4); return; + case 5 : + ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 5); return; + case 6 : + ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 6); return; + case 7 : + ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 7); return; + } +} + + +/*-************************************* +* Double Fast +***************************************/ +static void ZSTD_fillDoubleHashTable (ZSTD_CCtx* cctx, const void* end, const U32 mls) +{ + U32* const hashLarge = cctx->hashTable; + U32 const hBitsL = cctx->params.cParams.hashLog; + U32* const hashSmall = cctx->chainTable; + U32 const hBitsS = cctx->params.cParams.chainLog; + const BYTE* const base = cctx->base; + const BYTE* ip = base + cctx->nextToUpdate; + const BYTE* const iend = ((const BYTE*)end) - HASH_READ_SIZE; + const size_t fastHashFillStep = 3; + + while(ip <= iend) { + hashSmall[ZSTD_hashPtr(ip, hBitsS, mls)] = (U32)(ip - base); + hashLarge[ZSTD_hashPtr(ip, hBitsL, 8)] = (U32)(ip - base); + ip += fastHashFillStep; + } +} + + +FORCE_INLINE +void ZSTD_compressBlock_doubleFast_generic(ZSTD_CCtx* cctx, + const void* src, size_t srcSize, + const U32 mls) +{ + U32* const hashLong = cctx->hashTable; + const U32 hBitsL = cctx->params.cParams.hashLog; + U32* const hashSmall = cctx->chainTable; + const U32 hBitsS = cctx->params.cParams.chainLog; + seqStore_t* seqStorePtr = &(cctx->seqStore); + const BYTE* const base = cctx->base; + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const U32 lowestIndex = cctx->dictLimit; + const BYTE* const lowest = base + lowestIndex; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - HASH_READ_SIZE; + U32 offset_1=cctx->rep[0], offset_2=cctx->rep[1]; + U32 offsetSaved = 0; + + /* init */ + ip += (ip==lowest); + { U32 const maxRep = (U32)(ip-lowest); + if (offset_2 > maxRep) offsetSaved = offset_2, offset_2 = 0; + if (offset_1 > maxRep) offsetSaved = offset_1, offset_1 = 0; + } + + /* Main Search Loop */ + while (ip < ilimit) { /* < instead of <=, because repcode check at (ip+1) */ + size_t mLength; + size_t const h2 = ZSTD_hashPtr(ip, hBitsL, 8); + size_t const h = ZSTD_hashPtr(ip, hBitsS, mls); + U32 const current = (U32)(ip-base); + U32 const matchIndexL = hashLong[h2]; + U32 const matchIndexS = hashSmall[h]; + const BYTE* matchLong = base + matchIndexL; + const BYTE* match = base + matchIndexS; + hashLong[h2] = hashSmall[h] = current; /* update hash tables */ + + if ((offset_1 > 0) & (MEM_read32(ip+1-offset_1) == MEM_read32(ip+1))) { /* note : by construction, offset_1 <= current */ + mLength = ZSTD_count(ip+1+4, ip+1+4-offset_1, iend) + 4; + ip++; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); + } else { + U32 offset; + if ( (matchIndexL > lowestIndex) && (MEM_read64(matchLong) == MEM_read64(ip)) ) { + mLength = ZSTD_count(ip+8, matchLong+8, iend) + 8; + offset = (U32)(ip-matchLong); + while (((ip>anchor) & (matchLong>lowest)) && (ip[-1] == matchLong[-1])) { ip--; matchLong--; mLength++; } /* catch up */ + } else if ( (matchIndexS > lowestIndex) && (MEM_read32(match) == MEM_read32(ip)) ) { + size_t const h3 = ZSTD_hashPtr(ip+1, hBitsL, 8); + U32 const matchIndex3 = hashLong[h3]; + const BYTE* match3 = base + matchIndex3; + hashLong[h3] = current + 1; + if ( (matchIndex3 > lowestIndex) && (MEM_read64(match3) == MEM_read64(ip+1)) ) { + mLength = ZSTD_count(ip+9, match3+8, iend) + 8; + ip++; + offset = (U32)(ip-match3); + while (((ip>anchor) & (match3>lowest)) && (ip[-1] == match3[-1])) { ip--; match3--; mLength++; } /* catch up */ + } else { + mLength = ZSTD_count(ip+4, match+4, iend) + 4; + offset = (U32)(ip-match); + while (((ip>anchor) & (match>lowest)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ + } + } else { + ip += ((ip-anchor) >> g_searchStrength) + 1; + continue; + } + + offset_2 = offset_1; + offset_1 = offset; + + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); + } + + /* match found */ + ip += mLength; + anchor = ip; + + if (ip <= ilimit) { + /* Fill Table */ + hashLong[ZSTD_hashPtr(base+current+2, hBitsL, 8)] = + hashSmall[ZSTD_hashPtr(base+current+2, hBitsS, mls)] = current+2; /* here because current+2 could be > iend-8 */ + hashLong[ZSTD_hashPtr(ip-2, hBitsL, 8)] = + hashSmall[ZSTD_hashPtr(ip-2, hBitsS, mls)] = (U32)(ip-2-base); + + /* check immediate repcode */ + while ( (ip <= ilimit) + && ( (offset_2>0) + & (MEM_read32(ip) == MEM_read32(ip - offset_2)) )) { + /* store sequence */ + size_t const rLength = ZSTD_count(ip+4, ip+4-offset_2, iend) + 4; + { U32 const tmpOff = offset_2; offset_2 = offset_1; offset_1 = tmpOff; } /* swap offset_2 <=> offset_1 */ + hashSmall[ZSTD_hashPtr(ip, hBitsS, mls)] = (U32)(ip-base); + hashLong[ZSTD_hashPtr(ip, hBitsL, 8)] = (U32)(ip-base); + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, rLength-MINMATCH); + ip += rLength; + anchor = ip; + continue; /* faster when present ... (?) */ + } } } + + /* save reps for next block */ + cctx->repToConfirm[0] = offset_1 ? offset_1 : offsetSaved; + cctx->repToConfirm[1] = offset_2 ? offset_2 : offsetSaved; + + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } +} + + +static void ZSTD_compressBlock_doubleFast(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ + const U32 mls = ctx->params.cParams.searchLength; + switch(mls) + { + default: /* includes case 3 */ + case 4 : + ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 4); return; + case 5 : + ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 5); return; + case 6 : + ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 6); return; + case 7 : + ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 7); return; + } +} + + +static void ZSTD_compressBlock_doubleFast_extDict_generic(ZSTD_CCtx* ctx, + const void* src, size_t srcSize, + const U32 mls) +{ + U32* const hashLong = ctx->hashTable; + U32 const hBitsL = ctx->params.cParams.hashLog; + U32* const hashSmall = ctx->chainTable; + U32 const hBitsS = ctx->params.cParams.chainLog; + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const base = ctx->base; + const BYTE* const dictBase = ctx->dictBase; + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const U32 lowestIndex = ctx->lowLimit; + const BYTE* const dictStart = dictBase + lowestIndex; + const U32 dictLimit = ctx->dictLimit; + const BYTE* const lowPrefixPtr = base + dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + U32 offset_1=ctx->rep[0], offset_2=ctx->rep[1]; + + /* Search Loop */ + while (ip < ilimit) { /* < instead of <=, because (ip+1) */ + const size_t hSmall = ZSTD_hashPtr(ip, hBitsS, mls); + const U32 matchIndex = hashSmall[hSmall]; + const BYTE* matchBase = matchIndex < dictLimit ? dictBase : base; + const BYTE* match = matchBase + matchIndex; + + const size_t hLong = ZSTD_hashPtr(ip, hBitsL, 8); + const U32 matchLongIndex = hashLong[hLong]; + const BYTE* matchLongBase = matchLongIndex < dictLimit ? dictBase : base; + const BYTE* matchLong = matchLongBase + matchLongIndex; + + const U32 current = (U32)(ip-base); + const U32 repIndex = current + 1 - offset_1; /* offset_1 expected <= current +1 */ + const BYTE* repBase = repIndex < dictLimit ? dictBase : base; + const BYTE* repMatch = repBase + repIndex; + size_t mLength; + hashSmall[hSmall] = hashLong[hLong] = current; /* update hash table */ + + if ( (((U32)((dictLimit-1) - repIndex) >= 3) /* intentional underflow */ & (repIndex > lowestIndex)) + && (MEM_read32(repMatch) == MEM_read32(ip+1)) ) { + const BYTE* repMatchEnd = repIndex < dictLimit ? dictEnd : iend; + mLength = ZSTD_count_2segments(ip+1+4, repMatch+4, iend, repMatchEnd, lowPrefixPtr) + 4; + ip++; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); + } else { + if ((matchLongIndex > lowestIndex) && (MEM_read64(matchLong) == MEM_read64(ip))) { + const BYTE* matchEnd = matchLongIndex < dictLimit ? dictEnd : iend; + const BYTE* lowMatchPtr = matchLongIndex < dictLimit ? dictStart : lowPrefixPtr; + U32 offset; + mLength = ZSTD_count_2segments(ip+8, matchLong+8, iend, matchEnd, lowPrefixPtr) + 8; + offset = current - matchLongIndex; + while (((ip>anchor) & (matchLong>lowMatchPtr)) && (ip[-1] == matchLong[-1])) { ip--; matchLong--; mLength++; } /* catch up */ + offset_2 = offset_1; + offset_1 = offset; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); + + } else if ((matchIndex > lowestIndex) && (MEM_read32(match) == MEM_read32(ip))) { + size_t const h3 = ZSTD_hashPtr(ip+1, hBitsL, 8); + U32 const matchIndex3 = hashLong[h3]; + const BYTE* const match3Base = matchIndex3 < dictLimit ? dictBase : base; + const BYTE* match3 = match3Base + matchIndex3; + U32 offset; + hashLong[h3] = current + 1; + if ( (matchIndex3 > lowestIndex) && (MEM_read64(match3) == MEM_read64(ip+1)) ) { + const BYTE* matchEnd = matchIndex3 < dictLimit ? dictEnd : iend; + const BYTE* lowMatchPtr = matchIndex3 < dictLimit ? dictStart : lowPrefixPtr; + mLength = ZSTD_count_2segments(ip+9, match3+8, iend, matchEnd, lowPrefixPtr) + 8; + ip++; + offset = current+1 - matchIndex3; + while (((ip>anchor) & (match3>lowMatchPtr)) && (ip[-1] == match3[-1])) { ip--; match3--; mLength++; } /* catch up */ + } else { + const BYTE* matchEnd = matchIndex < dictLimit ? dictEnd : iend; + const BYTE* lowMatchPtr = matchIndex < dictLimit ? dictStart : lowPrefixPtr; + mLength = ZSTD_count_2segments(ip+4, match+4, iend, matchEnd, lowPrefixPtr) + 4; + offset = current - matchIndex; + while (((ip>anchor) & (match>lowMatchPtr)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ + } + offset_2 = offset_1; + offset_1 = offset; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); + + } else { + ip += ((ip-anchor) >> g_searchStrength) + 1; + continue; + } } + + /* found a match : store it */ + ip += mLength; + anchor = ip; + + if (ip <= ilimit) { + /* Fill Table */ + hashSmall[ZSTD_hashPtr(base+current+2, hBitsS, mls)] = current+2; + hashLong[ZSTD_hashPtr(base+current+2, hBitsL, 8)] = current+2; + hashSmall[ZSTD_hashPtr(ip-2, hBitsS, mls)] = (U32)(ip-2-base); + hashLong[ZSTD_hashPtr(ip-2, hBitsL, 8)] = (U32)(ip-2-base); + /* check immediate repcode */ + while (ip <= ilimit) { + U32 const current2 = (U32)(ip-base); + U32 const repIndex2 = current2 - offset_2; + const BYTE* repMatch2 = repIndex2 < dictLimit ? dictBase + repIndex2 : base + repIndex2; + if ( (((U32)((dictLimit-1) - repIndex2) >= 3) & (repIndex2 > lowestIndex)) /* intentional overflow */ + && (MEM_read32(repMatch2) == MEM_read32(ip)) ) { + const BYTE* const repEnd2 = repIndex2 < dictLimit ? dictEnd : iend; + size_t const repLength2 = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch2+EQUAL_READ32, iend, repEnd2, lowPrefixPtr) + EQUAL_READ32; + U32 tmpOffset = offset_2; offset_2 = offset_1; offset_1 = tmpOffset; /* swap offset_2 <=> offset_1 */ + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, repLength2-MINMATCH); + hashSmall[ZSTD_hashPtr(ip, hBitsS, mls)] = current2; + hashLong[ZSTD_hashPtr(ip, hBitsL, 8)] = current2; + ip += repLength2; + anchor = ip; + continue; + } + break; + } } } + + /* save reps for next block */ + ctx->repToConfirm[0] = offset_1; ctx->repToConfirm[1] = offset_2; + + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } +} + + +static void ZSTD_compressBlock_doubleFast_extDict(ZSTD_CCtx* ctx, + const void* src, size_t srcSize) +{ + U32 const mls = ctx->params.cParams.searchLength; + switch(mls) + { + default: /* includes case 3 */ + case 4 : + ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 4); return; + case 5 : + ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 5); return; + case 6 : + ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 6); return; + case 7 : + ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 7); return; + } +} + + +/*-************************************* +* Binary Tree search +***************************************/ +/** ZSTD_insertBt1() : add one or multiple positions to tree. +* ip : assumed <= iend-8 . +* @return : nb of positions added */ +static U32 ZSTD_insertBt1(ZSTD_CCtx* zc, const BYTE* const ip, const U32 mls, const BYTE* const iend, U32 nbCompares, + U32 extDict) +{ + U32* const hashTable = zc->hashTable; + U32 const hashLog = zc->params.cParams.hashLog; + size_t const h = ZSTD_hashPtr(ip, hashLog, mls); + U32* const bt = zc->chainTable; + U32 const btLog = zc->params.cParams.chainLog - 1; + U32 const btMask = (1 << btLog) - 1; + U32 matchIndex = hashTable[h]; + size_t commonLengthSmaller=0, commonLengthLarger=0; + const BYTE* const base = zc->base; + const BYTE* const dictBase = zc->dictBase; + const U32 dictLimit = zc->dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const prefixStart = base + dictLimit; + const BYTE* match; + const U32 current = (U32)(ip-base); + const U32 btLow = btMask >= current ? 0 : current - btMask; + U32* smallerPtr = bt + 2*(current&btMask); + U32* largerPtr = smallerPtr + 1; + U32 dummy32; /* to be nullified at the end */ + U32 const windowLow = zc->lowLimit; + U32 matchEndIdx = current+8; + size_t bestLength = 8; +#ifdef ZSTD_C_PREDICT + U32 predictedSmall = *(bt + 2*((current-1)&btMask) + 0); + U32 predictedLarge = *(bt + 2*((current-1)&btMask) + 1); + predictedSmall += (predictedSmall>0); + predictedLarge += (predictedLarge>0); +#endif /* ZSTD_C_PREDICT */ + + hashTable[h] = current; /* Update Hash Table */ + + while (nbCompares-- && (matchIndex > windowLow)) { + U32* const nextPtr = bt + 2*(matchIndex & btMask); + size_t matchLength = MIN(commonLengthSmaller, commonLengthLarger); /* guaranteed minimum nb of common bytes */ + +#ifdef ZSTD_C_PREDICT /* note : can create issues when hlog small <= 11 */ + const U32* predictPtr = bt + 2*((matchIndex-1) & btMask); /* written this way, as bt is a roll buffer */ + if (matchIndex == predictedSmall) { + /* no need to check length, result known */ + *smallerPtr = matchIndex; + if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ + matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ + predictedSmall = predictPtr[1] + (predictPtr[1]>0); + continue; + } + if (matchIndex == predictedLarge) { + *largerPtr = matchIndex; + if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + largerPtr = nextPtr; + matchIndex = nextPtr[0]; + predictedLarge = predictPtr[0] + (predictPtr[0]>0); + continue; + } +#endif + if ((!extDict) || (matchIndex+matchLength >= dictLimit)) { + match = base + matchIndex; + if (match[matchLength] == ip[matchLength]) + matchLength += ZSTD_count(ip+matchLength+1, match+matchLength+1, iend) +1; + } else { + match = dictBase + matchIndex; + matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iend, dictEnd, prefixStart); + if (matchIndex+matchLength >= dictLimit) + match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ + } + + if (matchLength > bestLength) { + bestLength = matchLength; + if (matchLength > matchEndIdx - matchIndex) + matchEndIdx = matchIndex + (U32)matchLength; + } + + if (ip+matchLength == iend) /* equal : no way to know if inf or sup */ + break; /* drop , to guarantee consistency ; miss a bit of compression, but other solutions can corrupt the tree */ + + if (match[matchLength] < ip[matchLength]) { /* necessarily within correct buffer */ + /* match is smaller than current */ + *smallerPtr = matchIndex; /* update smaller idx */ + commonLengthSmaller = matchLength; /* all smaller will now have at least this guaranteed common length */ + if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ + matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ + } else { + /* match is larger than current */ + *largerPtr = matchIndex; + commonLengthLarger = matchLength; + if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + largerPtr = nextPtr; + matchIndex = nextPtr[0]; + } } + + *smallerPtr = *largerPtr = 0; + if (bestLength > 384) return MIN(192, (U32)(bestLength - 384)); /* speed optimization */ + if (matchEndIdx > current + 8) return matchEndIdx - current - 8; + return 1; +} + + +static size_t ZSTD_insertBtAndFindBestMatch ( + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iend, + size_t* offsetPtr, + U32 nbCompares, const U32 mls, + U32 extDict) +{ + U32* const hashTable = zc->hashTable; + U32 const hashLog = zc->params.cParams.hashLog; + size_t const h = ZSTD_hashPtr(ip, hashLog, mls); + U32* const bt = zc->chainTable; + U32 const btLog = zc->params.cParams.chainLog - 1; + U32 const btMask = (1 << btLog) - 1; + U32 matchIndex = hashTable[h]; + size_t commonLengthSmaller=0, commonLengthLarger=0; + const BYTE* const base = zc->base; + const BYTE* const dictBase = zc->dictBase; + const U32 dictLimit = zc->dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const prefixStart = base + dictLimit; + const U32 current = (U32)(ip-base); + const U32 btLow = btMask >= current ? 0 : current - btMask; + const U32 windowLow = zc->lowLimit; + U32* smallerPtr = bt + 2*(current&btMask); + U32* largerPtr = bt + 2*(current&btMask) + 1; + U32 matchEndIdx = current+8; + U32 dummy32; /* to be nullified at the end */ + size_t bestLength = 0; + + hashTable[h] = current; /* Update Hash Table */ + + while (nbCompares-- && (matchIndex > windowLow)) { + U32* const nextPtr = bt + 2*(matchIndex & btMask); + size_t matchLength = MIN(commonLengthSmaller, commonLengthLarger); /* guaranteed minimum nb of common bytes */ + const BYTE* match; + + if ((!extDict) || (matchIndex+matchLength >= dictLimit)) { + match = base + matchIndex; + if (match[matchLength] == ip[matchLength]) + matchLength += ZSTD_count(ip+matchLength+1, match+matchLength+1, iend) +1; + } else { + match = dictBase + matchIndex; + matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iend, dictEnd, prefixStart); + if (matchIndex+matchLength >= dictLimit) + match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ + } + + if (matchLength > bestLength) { + if (matchLength > matchEndIdx - matchIndex) + matchEndIdx = matchIndex + (U32)matchLength; + if ( (4*(int)(matchLength-bestLength)) > (int)(ZSTD_highbit32(current-matchIndex+1) - ZSTD_highbit32((U32)offsetPtr[0]+1)) ) + bestLength = matchLength, *offsetPtr = ZSTD_REP_MOVE + current - matchIndex; + if (ip+matchLength == iend) /* equal : no way to know if inf or sup */ + break; /* drop, to guarantee consistency (miss a little bit of compression) */ + } + + if (match[matchLength] < ip[matchLength]) { + /* match is smaller than current */ + *smallerPtr = matchIndex; /* update smaller idx */ + commonLengthSmaller = matchLength; /* all smaller will now have at least this guaranteed common length */ + if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ + matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ + } else { + /* match is larger than current */ + *largerPtr = matchIndex; + commonLengthLarger = matchLength; + if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + largerPtr = nextPtr; + matchIndex = nextPtr[0]; + } } + + *smallerPtr = *largerPtr = 0; + + zc->nextToUpdate = (matchEndIdx > current + 8) ? matchEndIdx - 8 : current+1; + return bestLength; +} + + +static void ZSTD_updateTree(ZSTD_CCtx* zc, const BYTE* const ip, const BYTE* const iend, const U32 nbCompares, const U32 mls) +{ + const BYTE* const base = zc->base; + const U32 target = (U32)(ip - base); + U32 idx = zc->nextToUpdate; + + while(idx < target) + idx += ZSTD_insertBt1(zc, base+idx, mls, iend, nbCompares, 0); +} + +/** ZSTD_BtFindBestMatch() : Tree updater, providing best match */ +static size_t ZSTD_BtFindBestMatch ( + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 mls) +{ + if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ + ZSTD_updateTree(zc, ip, iLimit, maxNbAttempts, mls); + return ZSTD_insertBtAndFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, mls, 0); +} + + +static size_t ZSTD_BtFindBestMatch_selectMLS ( + ZSTD_CCtx* zc, /* Index table will be updated */ + const BYTE* ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 matchLengthSearch) +{ + switch(matchLengthSearch) + { + default : /* includes case 3 */ + case 4 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4); + case 5 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5); + case 7 : + case 6 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6); + } +} + + +static void ZSTD_updateTree_extDict(ZSTD_CCtx* zc, const BYTE* const ip, const BYTE* const iend, const U32 nbCompares, const U32 mls) +{ + const BYTE* const base = zc->base; + const U32 target = (U32)(ip - base); + U32 idx = zc->nextToUpdate; + + while (idx < target) idx += ZSTD_insertBt1(zc, base+idx, mls, iend, nbCompares, 1); +} + + +/** Tree updater, providing best match */ +static size_t ZSTD_BtFindBestMatch_extDict ( + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 mls) +{ + if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ + ZSTD_updateTree_extDict(zc, ip, iLimit, maxNbAttempts, mls); + return ZSTD_insertBtAndFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, mls, 1); +} + + +static size_t ZSTD_BtFindBestMatch_selectMLS_extDict ( + ZSTD_CCtx* zc, /* Index table will be updated */ + const BYTE* ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 matchLengthSearch) +{ + switch(matchLengthSearch) + { + default : /* includes case 3 */ + case 4 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4); + case 5 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5); + case 7 : + case 6 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6); + } +} + + + +/* ********************************* +* Hash Chain +***********************************/ +#define NEXT_IN_CHAIN(d, mask) chainTable[(d) & mask] + +/* Update chains up to ip (excluded) + Assumption : always within prefix (i.e. not within extDict) */ +FORCE_INLINE +U32 ZSTD_insertAndFindFirstIndex (ZSTD_CCtx* zc, const BYTE* ip, U32 mls) +{ + U32* const hashTable = zc->hashTable; + const U32 hashLog = zc->params.cParams.hashLog; + U32* const chainTable = zc->chainTable; + const U32 chainMask = (1 << zc->params.cParams.chainLog) - 1; + const BYTE* const base = zc->base; + const U32 target = (U32)(ip - base); + U32 idx = zc->nextToUpdate; + + while(idx < target) { /* catch up */ + size_t const h = ZSTD_hashPtr(base+idx, hashLog, mls); + NEXT_IN_CHAIN(idx, chainMask) = hashTable[h]; + hashTable[h] = idx; + idx++; + } + + zc->nextToUpdate = target; + return hashTable[ZSTD_hashPtr(ip, hashLog, mls)]; +} + + + +FORCE_INLINE /* inlining is important to hardwire a hot branch (template emulation) */ +size_t ZSTD_HcFindBestMatch_generic ( + ZSTD_CCtx* zc, /* Index table will be updated */ + const BYTE* const ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 mls, const U32 extDict) +{ + U32* const chainTable = zc->chainTable; + const U32 chainSize = (1 << zc->params.cParams.chainLog); + const U32 chainMask = chainSize-1; + const BYTE* const base = zc->base; + const BYTE* const dictBase = zc->dictBase; + const U32 dictLimit = zc->dictLimit; + const BYTE* const prefixStart = base + dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const U32 lowLimit = zc->lowLimit; + const U32 current = (U32)(ip-base); + const U32 minChain = current > chainSize ? current - chainSize : 0; + int nbAttempts=maxNbAttempts; + size_t ml=EQUAL_READ32-1; + + /* HC4 match finder */ + U32 matchIndex = ZSTD_insertAndFindFirstIndex (zc, ip, mls); + + for ( ; (matchIndex>lowLimit) & (nbAttempts>0) ; nbAttempts--) { + const BYTE* match; + size_t currentMl=0; + if ((!extDict) || matchIndex >= dictLimit) { + match = base + matchIndex; + if (match[ml] == ip[ml]) /* potentially better */ + currentMl = ZSTD_count(ip, match, iLimit); + } else { + match = dictBase + matchIndex; + if (MEM_read32(match) == MEM_read32(ip)) /* assumption : matchIndex <= dictLimit-4 (by table construction) */ + currentMl = ZSTD_count_2segments(ip+EQUAL_READ32, match+EQUAL_READ32, iLimit, dictEnd, prefixStart) + EQUAL_READ32; + } + + /* save best solution */ + if (currentMl > ml) { ml = currentMl; *offsetPtr = current - matchIndex + ZSTD_REP_MOVE; if (ip+currentMl == iLimit) break; /* best possible, and avoid read overflow*/ } + + if (matchIndex <= minChain) break; + matchIndex = NEXT_IN_CHAIN(matchIndex, chainMask); + } + + return ml; +} + + +FORCE_INLINE size_t ZSTD_HcFindBestMatch_selectMLS ( + ZSTD_CCtx* zc, + const BYTE* ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 matchLengthSearch) +{ + switch(matchLengthSearch) + { + default : /* includes case 3 */ + case 4 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4, 0); + case 5 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5, 0); + case 7 : + case 6 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6, 0); + } +} + + +FORCE_INLINE size_t ZSTD_HcFindBestMatch_extDict_selectMLS ( + ZSTD_CCtx* zc, + const BYTE* ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 matchLengthSearch) +{ + switch(matchLengthSearch) + { + default : /* includes case 3 */ + case 4 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4, 1); + case 5 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5, 1); + case 7 : + case 6 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6, 1); + } +} + + +/* ******************************* +* Common parser - lazy strategy +*********************************/ +FORCE_INLINE +void ZSTD_compressBlock_lazy_generic(ZSTD_CCtx* ctx, + const void* src, size_t srcSize, + const U32 searchMethod, const U32 depth) +{ + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + const BYTE* const base = ctx->base + ctx->dictLimit; + + U32 const maxSearches = 1 << ctx->params.cParams.searchLog; + U32 const mls = ctx->params.cParams.searchLength; + + typedef size_t (*searchMax_f)(ZSTD_CCtx* zc, const BYTE* ip, const BYTE* iLimit, + size_t* offsetPtr, + U32 maxNbAttempts, U32 matchLengthSearch); + searchMax_f const searchMax = searchMethod ? ZSTD_BtFindBestMatch_selectMLS : ZSTD_HcFindBestMatch_selectMLS; + U32 offset_1 = ctx->rep[0], offset_2 = ctx->rep[1], savedOffset=0; + + /* init */ + ip += (ip==base); + ctx->nextToUpdate3 = ctx->nextToUpdate; + { U32 const maxRep = (U32)(ip-base); + if (offset_2 > maxRep) savedOffset = offset_2, offset_2 = 0; + if (offset_1 > maxRep) savedOffset = offset_1, offset_1 = 0; + } + + /* Match Loop */ + while (ip < ilimit) { + size_t matchLength=0; + size_t offset=0; + const BYTE* start=ip+1; + + /* check repCode */ + if ((offset_1>0) & (MEM_read32(ip+1) == MEM_read32(ip+1 - offset_1))) { + /* repcode : we take it */ + matchLength = ZSTD_count(ip+1+EQUAL_READ32, ip+1+EQUAL_READ32-offset_1, iend) + EQUAL_READ32; + if (depth==0) goto _storeSequence; + } + + /* first search (depth 0) */ + { size_t offsetFound = 99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offsetFound, maxSearches, mls); + if (ml2 > matchLength) + matchLength = ml2, start = ip, offset=offsetFound; + } + + if (matchLength < EQUAL_READ32) { + ip += ((ip-anchor) >> g_searchStrength) + 1; /* jump faster over incompressible sections */ + continue; + } + + /* let's try to find a better solution */ + if (depth>=1) + while (ip0) & (MEM_read32(ip) == MEM_read32(ip - offset_1)))) { + size_t const mlRep = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_1, iend) + EQUAL_READ32; + int const gain2 = (int)(mlRep * 3); + int const gain1 = (int)(matchLength*3 - ZSTD_highbit32((U32)offset+1) + 1); + if ((mlRep >= EQUAL_READ32) && (gain2 > gain1)) + matchLength = mlRep, offset = 0, start = ip; + } + { size_t offset2=99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); + int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ + int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 4); + if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { + matchLength = ml2, offset = offset2, start = ip; + continue; /* search a better one */ + } } + + /* let's find an even better one */ + if ((depth==2) && (ip0) & (MEM_read32(ip) == MEM_read32(ip - offset_1)))) { + size_t const ml2 = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_1, iend) + EQUAL_READ32; + int const gain2 = (int)(ml2 * 4); + int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 1); + if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) + matchLength = ml2, offset = 0, start = ip; + } + { size_t offset2=99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); + int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ + int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 7); + if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { + matchLength = ml2, offset = offset2, start = ip; + continue; + } } } + break; /* nothing found : store previous solution */ + } + + /* catch up */ + if (offset) { + while ((start>anchor) && (start>base+offset-ZSTD_REP_MOVE) && (start[-1] == start[-1-offset+ZSTD_REP_MOVE])) /* only search for offset within prefix */ + { start--; matchLength++; } + offset_2 = offset_1; offset_1 = (U32)(offset - ZSTD_REP_MOVE); + } + + /* store sequence */ +_storeSequence: + { size_t const litLength = start - anchor; + ZSTD_storeSeq(seqStorePtr, litLength, anchor, (U32)offset, matchLength-MINMATCH); + anchor = ip = start + matchLength; + } + + /* check immediate repcode */ + while ( (ip <= ilimit) + && ((offset_2>0) + & (MEM_read32(ip) == MEM_read32(ip - offset_2)) )) { + /* store sequence */ + matchLength = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_2, iend) + EQUAL_READ32; + offset = offset_2; offset_2 = offset_1; offset_1 = (U32)offset; /* swap repcodes */ + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, matchLength-MINMATCH); + ip += matchLength; + anchor = ip; + continue; /* faster when present ... (?) */ + } } + + /* Save reps for next block */ + ctx->repToConfirm[0] = offset_1 ? offset_1 : savedOffset; + ctx->repToConfirm[1] = offset_2 ? offset_2 : savedOffset; + + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } +} + + +static void ZSTD_compressBlock_btlazy2(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ + ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 1, 2); +} + +static void ZSTD_compressBlock_lazy2(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ + ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 0, 2); +} + +static void ZSTD_compressBlock_lazy(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ + ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 0, 1); +} + +static void ZSTD_compressBlock_greedy(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ + ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 0, 0); +} + + +FORCE_INLINE +void ZSTD_compressBlock_lazy_extDict_generic(ZSTD_CCtx* ctx, + const void* src, size_t srcSize, + const U32 searchMethod, const U32 depth) +{ + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + const BYTE* const base = ctx->base; + const U32 dictLimit = ctx->dictLimit; + const U32 lowestIndex = ctx->lowLimit; + const BYTE* const prefixStart = base + dictLimit; + const BYTE* const dictBase = ctx->dictBase; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const dictStart = dictBase + ctx->lowLimit; + + const U32 maxSearches = 1 << ctx->params.cParams.searchLog; + const U32 mls = ctx->params.cParams.searchLength; + + typedef size_t (*searchMax_f)(ZSTD_CCtx* zc, const BYTE* ip, const BYTE* iLimit, + size_t* offsetPtr, + U32 maxNbAttempts, U32 matchLengthSearch); + searchMax_f searchMax = searchMethod ? ZSTD_BtFindBestMatch_selectMLS_extDict : ZSTD_HcFindBestMatch_extDict_selectMLS; + + U32 offset_1 = ctx->rep[0], offset_2 = ctx->rep[1]; + + /* init */ + ctx->nextToUpdate3 = ctx->nextToUpdate; + ip += (ip == prefixStart); + + /* Match Loop */ + while (ip < ilimit) { + size_t matchLength=0; + size_t offset=0; + const BYTE* start=ip+1; + U32 current = (U32)(ip-base); + + /* check repCode */ + { const U32 repIndex = (U32)(current+1 - offset_1); + const BYTE* const repBase = repIndex < dictLimit ? dictBase : base; + const BYTE* const repMatch = repBase + repIndex; + if (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex > lowestIndex)) /* intentional overflow */ + if (MEM_read32(ip+1) == MEM_read32(repMatch)) { + /* repcode detected we should take it */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + matchLength = ZSTD_count_2segments(ip+1+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; + if (depth==0) goto _storeSequence; + } } + + /* first search (depth 0) */ + { size_t offsetFound = 99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offsetFound, maxSearches, mls); + if (ml2 > matchLength) + matchLength = ml2, start = ip, offset=offsetFound; + } + + if (matchLength < EQUAL_READ32) { + ip += ((ip-anchor) >> g_searchStrength) + 1; /* jump faster over incompressible sections */ + continue; + } + + /* let's try to find a better solution */ + if (depth>=1) + while (ip= 3) & (repIndex > lowestIndex)) /* intentional overflow */ + if (MEM_read32(ip) == MEM_read32(repMatch)) { + /* repcode detected */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + size_t const repLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; + int const gain2 = (int)(repLength * 3); + int const gain1 = (int)(matchLength*3 - ZSTD_highbit32((U32)offset+1) + 1); + if ((repLength >= EQUAL_READ32) && (gain2 > gain1)) + matchLength = repLength, offset = 0, start = ip; + } } + + /* search match, depth 1 */ + { size_t offset2=99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); + int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ + int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 4); + if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { + matchLength = ml2, offset = offset2, start = ip; + continue; /* search a better one */ + } } + + /* let's find an even better one */ + if ((depth==2) && (ip= 3) & (repIndex > lowestIndex)) /* intentional overflow */ + if (MEM_read32(ip) == MEM_read32(repMatch)) { + /* repcode detected */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + size_t repLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; + int gain2 = (int)(repLength * 4); + int gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 1); + if ((repLength >= EQUAL_READ32) && (gain2 > gain1)) + matchLength = repLength, offset = 0, start = ip; + } } + + /* search match, depth 2 */ + { size_t offset2=99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); + int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ + int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 7); + if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { + matchLength = ml2, offset = offset2, start = ip; + continue; + } } } + break; /* nothing found : store previous solution */ + } + + /* catch up */ + if (offset) { + U32 const matchIndex = (U32)((start-base) - (offset - ZSTD_REP_MOVE)); + const BYTE* match = (matchIndex < dictLimit) ? dictBase + matchIndex : base + matchIndex; + const BYTE* const mStart = (matchIndex < dictLimit) ? dictStart : prefixStart; + while ((start>anchor) && (match>mStart) && (start[-1] == match[-1])) { start--; match--; matchLength++; } /* catch up */ + offset_2 = offset_1; offset_1 = (U32)(offset - ZSTD_REP_MOVE); + } + + /* store sequence */ +_storeSequence: + { size_t const litLength = start - anchor; + ZSTD_storeSeq(seqStorePtr, litLength, anchor, (U32)offset, matchLength-MINMATCH); + anchor = ip = start + matchLength; + } + + /* check immediate repcode */ + while (ip <= ilimit) { + const U32 repIndex = (U32)((ip-base) - offset_2); + const BYTE* const repBase = repIndex < dictLimit ? dictBase : base; + const BYTE* const repMatch = repBase + repIndex; + if (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex > lowestIndex)) /* intentional overflow */ + if (MEM_read32(ip) == MEM_read32(repMatch)) { + /* repcode detected we should take it */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + matchLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; + offset = offset_2; offset_2 = offset_1; offset_1 = (U32)offset; /* swap offset history */ + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, matchLength-MINMATCH); + ip += matchLength; + anchor = ip; + continue; /* faster when present ... (?) */ + } + break; + } } + + /* Save reps for next block */ + ctx->repToConfirm[0] = offset_1; ctx->repToConfirm[1] = offset_2; + + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } +} + + +void ZSTD_compressBlock_greedy_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ + ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 0, 0); +} + +static void ZSTD_compressBlock_lazy_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ + ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 0, 1); +} + +static void ZSTD_compressBlock_lazy2_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ + ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 0, 2); +} + +static void ZSTD_compressBlock_btlazy2_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ + ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 1, 2); +} + + +/* The optimal parser */ +#include "zstd_opt.h" + +static void ZSTD_compressBlock_btopt(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ +#ifdef ZSTD_OPT_H_91842398743 + ZSTD_compressBlock_opt_generic(ctx, src, srcSize, 0); +#else + (void)ctx; (void)src; (void)srcSize; + return; +#endif +} + +static void ZSTD_compressBlock_btopt2(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ +#ifdef ZSTD_OPT_H_91842398743 + ZSTD_compressBlock_opt_generic(ctx, src, srcSize, 1); +#else + (void)ctx; (void)src; (void)srcSize; + return; +#endif +} + +static void ZSTD_compressBlock_btopt_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ +#ifdef ZSTD_OPT_H_91842398743 + ZSTD_compressBlock_opt_extDict_generic(ctx, src, srcSize, 0); +#else + (void)ctx; (void)src; (void)srcSize; + return; +#endif +} + +static void ZSTD_compressBlock_btopt2_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) +{ +#ifdef ZSTD_OPT_H_91842398743 + ZSTD_compressBlock_opt_extDict_generic(ctx, src, srcSize, 1); +#else + (void)ctx; (void)src; (void)srcSize; + return; +#endif +} + + +typedef void (*ZSTD_blockCompressor) (ZSTD_CCtx* ctx, const void* src, size_t srcSize); + +static ZSTD_blockCompressor ZSTD_selectBlockCompressor(ZSTD_strategy strat, int extDict) +{ + static const ZSTD_blockCompressor blockCompressor[2][8] = { + { ZSTD_compressBlock_fast, ZSTD_compressBlock_doubleFast, ZSTD_compressBlock_greedy, ZSTD_compressBlock_lazy, ZSTD_compressBlock_lazy2, ZSTD_compressBlock_btlazy2, ZSTD_compressBlock_btopt, ZSTD_compressBlock_btopt2 }, + { ZSTD_compressBlock_fast_extDict, ZSTD_compressBlock_doubleFast_extDict, ZSTD_compressBlock_greedy_extDict, ZSTD_compressBlock_lazy_extDict,ZSTD_compressBlock_lazy2_extDict, ZSTD_compressBlock_btlazy2_extDict, ZSTD_compressBlock_btopt_extDict, ZSTD_compressBlock_btopt2_extDict } + }; + + return blockCompressor[extDict][(U32)strat]; +} + + +static size_t ZSTD_compressBlock_internal(ZSTD_CCtx* zc, void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + ZSTD_blockCompressor const blockCompressor = ZSTD_selectBlockCompressor(zc->params.cParams.strategy, zc->lowLimit < zc->dictLimit); + const BYTE* const base = zc->base; + const BYTE* const istart = (const BYTE*)src; + const U32 current = (U32)(istart-base); + if (srcSize < MIN_CBLOCK_SIZE+ZSTD_blockHeaderSize+1) return 0; /* don't even attempt compression below a certain srcSize */ + ZSTD_resetSeqStore(&(zc->seqStore)); + if (current > zc->nextToUpdate + 384) + zc->nextToUpdate = current - MIN(192, (U32)(current - zc->nextToUpdate - 384)); /* update tree not updated after finding very long rep matches */ + blockCompressor(zc, src, srcSize); + return ZSTD_compressSequences(zc, dst, dstCapacity, srcSize); +} + + +/*! ZSTD_compress_generic() : +* Compress a chunk of data into one or multiple blocks. +* All blocks will be terminated, all input will be consumed. +* Function will issue an error if there is not enough `dstCapacity` to hold the compressed content. +* Frame is supposed already started (header already produced) +* @return : compressed size, or an error code +*/ +static size_t ZSTD_compress_generic (ZSTD_CCtx* cctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + U32 lastFrameChunk) +{ + size_t blockSize = cctx->blockSize; + size_t remaining = srcSize; + const BYTE* ip = (const BYTE*)src; + BYTE* const ostart = (BYTE*)dst; + BYTE* op = ostart; + U32 const maxDist = 1 << cctx->params.cParams.windowLog; + + if (cctx->params.fParams.checksumFlag && srcSize) + XXH64_update(&cctx->xxhState, src, srcSize); + + while (remaining) { + U32 const lastBlock = lastFrameChunk & (blockSize >= remaining); + size_t cSize; + + if (dstCapacity < ZSTD_blockHeaderSize + MIN_CBLOCK_SIZE) return ERROR(dstSize_tooSmall); /* not enough space to store compressed block */ + if (remaining < blockSize) blockSize = remaining; + + /* preemptive overflow correction */ + if (cctx->lowLimit > (3U<<29)) { + U32 const cycleMask = (1 << ZSTD_cycleLog(cctx->params.cParams.hashLog, cctx->params.cParams.strategy)) - 1; + U32 const current = (U32)(ip - cctx->base); + U32 const newCurrent = (current & cycleMask) + (1 << cctx->params.cParams.windowLog); + U32 const correction = current - newCurrent; + ZSTD_STATIC_ASSERT(ZSTD_WINDOWLOG_MAX_64 <= 30); + ZSTD_reduceIndex(cctx, correction); + cctx->base += correction; + cctx->dictBase += correction; + cctx->lowLimit -= correction; + cctx->dictLimit -= correction; + if (cctx->nextToUpdate < correction) cctx->nextToUpdate = 0; + else cctx->nextToUpdate -= correction; + } + + if ((U32)(ip+blockSize - cctx->base) > cctx->loadedDictEnd + maxDist) { + /* enforce maxDist */ + U32 const newLowLimit = (U32)(ip+blockSize - cctx->base) - maxDist; + if (cctx->lowLimit < newLowLimit) cctx->lowLimit = newLowLimit; + if (cctx->dictLimit < cctx->lowLimit) cctx->dictLimit = cctx->lowLimit; + } + + cSize = ZSTD_compressBlock_internal(cctx, op+ZSTD_blockHeaderSize, dstCapacity-ZSTD_blockHeaderSize, ip, blockSize); + if (ZSTD_isError(cSize)) return cSize; + + if (cSize == 0) { /* block is not compressible */ + U32 const cBlockHeader24 = lastBlock + (((U32)bt_raw)<<1) + (U32)(blockSize << 3); + if (blockSize + ZSTD_blockHeaderSize > dstCapacity) return ERROR(dstSize_tooSmall); + MEM_writeLE32(op, cBlockHeader24); /* no pb, 4th byte will be overwritten */ + memcpy(op + ZSTD_blockHeaderSize, ip, blockSize); + cSize = ZSTD_blockHeaderSize+blockSize; + } else { + U32 const cBlockHeader24 = lastBlock + (((U32)bt_compressed)<<1) + (U32)(cSize << 3); + MEM_writeLE24(op, cBlockHeader24); + cSize += ZSTD_blockHeaderSize; + } + + remaining -= blockSize; + dstCapacity -= cSize; + ip += blockSize; + op += cSize; + } + + if (lastFrameChunk && (op>ostart)) cctx->stage = ZSTDcs_ending; + return op-ostart; +} + + +static size_t ZSTD_writeFrameHeader(void* dst, size_t dstCapacity, + ZSTD_parameters params, U64 pledgedSrcSize, U32 dictID) +{ BYTE* const op = (BYTE*)dst; + U32 const dictIDSizeCode = (dictID>0) + (dictID>=256) + (dictID>=65536); /* 0-3 */ + U32 const checksumFlag = params.fParams.checksumFlag>0; + U32 const windowSize = 1U << params.cParams.windowLog; + U32 const singleSegment = params.fParams.contentSizeFlag && (windowSize >= pledgedSrcSize); + BYTE const windowLogByte = (BYTE)((params.cParams.windowLog - ZSTD_WINDOWLOG_ABSOLUTEMIN) << 3); + U32 const fcsCode = params.fParams.contentSizeFlag ? + (pledgedSrcSize>=256) + (pledgedSrcSize>=65536+256) + (pledgedSrcSize>=0xFFFFFFFFU) : /* 0-3 */ + 0; + BYTE const frameHeaderDecriptionByte = (BYTE)(dictIDSizeCode + (checksumFlag<<2) + (singleSegment<<5) + (fcsCode<<6) ); + size_t pos; + + if (dstCapacity < ZSTD_frameHeaderSize_max) return ERROR(dstSize_tooSmall); + + MEM_writeLE32(dst, ZSTD_MAGICNUMBER); + op[4] = frameHeaderDecriptionByte; pos=5; + if (!singleSegment) op[pos++] = windowLogByte; + switch(dictIDSizeCode) + { + default: /* impossible */ + case 0 : break; + case 1 : op[pos] = (BYTE)(dictID); pos++; break; + case 2 : MEM_writeLE16(op+pos, (U16)dictID); pos+=2; break; + case 3 : MEM_writeLE32(op+pos, dictID); pos+=4; break; + } + switch(fcsCode) + { + default: /* impossible */ + case 0 : if (singleSegment) op[pos++] = (BYTE)(pledgedSrcSize); break; + case 1 : MEM_writeLE16(op+pos, (U16)(pledgedSrcSize-256)); pos+=2; break; + case 2 : MEM_writeLE32(op+pos, (U32)(pledgedSrcSize)); pos+=4; break; + case 3 : MEM_writeLE64(op+pos, (U64)(pledgedSrcSize)); pos+=8; break; + } + return pos; +} + + +static size_t ZSTD_compressContinue_internal (ZSTD_CCtx* cctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + U32 frame, U32 lastFrameChunk) +{ + const BYTE* const ip = (const BYTE*) src; + size_t fhSize = 0; + + if (cctx->stage==ZSTDcs_created) return ERROR(stage_wrong); /* missing init (ZSTD_compressBegin) */ + + if (frame && (cctx->stage==ZSTDcs_init)) { + fhSize = ZSTD_writeFrameHeader(dst, dstCapacity, cctx->params, cctx->frameContentSize, cctx->dictID); + if (ZSTD_isError(fhSize)) return fhSize; + dstCapacity -= fhSize; + dst = (char*)dst + fhSize; + cctx->stage = ZSTDcs_ongoing; + } + + /* Check if blocks follow each other */ + if (src != cctx->nextSrc) { + /* not contiguous */ + ptrdiff_t const delta = cctx->nextSrc - ip; + cctx->lowLimit = cctx->dictLimit; + cctx->dictLimit = (U32)(cctx->nextSrc - cctx->base); + cctx->dictBase = cctx->base; + cctx->base -= delta; + cctx->nextToUpdate = cctx->dictLimit; + if (cctx->dictLimit - cctx->lowLimit < HASH_READ_SIZE) cctx->lowLimit = cctx->dictLimit; /* too small extDict */ + } + + /* if input and dictionary overlap : reduce dictionary (area presumed modified by input) */ + if ((ip+srcSize > cctx->dictBase + cctx->lowLimit) & (ip < cctx->dictBase + cctx->dictLimit)) { + ptrdiff_t const highInputIdx = (ip + srcSize) - cctx->dictBase; + U32 const lowLimitMax = (highInputIdx > (ptrdiff_t)cctx->dictLimit) ? cctx->dictLimit : (U32)highInputIdx; + cctx->lowLimit = lowLimitMax; + } + + cctx->nextSrc = ip + srcSize; + + if (srcSize) { + size_t const cSize = frame ? + ZSTD_compress_generic (cctx, dst, dstCapacity, src, srcSize, lastFrameChunk) : + ZSTD_compressBlock_internal (cctx, dst, dstCapacity, src, srcSize); + if (ZSTD_isError(cSize)) return cSize; + return cSize + fhSize; + } else + return fhSize; +} + + +size_t ZSTD_compressContinue (ZSTD_CCtx* cctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize) +{ + return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1, 0); +} + + +size_t ZSTD_getBlockSizeMax(ZSTD_CCtx* cctx) +{ + return MIN (ZSTD_BLOCKSIZE_ABSOLUTEMAX, 1 << cctx->params.cParams.windowLog); +} + +size_t ZSTD_compressBlock(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + size_t const blockSizeMax = ZSTD_getBlockSizeMax(cctx); + if (srcSize > blockSizeMax) return ERROR(srcSize_wrong); + return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 0, 0); +} + +/*! ZSTD_loadDictionaryContent() : + * @return : 0, or an error code + */ +static size_t ZSTD_loadDictionaryContent(ZSTD_CCtx* zc, const void* src, size_t srcSize) +{ + const BYTE* const ip = (const BYTE*) src; + const BYTE* const iend = ip + srcSize; + + /* input becomes current prefix */ + zc->lowLimit = zc->dictLimit; + zc->dictLimit = (U32)(zc->nextSrc - zc->base); + zc->dictBase = zc->base; + zc->base += ip - zc->nextSrc; + zc->nextToUpdate = zc->dictLimit; + zc->loadedDictEnd = zc->forceWindow ? 0 : (U32)(iend - zc->base); + + zc->nextSrc = iend; + if (srcSize <= HASH_READ_SIZE) return 0; + + switch(zc->params.cParams.strategy) + { + case ZSTD_fast: + ZSTD_fillHashTable (zc, iend, zc->params.cParams.searchLength); + break; + + case ZSTD_dfast: + ZSTD_fillDoubleHashTable (zc, iend, zc->params.cParams.searchLength); + break; + + case ZSTD_greedy: + case ZSTD_lazy: + case ZSTD_lazy2: + if (srcSize >= HASH_READ_SIZE) + ZSTD_insertAndFindFirstIndex(zc, iend-HASH_READ_SIZE, zc->params.cParams.searchLength); + break; + + case ZSTD_btlazy2: + case ZSTD_btopt: + case ZSTD_btopt2: + if (srcSize >= HASH_READ_SIZE) + ZSTD_updateTree(zc, iend-HASH_READ_SIZE, iend, 1 << zc->params.cParams.searchLog, zc->params.cParams.searchLength); + break; + + default: + return ERROR(GENERIC); /* strategy doesn't exist; impossible */ + } + + zc->nextToUpdate = (U32)(iend - zc->base); + return 0; +} + + +/* Dictionaries that assign zero probability to symbols that show up causes problems + when FSE encoding. Refuse dictionaries that assign zero probability to symbols + that we may encounter during compression. + NOTE: This behavior is not standard and could be improved in the future. */ +static size_t ZSTD_checkDictNCount(short* normalizedCounter, unsigned dictMaxSymbolValue, unsigned maxSymbolValue) { + U32 s; + if (dictMaxSymbolValue < maxSymbolValue) return ERROR(dictionary_corrupted); + for (s = 0; s <= maxSymbolValue; ++s) { + if (normalizedCounter[s] == 0) return ERROR(dictionary_corrupted); + } + return 0; +} + + +/* Dictionary format : + * See : + * https://github.com/facebook/zstd/blob/master/doc/zstd_compression_format.md#dictionary-format + */ +/*! ZSTD_loadZstdDictionary() : + * @return : 0, or an error code + * assumptions : magic number supposed already checked + * dictSize supposed > 8 + */ +static size_t ZSTD_loadZstdDictionary(ZSTD_CCtx* cctx, const void* dict, size_t dictSize) +{ + const BYTE* dictPtr = (const BYTE*)dict; + const BYTE* const dictEnd = dictPtr + dictSize; + short offcodeNCount[MaxOff+1]; + unsigned offcodeMaxValue = MaxOff; + BYTE scratchBuffer[1<dictID = cctx->params.fParams.noDictIDFlag ? 0 : MEM_readLE32(dictPtr); + dictPtr += 4; + + { size_t const hufHeaderSize = HUF_readCTable(cctx->hufTable, 255, dictPtr, dictEnd-dictPtr); + if (HUF_isError(hufHeaderSize)) return ERROR(dictionary_corrupted); + dictPtr += hufHeaderSize; + } + + { unsigned offcodeLog; + size_t const offcodeHeaderSize = FSE_readNCount(offcodeNCount, &offcodeMaxValue, &offcodeLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(offcodeHeaderSize)) return ERROR(dictionary_corrupted); + if (offcodeLog > OffFSELog) return ERROR(dictionary_corrupted); + /* Defer checking offcodeMaxValue because we need to know the size of the dictionary content */ + CHECK_E (FSE_buildCTable_wksp(cctx->offcodeCTable, offcodeNCount, offcodeMaxValue, offcodeLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted); + dictPtr += offcodeHeaderSize; + } + + { short matchlengthNCount[MaxML+1]; + unsigned matchlengthMaxValue = MaxML, matchlengthLog; + size_t const matchlengthHeaderSize = FSE_readNCount(matchlengthNCount, &matchlengthMaxValue, &matchlengthLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(matchlengthHeaderSize)) return ERROR(dictionary_corrupted); + if (matchlengthLog > MLFSELog) return ERROR(dictionary_corrupted); + /* Every match length code must have non-zero probability */ + CHECK_F (ZSTD_checkDictNCount(matchlengthNCount, matchlengthMaxValue, MaxML)); + CHECK_E (FSE_buildCTable_wksp(cctx->matchlengthCTable, matchlengthNCount, matchlengthMaxValue, matchlengthLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted); + dictPtr += matchlengthHeaderSize; + } + + { short litlengthNCount[MaxLL+1]; + unsigned litlengthMaxValue = MaxLL, litlengthLog; + size_t const litlengthHeaderSize = FSE_readNCount(litlengthNCount, &litlengthMaxValue, &litlengthLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(litlengthHeaderSize)) return ERROR(dictionary_corrupted); + if (litlengthLog > LLFSELog) return ERROR(dictionary_corrupted); + /* Every literal length code must have non-zero probability */ + CHECK_F (ZSTD_checkDictNCount(litlengthNCount, litlengthMaxValue, MaxLL)); + CHECK_E(FSE_buildCTable_wksp(cctx->litlengthCTable, litlengthNCount, litlengthMaxValue, litlengthLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted); + dictPtr += litlengthHeaderSize; + } + + if (dictPtr+12 > dictEnd) return ERROR(dictionary_corrupted); + cctx->rep[0] = MEM_readLE32(dictPtr+0); + cctx->rep[1] = MEM_readLE32(dictPtr+4); + cctx->rep[2] = MEM_readLE32(dictPtr+8); + dictPtr += 12; + + { size_t const dictContentSize = (size_t)(dictEnd - dictPtr); + U32 offcodeMax = MaxOff; + if (dictContentSize <= ((U32)-1) - 128 KB) { + U32 const maxOffset = (U32)dictContentSize + 128 KB; /* The maximum offset that must be supported */ + offcodeMax = ZSTD_highbit32(maxOffset); /* Calculate minimum offset code required to represent maxOffset */ + } + /* All offset values <= dictContentSize + 128 KB must be representable */ + CHECK_F (ZSTD_checkDictNCount(offcodeNCount, offcodeMaxValue, MIN(offcodeMax, MaxOff))); + /* All repCodes must be <= dictContentSize and != 0*/ + { U32 u; + for (u=0; u<3; u++) { + if (cctx->rep[u] == 0) return ERROR(dictionary_corrupted); + if (cctx->rep[u] > dictContentSize) return ERROR(dictionary_corrupted); + } } + + cctx->flagStaticTables = 1; + cctx->flagStaticHufTable = HUF_repeat_valid; + return ZSTD_loadDictionaryContent(cctx, dictPtr, dictContentSize); + } +} + +/** ZSTD_compress_insertDictionary() : +* @return : 0, or an error code */ +static size_t ZSTD_compress_insertDictionary(ZSTD_CCtx* cctx, const void* dict, size_t dictSize) +{ + if ((dict==NULL) || (dictSize<=8)) return 0; + + /* dict as pure content */ + if ((MEM_readLE32(dict) != ZSTD_DICT_MAGIC) || (cctx->forceRawDict)) + return ZSTD_loadDictionaryContent(cctx, dict, dictSize); + + /* dict as zstd dictionary */ + return ZSTD_loadZstdDictionary(cctx, dict, dictSize); +} + +/*! ZSTD_compressBegin_internal() : +* @return : 0, or an error code */ +static size_t ZSTD_compressBegin_internal(ZSTD_CCtx* cctx, + const void* dict, size_t dictSize, + ZSTD_parameters params, U64 pledgedSrcSize) +{ + ZSTD_compResetPolicy_e const crp = dictSize ? ZSTDcrp_fullReset : ZSTDcrp_continue; + CHECK_F(ZSTD_resetCCtx_advanced(cctx, params, pledgedSrcSize, crp)); + return ZSTD_compress_insertDictionary(cctx, dict, dictSize); +} + + +/*! ZSTD_compressBegin_advanced() : +* @return : 0, or an error code */ +size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, + const void* dict, size_t dictSize, + ZSTD_parameters params, unsigned long long pledgedSrcSize) +{ + /* compression parameters verification and optimization */ + CHECK_F(ZSTD_checkCParams(params.cParams)); + return ZSTD_compressBegin_internal(cctx, dict, dictSize, params, pledgedSrcSize); +} + + +size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel) +{ + ZSTD_parameters const params = ZSTD_getParams(compressionLevel, 0, dictSize); + return ZSTD_compressBegin_internal(cctx, dict, dictSize, params, 0); +} + + +size_t ZSTD_compressBegin(ZSTD_CCtx* cctx, int compressionLevel) +{ + return ZSTD_compressBegin_usingDict(cctx, NULL, 0, compressionLevel); +} + + +/*! ZSTD_writeEpilogue() : +* Ends a frame. +* @return : nb of bytes written into dst (or an error code) */ +static size_t ZSTD_writeEpilogue(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity) +{ + BYTE* const ostart = (BYTE*)dst; + BYTE* op = ostart; + size_t fhSize = 0; + + if (cctx->stage == ZSTDcs_created) return ERROR(stage_wrong); /* init missing */ + + /* special case : empty frame */ + if (cctx->stage == ZSTDcs_init) { + fhSize = ZSTD_writeFrameHeader(dst, dstCapacity, cctx->params, 0, 0); + if (ZSTD_isError(fhSize)) return fhSize; + dstCapacity -= fhSize; + op += fhSize; + cctx->stage = ZSTDcs_ongoing; + } + + if (cctx->stage != ZSTDcs_ending) { + /* write one last empty block, make it the "last" block */ + U32 const cBlockHeader24 = 1 /* last block */ + (((U32)bt_raw)<<1) + 0; + if (dstCapacity<4) return ERROR(dstSize_tooSmall); + MEM_writeLE32(op, cBlockHeader24); + op += ZSTD_blockHeaderSize; + dstCapacity -= ZSTD_blockHeaderSize; + } + + if (cctx->params.fParams.checksumFlag) { + U32 const checksum = (U32) XXH64_digest(&cctx->xxhState); + if (dstCapacity<4) return ERROR(dstSize_tooSmall); + MEM_writeLE32(op, checksum); + op += 4; + } + + cctx->stage = ZSTDcs_created; /* return to "created but no init" status */ + return op-ostart; +} + + +size_t ZSTD_compressEnd (ZSTD_CCtx* cctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize) +{ + size_t endResult; + size_t const cSize = ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1, 1); + if (ZSTD_isError(cSize)) return cSize; + endResult = ZSTD_writeEpilogue(cctx, (char*)dst + cSize, dstCapacity-cSize); + if (ZSTD_isError(endResult)) return endResult; + return cSize + endResult; +} + + +static size_t ZSTD_compress_internal (ZSTD_CCtx* cctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict,size_t dictSize, + ZSTD_parameters params) +{ + CHECK_F(ZSTD_compressBegin_internal(cctx, dict, dictSize, params, srcSize)); + return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize); +} + +size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict,size_t dictSize, + ZSTD_parameters params) +{ + CHECK_F(ZSTD_checkCParams(params.cParams)); + return ZSTD_compress_internal(ctx, dst, dstCapacity, src, srcSize, dict, dictSize, params); +} + +size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void* dict, size_t dictSize, int compressionLevel) +{ + ZSTD_parameters params = ZSTD_getParams(compressionLevel, srcSize, dict ? dictSize : 0); + params.fParams.contentSizeFlag = 1; + return ZSTD_compress_internal(ctx, dst, dstCapacity, src, srcSize, dict, dictSize, params); +} + +size_t ZSTD_compressCCtx (ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel) +{ + return ZSTD_compress_usingDict(ctx, dst, dstCapacity, src, srcSize, NULL, 0, compressionLevel); +} + +size_t ZSTD_compress(void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel) +{ + size_t result; + ZSTD_CCtx ctxBody; + memset(&ctxBody, 0, sizeof(ctxBody)); + memcpy(&ctxBody.customMem, &defaultCustomMem, sizeof(ZSTD_customMem)); + result = ZSTD_compressCCtx(&ctxBody, dst, dstCapacity, src, srcSize, compressionLevel); + ZSTD_free(ctxBody.workSpace, defaultCustomMem); /* can't free ctxBody itself, as it's on stack; free only heap content */ + return result; +} + + +/* ===== Dictionary API ===== */ + +struct ZSTD_CDict_s { + void* dictBuffer; + const void* dictContent; + size_t dictContentSize; + ZSTD_CCtx* refContext; +}; /* typedef'd tp ZSTD_CDict within "zstd.h" */ + +size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict) +{ + if (cdict==NULL) return 0; /* support sizeof on NULL */ + return ZSTD_sizeof_CCtx(cdict->refContext) + (cdict->dictBuffer ? cdict->dictContentSize : 0) + sizeof(*cdict); +} + +ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, unsigned byReference, + ZSTD_parameters params, ZSTD_customMem customMem) +{ + if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; + if (!customMem.customAlloc || !customMem.customFree) return NULL; + + { ZSTD_CDict* const cdict = (ZSTD_CDict*) ZSTD_malloc(sizeof(ZSTD_CDict), customMem); + ZSTD_CCtx* const cctx = ZSTD_createCCtx_advanced(customMem); + + if (!cdict || !cctx) { + ZSTD_free(cdict, customMem); + ZSTD_freeCCtx(cctx); + return NULL; + } + + if ((byReference) || (!dictBuffer) || (!dictSize)) { + cdict->dictBuffer = NULL; + cdict->dictContent = dictBuffer; + } else { + void* const internalBuffer = ZSTD_malloc(dictSize, customMem); + if (!internalBuffer) { ZSTD_free(cctx, customMem); ZSTD_free(cdict, customMem); return NULL; } + memcpy(internalBuffer, dictBuffer, dictSize); + cdict->dictBuffer = internalBuffer; + cdict->dictContent = internalBuffer; + } + + { size_t const errorCode = ZSTD_compressBegin_advanced(cctx, cdict->dictContent, dictSize, params, 0); + if (ZSTD_isError(errorCode)) { + ZSTD_free(cdict->dictBuffer, customMem); + ZSTD_free(cdict, customMem); + ZSTD_freeCCtx(cctx); + return NULL; + } } + + cdict->refContext = cctx; + cdict->dictContentSize = dictSize; + return cdict; + } +} + +ZSTD_CDict* ZSTD_createCDict(const void* dict, size_t dictSize, int compressionLevel) +{ + ZSTD_customMem const allocator = { NULL, NULL, NULL }; + ZSTD_parameters params = ZSTD_getParams(compressionLevel, 0, dictSize); + params.fParams.contentSizeFlag = 1; + return ZSTD_createCDict_advanced(dict, dictSize, 0, params, allocator); +} + +ZSTD_CDict* ZSTD_createCDict_byReference(const void* dict, size_t dictSize, int compressionLevel) +{ + ZSTD_customMem const allocator = { NULL, NULL, NULL }; + ZSTD_parameters params = ZSTD_getParams(compressionLevel, 0, dictSize); + params.fParams.contentSizeFlag = 1; + return ZSTD_createCDict_advanced(dict, dictSize, 1, params, allocator); +} + +size_t ZSTD_freeCDict(ZSTD_CDict* cdict) +{ + if (cdict==NULL) return 0; /* support free on NULL */ + { ZSTD_customMem const cMem = cdict->refContext->customMem; + ZSTD_freeCCtx(cdict->refContext); + ZSTD_free(cdict->dictBuffer, cMem); + ZSTD_free(cdict, cMem); + return 0; + } +} + +static ZSTD_parameters ZSTD_getParamsFromCDict(const ZSTD_CDict* cdict) { + return ZSTD_getParamsFromCCtx(cdict->refContext); +} + +size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize) +{ + if (cdict->dictContentSize) CHECK_F(ZSTD_copyCCtx(cctx, cdict->refContext, pledgedSrcSize)) + else { + ZSTD_parameters params = cdict->refContext->params; + params.fParams.contentSizeFlag = (pledgedSrcSize > 0); + CHECK_F(ZSTD_compressBegin_advanced(cctx, NULL, 0, params, pledgedSrcSize)); + } + return 0; +} + +/*! ZSTD_compress_usingCDict() : +* Compression using a digested Dictionary. +* Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. +* Note that compression level is decided during dictionary creation */ +size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const ZSTD_CDict* cdict) +{ + CHECK_F(ZSTD_compressBegin_usingCDict(cctx, cdict, srcSize)); + + if (cdict->refContext->params.fParams.contentSizeFlag==1) { + cctx->params.fParams.contentSizeFlag = 1; + cctx->frameContentSize = srcSize; + } + + return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize); +} + + + +/* ****************************************************************** +* Streaming +********************************************************************/ + +typedef enum { zcss_init, zcss_load, zcss_flush, zcss_final } ZSTD_cStreamStage; + +struct ZSTD_CStream_s { + ZSTD_CCtx* cctx; + ZSTD_CDict* cdictLocal; + const ZSTD_CDict* cdict; + char* inBuff; + size_t inBuffSize; + size_t inToCompress; + size_t inBuffPos; + size_t inBuffTarget; + size_t blockSize; + char* outBuff; + size_t outBuffSize; + size_t outBuffContentSize; + size_t outBuffFlushedSize; + ZSTD_cStreamStage stage; + U32 checksum; + U32 frameEnded; + U64 pledgedSrcSize; + U64 inputProcessed; + ZSTD_parameters params; + ZSTD_customMem customMem; +}; /* typedef'd to ZSTD_CStream within "zstd.h" */ + +ZSTD_CStream* ZSTD_createCStream(void) +{ + return ZSTD_createCStream_advanced(defaultCustomMem); +} + +ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem) +{ + ZSTD_CStream* zcs; + + if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; + if (!customMem.customAlloc || !customMem.customFree) return NULL; + + zcs = (ZSTD_CStream*)ZSTD_malloc(sizeof(ZSTD_CStream), customMem); + if (zcs==NULL) return NULL; + memset(zcs, 0, sizeof(ZSTD_CStream)); + memcpy(&zcs->customMem, &customMem, sizeof(ZSTD_customMem)); + zcs->cctx = ZSTD_createCCtx_advanced(customMem); + if (zcs->cctx == NULL) { ZSTD_freeCStream(zcs); return NULL; } + return zcs; +} + +size_t ZSTD_freeCStream(ZSTD_CStream* zcs) +{ + if (zcs==NULL) return 0; /* support free on NULL */ + { ZSTD_customMem const cMem = zcs->customMem; + ZSTD_freeCCtx(zcs->cctx); + zcs->cctx = NULL; + ZSTD_freeCDict(zcs->cdictLocal); + zcs->cdictLocal = NULL; + ZSTD_free(zcs->inBuff, cMem); + zcs->inBuff = NULL; + ZSTD_free(zcs->outBuff, cMem); + zcs->outBuff = NULL; + ZSTD_free(zcs, cMem); + return 0; + } +} + + +/*====== Initialization ======*/ + +size_t ZSTD_CStreamInSize(void) { return ZSTD_BLOCKSIZE_ABSOLUTEMAX; } +size_t ZSTD_CStreamOutSize(void) { return ZSTD_compressBound(ZSTD_BLOCKSIZE_ABSOLUTEMAX) + ZSTD_blockHeaderSize + 4 /* 32-bits hash */ ; } + +static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize) +{ + if (zcs->inBuffSize==0) return ERROR(stage_wrong); /* zcs has not been init at least once => can't reset */ + + if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict(zcs->cctx, zcs->cdict, pledgedSrcSize)) + else CHECK_F(ZSTD_compressBegin_advanced(zcs->cctx, NULL, 0, zcs->params, pledgedSrcSize)); + + zcs->inToCompress = 0; + zcs->inBuffPos = 0; + zcs->inBuffTarget = zcs->blockSize; + zcs->outBuffContentSize = zcs->outBuffFlushedSize = 0; + zcs->stage = zcss_load; + zcs->frameEnded = 0; + zcs->pledgedSrcSize = pledgedSrcSize; + zcs->inputProcessed = 0; + return 0; /* ready to go */ +} + +size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize) +{ + + zcs->params.fParams.contentSizeFlag = (pledgedSrcSize > 0); + + return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); +} + +size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, + const void* dict, size_t dictSize, + ZSTD_parameters params, unsigned long long pledgedSrcSize) +{ + /* allocate buffers */ + { size_t const neededInBuffSize = (size_t)1 << params.cParams.windowLog; + if (zcs->inBuffSize < neededInBuffSize) { + zcs->inBuffSize = neededInBuffSize; + ZSTD_free(zcs->inBuff, zcs->customMem); + zcs->inBuff = (char*) ZSTD_malloc(neededInBuffSize, zcs->customMem); + if (zcs->inBuff == NULL) return ERROR(memory_allocation); + } + zcs->blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, neededInBuffSize); + } + if (zcs->outBuffSize < ZSTD_compressBound(zcs->blockSize)+1) { + zcs->outBuffSize = ZSTD_compressBound(zcs->blockSize)+1; + ZSTD_free(zcs->outBuff, zcs->customMem); + zcs->outBuff = (char*) ZSTD_malloc(zcs->outBuffSize, zcs->customMem); + if (zcs->outBuff == NULL) return ERROR(memory_allocation); + } + + if (dict && dictSize >= 8) { + ZSTD_freeCDict(zcs->cdictLocal); + zcs->cdictLocal = ZSTD_createCDict_advanced(dict, dictSize, 0, params, zcs->customMem); + if (zcs->cdictLocal == NULL) return ERROR(memory_allocation); + zcs->cdict = zcs->cdictLocal; + } else zcs->cdict = NULL; + + zcs->checksum = params.fParams.checksumFlag > 0; + zcs->params = params; + + return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); +} + +/* note : cdict must outlive compression session */ +size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) +{ + ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict); + size_t const initError = ZSTD_initCStream_advanced(zcs, NULL, 0, params, 0); + zcs->cdict = cdict; + zcs->cctx->dictID = params.fParams.noDictIDFlag ? 0 : cdict->refContext->dictID; + return initError; +} + +size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel) +{ + ZSTD_parameters const params = ZSTD_getParams(compressionLevel, 0, dictSize); + return ZSTD_initCStream_advanced(zcs, dict, dictSize, params, 0); +} + +size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize) +{ + ZSTD_parameters params = ZSTD_getParams(compressionLevel, pledgedSrcSize, 0); + if (pledgedSrcSize) params.fParams.contentSizeFlag = 1; + return ZSTD_initCStream_advanced(zcs, NULL, 0, params, pledgedSrcSize); +} + +size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel) +{ + return ZSTD_initCStream_usingDict(zcs, NULL, 0, compressionLevel); +} + +size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs) +{ + if (zcs==NULL) return 0; /* support sizeof on NULL */ + return sizeof(*zcs) + ZSTD_sizeof_CCtx(zcs->cctx) + ZSTD_sizeof_CDict(zcs->cdictLocal) + zcs->outBuffSize + zcs->inBuffSize; +} + +/*====== Compression ======*/ + +typedef enum { zsf_gather, zsf_flush, zsf_end } ZSTD_flush_e; + +MEM_STATIC size_t ZSTD_limitCopy(void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + size_t const length = MIN(dstCapacity, srcSize); + memcpy(dst, src, length); + return length; +} + +static size_t ZSTD_compressStream_generic(ZSTD_CStream* zcs, + void* dst, size_t* dstCapacityPtr, + const void* src, size_t* srcSizePtr, + ZSTD_flush_e const flush) +{ + U32 someMoreWork = 1; + const char* const istart = (const char*)src; + const char* const iend = istart + *srcSizePtr; + const char* ip = istart; + char* const ostart = (char*)dst; + char* const oend = ostart + *dstCapacityPtr; + char* op = ostart; + + while (someMoreWork) { + switch(zcs->stage) + { + case zcss_init: return ERROR(init_missing); /* call ZBUFF_compressInit() first ! */ + + case zcss_load: + /* complete inBuffer */ + { size_t const toLoad = zcs->inBuffTarget - zcs->inBuffPos; + size_t const loaded = ZSTD_limitCopy(zcs->inBuff + zcs->inBuffPos, toLoad, ip, iend-ip); + zcs->inBuffPos += loaded; + ip += loaded; + if ( (zcs->inBuffPos==zcs->inToCompress) || (!flush && (toLoad != loaded)) ) { + someMoreWork = 0; break; /* not enough input to get a full block : stop there, wait for more */ + } } + /* compress current block (note : this stage cannot be stopped in the middle) */ + { void* cDst; + size_t cSize; + size_t const iSize = zcs->inBuffPos - zcs->inToCompress; + size_t oSize = oend-op; + if (oSize >= ZSTD_compressBound(iSize)) + cDst = op; /* compress directly into output buffer (avoid flush stage) */ + else + cDst = zcs->outBuff, oSize = zcs->outBuffSize; + cSize = (flush == zsf_end) ? + ZSTD_compressEnd(zcs->cctx, cDst, oSize, zcs->inBuff + zcs->inToCompress, iSize) : + ZSTD_compressContinue(zcs->cctx, cDst, oSize, zcs->inBuff + zcs->inToCompress, iSize); + if (ZSTD_isError(cSize)) return cSize; + if (flush == zsf_end) zcs->frameEnded = 1; + /* prepare next block */ + zcs->inBuffTarget = zcs->inBuffPos + zcs->blockSize; + if (zcs->inBuffTarget > zcs->inBuffSize) + zcs->inBuffPos = 0, zcs->inBuffTarget = zcs->blockSize; /* note : inBuffSize >= blockSize */ + zcs->inToCompress = zcs->inBuffPos; + if (cDst == op) { op += cSize; break; } /* no need to flush */ + zcs->outBuffContentSize = cSize; + zcs->outBuffFlushedSize = 0; + zcs->stage = zcss_flush; /* pass-through to flush stage */ + } + + case zcss_flush: + { size_t const toFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize; + size_t const flushed = ZSTD_limitCopy(op, oend-op, zcs->outBuff + zcs->outBuffFlushedSize, toFlush); + op += flushed; + zcs->outBuffFlushedSize += flushed; + if (toFlush!=flushed) { someMoreWork = 0; break; } /* dst too small to store flushed data : stop there */ + zcs->outBuffContentSize = zcs->outBuffFlushedSize = 0; + zcs->stage = zcss_load; + break; + } + + case zcss_final: + someMoreWork = 0; /* do nothing */ + break; + + default: + return ERROR(GENERIC); /* impossible */ + } + } + + *srcSizePtr = ip - istart; + *dstCapacityPtr = op - ostart; + zcs->inputProcessed += *srcSizePtr; + if (zcs->frameEnded) return 0; + { size_t hintInSize = zcs->inBuffTarget - zcs->inBuffPos; + if (hintInSize==0) hintInSize = zcs->blockSize; + return hintInSize; + } +} + +size_t ZSTD_compressStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input) +{ + size_t sizeRead = input->size - input->pos; + size_t sizeWritten = output->size - output->pos; + size_t const result = ZSTD_compressStream_generic(zcs, + (char*)(output->dst) + output->pos, &sizeWritten, + (const char*)(input->src) + input->pos, &sizeRead, zsf_gather); + input->pos += sizeRead; + output->pos += sizeWritten; + return result; +} + + +/*====== Finalize ======*/ + +/*! ZSTD_flushStream() : +* @return : amount of data remaining to flush */ +size_t ZSTD_flushStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output) +{ + size_t srcSize = 0; + size_t sizeWritten = output->size - output->pos; + size_t const result = ZSTD_compressStream_generic(zcs, + (char*)(output->dst) + output->pos, &sizeWritten, + &srcSize, &srcSize, /* use a valid src address instead of NULL */ + zsf_flush); + output->pos += sizeWritten; + if (ZSTD_isError(result)) return result; + return zcs->outBuffContentSize - zcs->outBuffFlushedSize; /* remaining to flush */ +} + + +size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output) +{ + BYTE* const ostart = (BYTE*)(output->dst) + output->pos; + BYTE* const oend = (BYTE*)(output->dst) + output->size; + BYTE* op = ostart; + + if ((zcs->pledgedSrcSize) && (zcs->inputProcessed != zcs->pledgedSrcSize)) + return ERROR(srcSize_wrong); /* pledgedSrcSize not respected */ + + if (zcs->stage != zcss_final) { + /* flush whatever remains */ + size_t srcSize = 0; + size_t sizeWritten = output->size - output->pos; + size_t const notEnded = ZSTD_compressStream_generic(zcs, ostart, &sizeWritten, &srcSize, &srcSize, zsf_end); /* use a valid src address instead of NULL */ + size_t const remainingToFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize; + op += sizeWritten; + if (remainingToFlush) { + output->pos += sizeWritten; + return remainingToFlush + ZSTD_BLOCKHEADERSIZE /* final empty block */ + (zcs->checksum * 4); + } + /* create epilogue */ + zcs->stage = zcss_final; + zcs->outBuffContentSize = !notEnded ? 0 : + ZSTD_compressEnd(zcs->cctx, zcs->outBuff, zcs->outBuffSize, NULL, 0); /* write epilogue, including final empty block, into outBuff */ + } + + /* flush epilogue */ + { size_t const toFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize; + size_t const flushed = ZSTD_limitCopy(op, oend-op, zcs->outBuff + zcs->outBuffFlushedSize, toFlush); + op += flushed; + zcs->outBuffFlushedSize += flushed; + output->pos += op-ostart; + if (toFlush==flushed) zcs->stage = zcss_init; /* end reached */ + return toFlush - flushed; + } +} + + + +/*-===== Pre-defined compression levels =====-*/ + +#define ZSTD_DEFAULT_CLEVEL 1 +#define ZSTD_MAX_CLEVEL 22 +int ZSTD_maxCLevel(void) { return ZSTD_MAX_CLEVEL; } + +static const ZSTD_compressionParameters ZSTD_defaultCParameters[4][ZSTD_MAX_CLEVEL+1] = { +{ /* "default" */ + /* W, C, H, S, L, TL, strat */ + { 18, 12, 12, 1, 7, 16, ZSTD_fast }, /* level 0 - never used */ + { 19, 13, 14, 1, 7, 16, ZSTD_fast }, /* level 1 */ + { 19, 15, 16, 1, 6, 16, ZSTD_fast }, /* level 2 */ + { 20, 16, 17, 1, 5, 16, ZSTD_dfast }, /* level 3.*/ + { 20, 18, 18, 1, 5, 16, ZSTD_dfast }, /* level 4.*/ + { 20, 15, 18, 3, 5, 16, ZSTD_greedy }, /* level 5 */ + { 21, 16, 19, 2, 5, 16, ZSTD_lazy }, /* level 6 */ + { 21, 17, 20, 3, 5, 16, ZSTD_lazy }, /* level 7 */ + { 21, 18, 20, 3, 5, 16, ZSTD_lazy2 }, /* level 8 */ + { 21, 20, 20, 3, 5, 16, ZSTD_lazy2 }, /* level 9 */ + { 21, 19, 21, 4, 5, 16, ZSTD_lazy2 }, /* level 10 */ + { 22, 20, 22, 4, 5, 16, ZSTD_lazy2 }, /* level 11 */ + { 22, 20, 22, 5, 5, 16, ZSTD_lazy2 }, /* level 12 */ + { 22, 21, 22, 5, 5, 16, ZSTD_lazy2 }, /* level 13 */ + { 22, 21, 22, 6, 5, 16, ZSTD_lazy2 }, /* level 14 */ + { 22, 21, 21, 5, 5, 16, ZSTD_btlazy2 }, /* level 15 */ + { 23, 22, 22, 5, 5, 16, ZSTD_btlazy2 }, /* level 16 */ + { 23, 21, 22, 4, 5, 24, ZSTD_btopt }, /* level 17 */ + { 23, 23, 22, 6, 5, 32, ZSTD_btopt }, /* level 18 */ + { 23, 23, 22, 6, 3, 48, ZSTD_btopt }, /* level 19 */ + { 25, 25, 23, 7, 3, 64, ZSTD_btopt2 }, /* level 20 */ + { 26, 26, 23, 7, 3,256, ZSTD_btopt2 }, /* level 21 */ + { 27, 27, 25, 9, 3,512, ZSTD_btopt2 }, /* level 22 */ +}, +{ /* for srcSize <= 256 KB */ + /* W, C, H, S, L, T, strat */ + { 0, 0, 0, 0, 0, 0, ZSTD_fast }, /* level 0 - not used */ + { 18, 13, 14, 1, 6, 8, ZSTD_fast }, /* level 1 */ + { 18, 14, 13, 1, 5, 8, ZSTD_dfast }, /* level 2 */ + { 18, 16, 15, 1, 5, 8, ZSTD_dfast }, /* level 3 */ + { 18, 15, 17, 1, 5, 8, ZSTD_greedy }, /* level 4.*/ + { 18, 16, 17, 4, 5, 8, ZSTD_greedy }, /* level 5.*/ + { 18, 16, 17, 3, 5, 8, ZSTD_lazy }, /* level 6.*/ + { 18, 17, 17, 4, 4, 8, ZSTD_lazy }, /* level 7 */ + { 18, 17, 17, 4, 4, 8, ZSTD_lazy2 }, /* level 8 */ + { 18, 17, 17, 5, 4, 8, ZSTD_lazy2 }, /* level 9 */ + { 18, 17, 17, 6, 4, 8, ZSTD_lazy2 }, /* level 10 */ + { 18, 18, 17, 6, 4, 8, ZSTD_lazy2 }, /* level 11.*/ + { 18, 18, 17, 7, 4, 8, ZSTD_lazy2 }, /* level 12.*/ + { 18, 19, 17, 6, 4, 8, ZSTD_btlazy2 }, /* level 13 */ + { 18, 18, 18, 4, 4, 16, ZSTD_btopt }, /* level 14.*/ + { 18, 18, 18, 4, 3, 16, ZSTD_btopt }, /* level 15.*/ + { 18, 19, 18, 6, 3, 32, ZSTD_btopt }, /* level 16.*/ + { 18, 19, 18, 8, 3, 64, ZSTD_btopt }, /* level 17.*/ + { 18, 19, 18, 9, 3,128, ZSTD_btopt }, /* level 18.*/ + { 18, 19, 18, 10, 3,256, ZSTD_btopt }, /* level 19.*/ + { 18, 19, 18, 11, 3,512, ZSTD_btopt2 }, /* level 20.*/ + { 18, 19, 18, 12, 3,512, ZSTD_btopt2 }, /* level 21.*/ + { 18, 19, 18, 13, 3,512, ZSTD_btopt2 }, /* level 22.*/ +}, +{ /* for srcSize <= 128 KB */ + /* W, C, H, S, L, T, strat */ + { 17, 12, 12, 1, 7, 8, ZSTD_fast }, /* level 0 - not used */ + { 17, 12, 13, 1, 6, 8, ZSTD_fast }, /* level 1 */ + { 17, 13, 16, 1, 5, 8, ZSTD_fast }, /* level 2 */ + { 17, 16, 16, 2, 5, 8, ZSTD_dfast }, /* level 3 */ + { 17, 13, 15, 3, 4, 8, ZSTD_greedy }, /* level 4 */ + { 17, 15, 17, 4, 4, 8, ZSTD_greedy }, /* level 5 */ + { 17, 16, 17, 3, 4, 8, ZSTD_lazy }, /* level 6 */ + { 17, 15, 17, 4, 4, 8, ZSTD_lazy2 }, /* level 7 */ + { 17, 17, 17, 4, 4, 8, ZSTD_lazy2 }, /* level 8 */ + { 17, 17, 17, 5, 4, 8, ZSTD_lazy2 }, /* level 9 */ + { 17, 17, 17, 6, 4, 8, ZSTD_lazy2 }, /* level 10 */ + { 17, 17, 17, 7, 4, 8, ZSTD_lazy2 }, /* level 11 */ + { 17, 17, 17, 8, 4, 8, ZSTD_lazy2 }, /* level 12 */ + { 17, 18, 17, 6, 4, 8, ZSTD_btlazy2 }, /* level 13.*/ + { 17, 17, 17, 7, 3, 8, ZSTD_btopt }, /* level 14.*/ + { 17, 17, 17, 7, 3, 16, ZSTD_btopt }, /* level 15.*/ + { 17, 18, 17, 7, 3, 32, ZSTD_btopt }, /* level 16.*/ + { 17, 18, 17, 7, 3, 64, ZSTD_btopt }, /* level 17.*/ + { 17, 18, 17, 7, 3,256, ZSTD_btopt }, /* level 18.*/ + { 17, 18, 17, 8, 3,256, ZSTD_btopt }, /* level 19.*/ + { 17, 18, 17, 9, 3,256, ZSTD_btopt2 }, /* level 20.*/ + { 17, 18, 17, 10, 3,256, ZSTD_btopt2 }, /* level 21.*/ + { 17, 18, 17, 11, 3,512, ZSTD_btopt2 }, /* level 22.*/ +}, +{ /* for srcSize <= 16 KB */ + /* W, C, H, S, L, T, strat */ + { 14, 12, 12, 1, 7, 6, ZSTD_fast }, /* level 0 - not used */ + { 14, 14, 14, 1, 6, 6, ZSTD_fast }, /* level 1 */ + { 14, 14, 14, 1, 4, 6, ZSTD_fast }, /* level 2 */ + { 14, 14, 14, 1, 4, 6, ZSTD_dfast }, /* level 3.*/ + { 14, 14, 14, 4, 4, 6, ZSTD_greedy }, /* level 4.*/ + { 14, 14, 14, 3, 4, 6, ZSTD_lazy }, /* level 5.*/ + { 14, 14, 14, 4, 4, 6, ZSTD_lazy2 }, /* level 6 */ + { 14, 14, 14, 5, 4, 6, ZSTD_lazy2 }, /* level 7 */ + { 14, 14, 14, 6, 4, 6, ZSTD_lazy2 }, /* level 8.*/ + { 14, 15, 14, 6, 4, 6, ZSTD_btlazy2 }, /* level 9.*/ + { 14, 15, 14, 3, 3, 6, ZSTD_btopt }, /* level 10.*/ + { 14, 15, 14, 6, 3, 8, ZSTD_btopt }, /* level 11.*/ + { 14, 15, 14, 6, 3, 16, ZSTD_btopt }, /* level 12.*/ + { 14, 15, 14, 6, 3, 24, ZSTD_btopt }, /* level 13.*/ + { 14, 15, 15, 6, 3, 48, ZSTD_btopt }, /* level 14.*/ + { 14, 15, 15, 6, 3, 64, ZSTD_btopt }, /* level 15.*/ + { 14, 15, 15, 6, 3, 96, ZSTD_btopt }, /* level 16.*/ + { 14, 15, 15, 6, 3,128, ZSTD_btopt }, /* level 17.*/ + { 14, 15, 15, 6, 3,256, ZSTD_btopt }, /* level 18.*/ + { 14, 15, 15, 7, 3,256, ZSTD_btopt }, /* level 19.*/ + { 14, 15, 15, 8, 3,256, ZSTD_btopt2 }, /* level 20.*/ + { 14, 15, 15, 9, 3,256, ZSTD_btopt2 }, /* level 21.*/ + { 14, 15, 15, 10, 3,256, ZSTD_btopt2 }, /* level 22.*/ +}, +}; + +/*! ZSTD_getCParams() : +* @return ZSTD_compressionParameters structure for a selected compression level, `srcSize` and `dictSize`. +* Size values are optional, provide 0 if not known or unused */ +ZSTD_compressionParameters ZSTD_getCParams(int compressionLevel, unsigned long long srcSize, size_t dictSize) +{ + ZSTD_compressionParameters cp; + size_t const addedSize = srcSize ? 0 : 500; + U64 const rSize = srcSize+dictSize ? srcSize+dictSize+addedSize : (U64)-1; + U32 const tableID = (rSize <= 256 KB) + (rSize <= 128 KB) + (rSize <= 16 KB); /* intentional underflow for srcSizeHint == 0 */ + if (compressionLevel <= 0) compressionLevel = ZSTD_DEFAULT_CLEVEL; /* 0 == default; no negative compressionLevel yet */ + if (compressionLevel > ZSTD_MAX_CLEVEL) compressionLevel = ZSTD_MAX_CLEVEL; + cp = ZSTD_defaultCParameters[tableID][compressionLevel]; + if (MEM_32bits()) { /* auto-correction, for 32-bits mode */ + if (cp.windowLog > ZSTD_WINDOWLOG_MAX) cp.windowLog = ZSTD_WINDOWLOG_MAX; + if (cp.chainLog > ZSTD_CHAINLOG_MAX) cp.chainLog = ZSTD_CHAINLOG_MAX; + if (cp.hashLog > ZSTD_HASHLOG_MAX) cp.hashLog = ZSTD_HASHLOG_MAX; + } + cp = ZSTD_adjustCParams(cp, srcSize, dictSize); + return cp; +} + +/*! ZSTD_getParams() : +* same as ZSTD_getCParams(), but @return a `ZSTD_parameters` object (instead of `ZSTD_compressionParameters`). +* All fields of `ZSTD_frameParameters` are set to default (0) */ +ZSTD_parameters ZSTD_getParams(int compressionLevel, unsigned long long srcSize, size_t dictSize) { + ZSTD_parameters params; + ZSTD_compressionParameters const cParams = ZSTD_getCParams(compressionLevel, srcSize, dictSize); + memset(¶ms, 0, sizeof(params)); + params.cParams = cParams; + return params; +} diff --git a/contrib/linux-kernel/lib/zstd_decompress.c b/contrib/linux-kernel/lib/zstd_decompress.c new file mode 100644 index 000000000..06337dbd5 --- /dev/null +++ b/contrib/linux-kernel/lib/zstd_decompress.c @@ -0,0 +1,2484 @@ +/** + * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + + +/* *************************************************************** +* Tuning parameters +*****************************************************************/ +/*! + * HEAPMODE : + * Select how default decompression function ZSTD_decompress() will allocate memory, + * in memory stack (0), or in memory heap (1, requires malloc()) + */ +#ifndef ZSTD_HEAPMODE +# define ZSTD_HEAPMODE 1 +#endif + +/*! +* LEGACY_SUPPORT : +* if set to 1, ZSTD_decompress() can decode older formats (v0.1+) +*/ +#ifndef ZSTD_LEGACY_SUPPORT +# define ZSTD_LEGACY_SUPPORT 0 +#endif + +/*! +* MAXWINDOWSIZE_DEFAULT : +* maximum window size accepted by DStream, by default. +* Frames requiring more memory will be rejected. +*/ +#ifndef ZSTD_MAXWINDOWSIZE_DEFAULT +# define ZSTD_MAXWINDOWSIZE_DEFAULT ((1 << ZSTD_WINDOWLOG_MAX) + 1) /* defined within zstd.h */ +#endif + + +/*-******************************************************* +* Dependencies +*********************************************************/ +#include /* memcpy, memmove, memset */ +#include "mem.h" /* low level memory routines */ +#define FSE_STATIC_LINKING_ONLY +#include "fse.h" +#define HUF_STATIC_LINKING_ONLY +#include "huf.h" +#include "zstd_internal.h" + +#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1) +# include "zstd_legacy.h" +#endif + + +#if defined(_MSC_VER) +# include /* https://msdn.microsoft.com/fr-fr/library/84szxsww(v=vs.90).aspx */ +# define ZSTD_PREFETCH(ptr) _mm_prefetch((const char*)ptr, _MM_HINT_T0) +#elif defined(__GNUC__) +# define ZSTD_PREFETCH(ptr) __builtin_prefetch(ptr, 0, 0) +#else +# define ZSTD_PREFETCH(ptr) /* disabled */ +#endif + +/*-************************************* +* Macros +***************************************/ +#define ZSTD_isError ERR_isError /* for inlining */ +#define FSE_isError ERR_isError +#define HUF_isError ERR_isError + + +/*_******************************************************* +* Memory operations +**********************************************************/ +static void ZSTD_copy4(void* dst, const void* src) { memcpy(dst, src, 4); } + + +/*-************************************************************* +* Context management +***************************************************************/ +typedef enum { ZSTDds_getFrameHeaderSize, ZSTDds_decodeFrameHeader, + ZSTDds_decodeBlockHeader, ZSTDds_decompressBlock, + ZSTDds_decompressLastBlock, ZSTDds_checkChecksum, + ZSTDds_decodeSkippableHeader, ZSTDds_skipFrame } ZSTD_dStage; + +typedef struct { + FSE_DTable LLTable[FSE_DTABLE_SIZE_U32(LLFSELog)]; + FSE_DTable OFTable[FSE_DTABLE_SIZE_U32(OffFSELog)]; + FSE_DTable MLTable[FSE_DTABLE_SIZE_U32(MLFSELog)]; + HUF_DTable hufTable[HUF_DTABLE_SIZE(HufLog)]; /* can accommodate HUF_decompress4X */ + U32 rep[ZSTD_REP_NUM]; +} ZSTD_entropyTables_t; + +struct ZSTD_DCtx_s +{ + const FSE_DTable* LLTptr; + const FSE_DTable* MLTptr; + const FSE_DTable* OFTptr; + const HUF_DTable* HUFptr; + ZSTD_entropyTables_t entropy; + const void* previousDstEnd; /* detect continuity */ + const void* base; /* start of current segment */ + const void* vBase; /* virtual start of previous segment if it was just before current one */ + const void* dictEnd; /* end of previous segment */ + size_t expected; + ZSTD_frameParams fParams; + blockType_e bType; /* used in ZSTD_decompressContinue(), to transfer blockType between header decoding and block decoding stages */ + ZSTD_dStage stage; + U32 litEntropy; + U32 fseEntropy; + XXH64_state_t xxhState; + size_t headerSize; + U32 dictID; + const BYTE* litPtr; + ZSTD_customMem customMem; + size_t litSize; + size_t rleSize; + BYTE litBuffer[ZSTD_BLOCKSIZE_ABSOLUTEMAX + WILDCOPY_OVERLENGTH]; + BYTE headerBuffer[ZSTD_FRAMEHEADERSIZE_MAX]; +}; /* typedef'd to ZSTD_DCtx within "zstd.h" */ + +size_t ZSTD_sizeof_DCtx (const ZSTD_DCtx* dctx) { return (dctx==NULL) ? 0 : sizeof(ZSTD_DCtx); } + +size_t ZSTD_estimateDCtxSize(void) { return sizeof(ZSTD_DCtx); } + +size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx) +{ + dctx->expected = ZSTD_frameHeaderSize_prefix; + dctx->stage = ZSTDds_getFrameHeaderSize; + dctx->previousDstEnd = NULL; + dctx->base = NULL; + dctx->vBase = NULL; + dctx->dictEnd = NULL; + dctx->entropy.hufTable[0] = (HUF_DTable)((HufLog)*0x1000001); /* cover both little and big endian */ + dctx->litEntropy = dctx->fseEntropy = 0; + dctx->dictID = 0; + MEM_STATIC_ASSERT(sizeof(dctx->entropy.rep) == sizeof(repStartValue)); + memcpy(dctx->entropy.rep, repStartValue, sizeof(repStartValue)); /* initial repcodes */ + dctx->LLTptr = dctx->entropy.LLTable; + dctx->MLTptr = dctx->entropy.MLTable; + dctx->OFTptr = dctx->entropy.OFTable; + dctx->HUFptr = dctx->entropy.hufTable; + return 0; +} + +ZSTD_DCtx* ZSTD_createDCtx_advanced(ZSTD_customMem customMem) +{ + ZSTD_DCtx* dctx; + + if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; + if (!customMem.customAlloc || !customMem.customFree) return NULL; + + dctx = (ZSTD_DCtx*)ZSTD_malloc(sizeof(ZSTD_DCtx), customMem); + if (!dctx) return NULL; + memcpy(&dctx->customMem, &customMem, sizeof(customMem)); + ZSTD_decompressBegin(dctx); + return dctx; +} + +ZSTD_DCtx* ZSTD_createDCtx(void) +{ + return ZSTD_createDCtx_advanced(defaultCustomMem); +} + +size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx) +{ + if (dctx==NULL) return 0; /* support free on NULL */ + ZSTD_free(dctx, dctx->customMem); + return 0; /* reserved as a potential error code in the future */ +} + +void ZSTD_copyDCtx(ZSTD_DCtx* dstDCtx, const ZSTD_DCtx* srcDCtx) +{ + size_t const workSpaceSize = (ZSTD_BLOCKSIZE_ABSOLUTEMAX+WILDCOPY_OVERLENGTH) + ZSTD_frameHeaderSize_max; + memcpy(dstDCtx, srcDCtx, sizeof(ZSTD_DCtx) - workSpaceSize); /* no need to copy workspace */ +} + +#if 0 +/* deprecated */ +static void ZSTD_refDCtx(ZSTD_DCtx* dstDCtx, const ZSTD_DCtx* srcDCtx) +{ + ZSTD_decompressBegin(dstDCtx); /* init */ + if (srcDCtx) { /* support refDCtx on NULL */ + dstDCtx->dictEnd = srcDCtx->dictEnd; + dstDCtx->vBase = srcDCtx->vBase; + dstDCtx->base = srcDCtx->base; + dstDCtx->previousDstEnd = srcDCtx->previousDstEnd; + dstDCtx->dictID = srcDCtx->dictID; + dstDCtx->litEntropy = srcDCtx->litEntropy; + dstDCtx->fseEntropy = srcDCtx->fseEntropy; + dstDCtx->LLTptr = srcDCtx->entropy.LLTable; + dstDCtx->MLTptr = srcDCtx->entropy.MLTable; + dstDCtx->OFTptr = srcDCtx->entropy.OFTable; + dstDCtx->HUFptr = srcDCtx->entropy.hufTable; + dstDCtx->entropy.rep[0] = srcDCtx->entropy.rep[0]; + dstDCtx->entropy.rep[1] = srcDCtx->entropy.rep[1]; + dstDCtx->entropy.rep[2] = srcDCtx->entropy.rep[2]; + } +} +#endif + +static void ZSTD_refDDict(ZSTD_DCtx* dstDCtx, const ZSTD_DDict* ddict); + + +/*-************************************************************* +* Decompression section +***************************************************************/ + +/*! ZSTD_isFrame() : + * Tells if the content of `buffer` starts with a valid Frame Identifier. + * Note : Frame Identifier is 4 bytes. If `size < 4`, @return will always be 0. + * Note 2 : Legacy Frame Identifiers are considered valid only if Legacy Support is enabled. + * Note 3 : Skippable Frame Identifiers are considered valid. */ +unsigned ZSTD_isFrame(const void* buffer, size_t size) +{ + if (size < 4) return 0; + { U32 const magic = MEM_readLE32(buffer); + if (magic == ZSTD_MAGICNUMBER) return 1; + if ((magic & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) return 1; + } +#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) + if (ZSTD_isLegacy(buffer, size)) return 1; +#endif + return 0; +} + + +/** ZSTD_frameHeaderSize() : +* srcSize must be >= ZSTD_frameHeaderSize_prefix. +* @return : size of the Frame Header */ +static size_t ZSTD_frameHeaderSize(const void* src, size_t srcSize) +{ + if (srcSize < ZSTD_frameHeaderSize_prefix) return ERROR(srcSize_wrong); + { BYTE const fhd = ((const BYTE*)src)[4]; + U32 const dictID= fhd & 3; + U32 const singleSegment = (fhd >> 5) & 1; + U32 const fcsId = fhd >> 6; + return ZSTD_frameHeaderSize_prefix + !singleSegment + ZSTD_did_fieldSize[dictID] + ZSTD_fcs_fieldSize[fcsId] + + (singleSegment && !fcsId); + } +} + + +/** ZSTD_getFrameParams() : +* decode Frame Header, or require larger `srcSize`. +* @return : 0, `fparamsPtr` is correctly filled, +* >0, `srcSize` is too small, result is expected `srcSize`, +* or an error code, which can be tested using ZSTD_isError() */ +size_t ZSTD_getFrameParams(ZSTD_frameParams* fparamsPtr, const void* src, size_t srcSize) +{ + const BYTE* ip = (const BYTE*)src; + + if (srcSize < ZSTD_frameHeaderSize_prefix) return ZSTD_frameHeaderSize_prefix; + if (MEM_readLE32(src) != ZSTD_MAGICNUMBER) { + if ((MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { + if (srcSize < ZSTD_skippableHeaderSize) return ZSTD_skippableHeaderSize; /* magic number + skippable frame length */ + memset(fparamsPtr, 0, sizeof(*fparamsPtr)); + fparamsPtr->frameContentSize = MEM_readLE32((const char *)src + 4); + fparamsPtr->windowSize = 0; /* windowSize==0 means a frame is skippable */ + return 0; + } + return ERROR(prefix_unknown); + } + + /* ensure there is enough `srcSize` to fully read/decode frame header */ + { size_t const fhsize = ZSTD_frameHeaderSize(src, srcSize); + if (srcSize < fhsize) return fhsize; } + + { BYTE const fhdByte = ip[4]; + size_t pos = 5; + U32 const dictIDSizeCode = fhdByte&3; + U32 const checksumFlag = (fhdByte>>2)&1; + U32 const singleSegment = (fhdByte>>5)&1; + U32 const fcsID = fhdByte>>6; + U32 const windowSizeMax = 1U << ZSTD_WINDOWLOG_MAX; + U32 windowSize = 0; + U32 dictID = 0; + U64 frameContentSize = 0; + if ((fhdByte & 0x08) != 0) return ERROR(frameParameter_unsupported); /* reserved bits, which must be zero */ + if (!singleSegment) { + BYTE const wlByte = ip[pos++]; + U32 const windowLog = (wlByte >> 3) + ZSTD_WINDOWLOG_ABSOLUTEMIN; + if (windowLog > ZSTD_WINDOWLOG_MAX) return ERROR(frameParameter_windowTooLarge); /* avoids issue with 1 << windowLog */ + windowSize = (1U << windowLog); + windowSize += (windowSize >> 3) * (wlByte&7); + } + + switch(dictIDSizeCode) + { + default: /* impossible */ + case 0 : break; + case 1 : dictID = ip[pos]; pos++; break; + case 2 : dictID = MEM_readLE16(ip+pos); pos+=2; break; + case 3 : dictID = MEM_readLE32(ip+pos); pos+=4; break; + } + switch(fcsID) + { + default: /* impossible */ + case 0 : if (singleSegment) frameContentSize = ip[pos]; break; + case 1 : frameContentSize = MEM_readLE16(ip+pos)+256; break; + case 2 : frameContentSize = MEM_readLE32(ip+pos); break; + case 3 : frameContentSize = MEM_readLE64(ip+pos); break; + } + if (!windowSize) windowSize = (U32)frameContentSize; + if (windowSize > windowSizeMax) return ERROR(frameParameter_windowTooLarge); + fparamsPtr->frameContentSize = frameContentSize; + fparamsPtr->windowSize = windowSize; + fparamsPtr->dictID = dictID; + fparamsPtr->checksumFlag = checksumFlag; + } + return 0; +} + +/** ZSTD_getFrameContentSize() : +* compatible with legacy mode +* @return : decompressed size of the single frame pointed to be `src` if known, otherwise +* - ZSTD_CONTENTSIZE_UNKNOWN if the size cannot be determined +* - ZSTD_CONTENTSIZE_ERROR if an error occurred (e.g. invalid magic number, srcSize too small) */ +unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize) +{ +#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) + if (ZSTD_isLegacy(src, srcSize)) { + unsigned long long const ret = ZSTD_getDecompressedSize_legacy(src, srcSize); + return ret == 0 ? ZSTD_CONTENTSIZE_UNKNOWN : ret; + } +#endif + { + ZSTD_frameParams fParams; + if (ZSTD_getFrameParams(&fParams, src, srcSize) != 0) return ZSTD_CONTENTSIZE_ERROR; + if (fParams.windowSize == 0) { + /* Either skippable or empty frame, size == 0 either way */ + return 0; + } else if (fParams.frameContentSize != 0) { + return fParams.frameContentSize; + } else { + return ZSTD_CONTENTSIZE_UNKNOWN; + } + } +} + +/** ZSTD_findDecompressedSize() : + * compatible with legacy mode + * `srcSize` must be the exact length of some number of ZSTD compressed and/or + * skippable frames + * @return : decompressed size of the frames contained */ +unsigned long long ZSTD_findDecompressedSize(const void* src, size_t srcSize) +{ + { + unsigned long long totalDstSize = 0; + while (srcSize >= ZSTD_frameHeaderSize_prefix) { + const U32 magicNumber = MEM_readLE32(src); + + if ((magicNumber & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { + size_t skippableSize; + if (srcSize < ZSTD_skippableHeaderSize) + return ERROR(srcSize_wrong); + skippableSize = MEM_readLE32((const BYTE *)src + 4) + + ZSTD_skippableHeaderSize; + if (srcSize < skippableSize) { + return ZSTD_CONTENTSIZE_ERROR; + } + + src = (const BYTE *)src + skippableSize; + srcSize -= skippableSize; + continue; + } + + { + unsigned long long const ret = ZSTD_getFrameContentSize(src, srcSize); + if (ret >= ZSTD_CONTENTSIZE_ERROR) return ret; + + /* check for overflow */ + if (totalDstSize + ret < totalDstSize) return ZSTD_CONTENTSIZE_ERROR; + totalDstSize += ret; + } + { + size_t const frameSrcSize = ZSTD_findFrameCompressedSize(src, srcSize); + if (ZSTD_isError(frameSrcSize)) { + return ZSTD_CONTENTSIZE_ERROR; + } + + src = (const BYTE *)src + frameSrcSize; + srcSize -= frameSrcSize; + } + } + + if (srcSize) { + return ZSTD_CONTENTSIZE_ERROR; + } + + return totalDstSize; + } +} + +/** ZSTD_getDecompressedSize() : +* compatible with legacy mode +* @return : decompressed size if known, 0 otherwise + note : 0 can mean any of the following : + - decompressed size is not present within frame header + - frame header unknown / not supported + - frame header not complete (`srcSize` too small) */ +unsigned long long ZSTD_getDecompressedSize(const void* src, size_t srcSize) +{ + unsigned long long const ret = ZSTD_getFrameContentSize(src, srcSize); + return ret >= ZSTD_CONTENTSIZE_ERROR ? 0 : ret; +} + + +/** ZSTD_decodeFrameHeader() : +* `headerSize` must be the size provided by ZSTD_frameHeaderSize(). +* @return : 0 if success, or an error code, which can be tested using ZSTD_isError() */ +static size_t ZSTD_decodeFrameHeader(ZSTD_DCtx* dctx, const void* src, size_t headerSize) +{ + size_t const result = ZSTD_getFrameParams(&(dctx->fParams), src, headerSize); + if (ZSTD_isError(result)) return result; /* invalid header */ + if (result>0) return ERROR(srcSize_wrong); /* headerSize too small */ + if (dctx->fParams.dictID && (dctx->dictID != dctx->fParams.dictID)) return ERROR(dictionary_wrong); + if (dctx->fParams.checksumFlag) XXH64_reset(&dctx->xxhState, 0); + return 0; +} + + +typedef struct +{ + blockType_e blockType; + U32 lastBlock; + U32 origSize; +} blockProperties_t; + +/*! ZSTD_getcBlockSize() : +* Provides the size of compressed block from block header `src` */ +size_t ZSTD_getcBlockSize(const void* src, size_t srcSize, blockProperties_t* bpPtr) +{ + if (srcSize < ZSTD_blockHeaderSize) return ERROR(srcSize_wrong); + { U32 const cBlockHeader = MEM_readLE24(src); + U32 const cSize = cBlockHeader >> 3; + bpPtr->lastBlock = cBlockHeader & 1; + bpPtr->blockType = (blockType_e)((cBlockHeader >> 1) & 3); + bpPtr->origSize = cSize; /* only useful for RLE */ + if (bpPtr->blockType == bt_rle) return 1; + if (bpPtr->blockType == bt_reserved) return ERROR(corruption_detected); + return cSize; + } +} + + +static size_t ZSTD_copyRawBlock(void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + if (srcSize > dstCapacity) return ERROR(dstSize_tooSmall); + memcpy(dst, src, srcSize); + return srcSize; +} + + +static size_t ZSTD_setRleBlock(void* dst, size_t dstCapacity, const void* src, size_t srcSize, size_t regenSize) +{ + if (srcSize != 1) return ERROR(srcSize_wrong); + if (regenSize > dstCapacity) return ERROR(dstSize_tooSmall); + memset(dst, *(const BYTE*)src, regenSize); + return regenSize; +} + +/*! ZSTD_decodeLiteralsBlock() : + @return : nb of bytes read from src (< srcSize ) */ +size_t ZSTD_decodeLiteralsBlock(ZSTD_DCtx* dctx, + const void* src, size_t srcSize) /* note : srcSize < BLOCKSIZE */ +{ + if (srcSize < MIN_CBLOCK_SIZE) return ERROR(corruption_detected); + + { const BYTE* const istart = (const BYTE*) src; + symbolEncodingType_e const litEncType = (symbolEncodingType_e)(istart[0] & 3); + + switch(litEncType) + { + case set_repeat: + if (dctx->litEntropy==0) return ERROR(dictionary_corrupted); + /* fall-through */ + case set_compressed: + if (srcSize < 5) return ERROR(corruption_detected); /* srcSize >= MIN_CBLOCK_SIZE == 3; here we need up to 5 for case 3 */ + { size_t lhSize, litSize, litCSize; + U32 singleStream=0; + U32 const lhlCode = (istart[0] >> 2) & 3; + U32 const lhc = MEM_readLE32(istart); + switch(lhlCode) + { + case 0: case 1: default: /* note : default is impossible, since lhlCode into [0..3] */ + /* 2 - 2 - 10 - 10 */ + singleStream = !lhlCode; + lhSize = 3; + litSize = (lhc >> 4) & 0x3FF; + litCSize = (lhc >> 14) & 0x3FF; + break; + case 2: + /* 2 - 2 - 14 - 14 */ + lhSize = 4; + litSize = (lhc >> 4) & 0x3FFF; + litCSize = lhc >> 18; + break; + case 3: + /* 2 - 2 - 18 - 18 */ + lhSize = 5; + litSize = (lhc >> 4) & 0x3FFFF; + litCSize = (lhc >> 22) + (istart[4] << 10); + break; + } + if (litSize > ZSTD_BLOCKSIZE_ABSOLUTEMAX) return ERROR(corruption_detected); + if (litCSize + lhSize > srcSize) return ERROR(corruption_detected); + + if (HUF_isError((litEncType==set_repeat) ? + ( singleStream ? + HUF_decompress1X_usingDTable(dctx->litBuffer, litSize, istart+lhSize, litCSize, dctx->HUFptr) : + HUF_decompress4X_usingDTable(dctx->litBuffer, litSize, istart+lhSize, litCSize, dctx->HUFptr) ) : + ( singleStream ? + HUF_decompress1X2_DCtx(dctx->entropy.hufTable, dctx->litBuffer, litSize, istart+lhSize, litCSize) : + HUF_decompress4X_hufOnly (dctx->entropy.hufTable, dctx->litBuffer, litSize, istart+lhSize, litCSize)) )) + return ERROR(corruption_detected); + + dctx->litPtr = dctx->litBuffer; + dctx->litSize = litSize; + dctx->litEntropy = 1; + if (litEncType==set_compressed) dctx->HUFptr = dctx->entropy.hufTable; + memset(dctx->litBuffer + dctx->litSize, 0, WILDCOPY_OVERLENGTH); + return litCSize + lhSize; + } + + case set_basic: + { size_t litSize, lhSize; + U32 const lhlCode = ((istart[0]) >> 2) & 3; + switch(lhlCode) + { + case 0: case 2: default: /* note : default is impossible, since lhlCode into [0..3] */ + lhSize = 1; + litSize = istart[0] >> 3; + break; + case 1: + lhSize = 2; + litSize = MEM_readLE16(istart) >> 4; + break; + case 3: + lhSize = 3; + litSize = MEM_readLE24(istart) >> 4; + break; + } + + if (lhSize+litSize+WILDCOPY_OVERLENGTH > srcSize) { /* risk reading beyond src buffer with wildcopy */ + if (litSize+lhSize > srcSize) return ERROR(corruption_detected); + memcpy(dctx->litBuffer, istart+lhSize, litSize); + dctx->litPtr = dctx->litBuffer; + dctx->litSize = litSize; + memset(dctx->litBuffer + dctx->litSize, 0, WILDCOPY_OVERLENGTH); + return lhSize+litSize; + } + /* direct reference into compressed stream */ + dctx->litPtr = istart+lhSize; + dctx->litSize = litSize; + return lhSize+litSize; + } + + case set_rle: + { U32 const lhlCode = ((istart[0]) >> 2) & 3; + size_t litSize, lhSize; + switch(lhlCode) + { + case 0: case 2: default: /* note : default is impossible, since lhlCode into [0..3] */ + lhSize = 1; + litSize = istart[0] >> 3; + break; + case 1: + lhSize = 2; + litSize = MEM_readLE16(istart) >> 4; + break; + case 3: + lhSize = 3; + litSize = MEM_readLE24(istart) >> 4; + if (srcSize<4) return ERROR(corruption_detected); /* srcSize >= MIN_CBLOCK_SIZE == 3; here we need lhSize+1 = 4 */ + break; + } + if (litSize > ZSTD_BLOCKSIZE_ABSOLUTEMAX) return ERROR(corruption_detected); + memset(dctx->litBuffer, istart[lhSize], litSize + WILDCOPY_OVERLENGTH); + dctx->litPtr = dctx->litBuffer; + dctx->litSize = litSize; + return lhSize+1; + } + default: + return ERROR(corruption_detected); /* impossible */ + } + } +} + + +typedef union { + FSE_decode_t realData; + U32 alignedBy4; +} FSE_decode_t4; + +static const FSE_decode_t4 LL_defaultDTable[(1< max) return ERROR(corruption_detected); + FSE_buildDTable_rle(DTableSpace, *(const BYTE*)src); + *DTablePtr = DTableSpace; + return 1; + case set_basic : + *DTablePtr = (const FSE_DTable*)tmpPtr; + return 0; + case set_repeat: + if (!flagRepeatTable) return ERROR(corruption_detected); + return 0; + default : /* impossible */ + case set_compressed : + { U32 tableLog; + S16 norm[MaxSeq+1]; + size_t const headerSize = FSE_readNCount(norm, &max, &tableLog, src, srcSize); + if (FSE_isError(headerSize)) return ERROR(corruption_detected); + if (tableLog > maxLog) return ERROR(corruption_detected); + FSE_buildDTable(DTableSpace, norm, max, tableLog); + *DTablePtr = DTableSpace; + return headerSize; + } } +} + +size_t ZSTD_decodeSeqHeaders(ZSTD_DCtx* dctx, int* nbSeqPtr, + const void* src, size_t srcSize) +{ + const BYTE* const istart = (const BYTE* const)src; + const BYTE* const iend = istart + srcSize; + const BYTE* ip = istart; + + /* check */ + if (srcSize < MIN_SEQUENCES_SIZE) return ERROR(srcSize_wrong); + + /* SeqHead */ + { int nbSeq = *ip++; + if (!nbSeq) { *nbSeqPtr=0; return 1; } + if (nbSeq > 0x7F) { + if (nbSeq == 0xFF) { + if (ip+2 > iend) return ERROR(srcSize_wrong); + nbSeq = MEM_readLE16(ip) + LONGNBSEQ, ip+=2; + } else { + if (ip >= iend) return ERROR(srcSize_wrong); + nbSeq = ((nbSeq-0x80)<<8) + *ip++; + } + } + *nbSeqPtr = nbSeq; + } + + /* FSE table descriptors */ + if (ip+4 > iend) return ERROR(srcSize_wrong); /* minimum possible size */ + { symbolEncodingType_e const LLtype = (symbolEncodingType_e)(*ip >> 6); + symbolEncodingType_e const OFtype = (symbolEncodingType_e)((*ip >> 4) & 3); + symbolEncodingType_e const MLtype = (symbolEncodingType_e)((*ip >> 2) & 3); + ip++; + + /* Build DTables */ + { size_t const llhSize = ZSTD_buildSeqTable(dctx->entropy.LLTable, &dctx->LLTptr, + LLtype, MaxLL, LLFSELog, + ip, iend-ip, LL_defaultDTable, dctx->fseEntropy); + if (ZSTD_isError(llhSize)) return ERROR(corruption_detected); + ip += llhSize; + } + { size_t const ofhSize = ZSTD_buildSeqTable(dctx->entropy.OFTable, &dctx->OFTptr, + OFtype, MaxOff, OffFSELog, + ip, iend-ip, OF_defaultDTable, dctx->fseEntropy); + if (ZSTD_isError(ofhSize)) return ERROR(corruption_detected); + ip += ofhSize; + } + { size_t const mlhSize = ZSTD_buildSeqTable(dctx->entropy.MLTable, &dctx->MLTptr, + MLtype, MaxML, MLFSELog, + ip, iend-ip, ML_defaultDTable, dctx->fseEntropy); + if (ZSTD_isError(mlhSize)) return ERROR(corruption_detected); + ip += mlhSize; + } + } + + return ip-istart; +} + + +typedef struct { + size_t litLength; + size_t matchLength; + size_t offset; + const BYTE* match; +} seq_t; + +typedef struct { + BIT_DStream_t DStream; + FSE_DState_t stateLL; + FSE_DState_t stateOffb; + FSE_DState_t stateML; + size_t prevOffset[ZSTD_REP_NUM]; + const BYTE* base; + size_t pos; + uPtrDiff gotoDict; +} seqState_t; + + +FORCE_NOINLINE +size_t ZSTD_execSequenceLast7(BYTE* op, + BYTE* const oend, seq_t sequence, + const BYTE** litPtr, const BYTE* const litLimit, + const BYTE* const base, const BYTE* const vBase, const BYTE* const dictEnd) +{ + BYTE* const oLitEnd = op + sequence.litLength; + size_t const sequenceLength = sequence.litLength + sequence.matchLength; + BYTE* const oMatchEnd = op + sequenceLength; /* risk : address space overflow (32-bits) */ + BYTE* const oend_w = oend - WILDCOPY_OVERLENGTH; + const BYTE* const iLitEnd = *litPtr + sequence.litLength; + const BYTE* match = oLitEnd - sequence.offset; + + /* check */ + if (oMatchEnd>oend) return ERROR(dstSize_tooSmall); /* last match must start at a minimum distance of WILDCOPY_OVERLENGTH from oend */ + if (iLitEnd > litLimit) return ERROR(corruption_detected); /* over-read beyond lit buffer */ + if (oLitEnd <= oend_w) return ERROR(GENERIC); /* Precondition */ + + /* copy literals */ + if (op < oend_w) { + ZSTD_wildcopy(op, *litPtr, oend_w - op); + *litPtr += oend_w - op; + op = oend_w; + } + while (op < oLitEnd) *op++ = *(*litPtr)++; + + /* copy Match */ + if (sequence.offset > (size_t)(oLitEnd - base)) { + /* offset beyond prefix */ + if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); + match = dictEnd - (base-match); + if (match + sequence.matchLength <= dictEnd) { + memmove(oLitEnd, match, sequence.matchLength); + return sequenceLength; + } + /* span extDict & currentPrefixSegment */ + { size_t const length1 = dictEnd - match; + memmove(oLitEnd, match, length1); + op = oLitEnd + length1; + sequence.matchLength -= length1; + match = base; + } } + while (op < oMatchEnd) *op++ = *match++; + return sequenceLength; +} + + + + +static seq_t ZSTD_decodeSequence(seqState_t* seqState) +{ + seq_t seq; + + U32 const llCode = FSE_peekSymbol(&seqState->stateLL); + U32 const mlCode = FSE_peekSymbol(&seqState->stateML); + U32 const ofCode = FSE_peekSymbol(&seqState->stateOffb); /* <= maxOff, by table construction */ + + U32 const llBits = LL_bits[llCode]; + U32 const mlBits = ML_bits[mlCode]; + U32 const ofBits = ofCode; + U32 const totalBits = llBits+mlBits+ofBits; + + static const U32 LL_base[MaxLL+1] = { + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, + 16, 18, 20, 22, 24, 28, 32, 40, 48, 64, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000, + 0x2000, 0x4000, 0x8000, 0x10000 }; + + static const U32 ML_base[MaxML+1] = { + 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, 37, 39, 41, 43, 47, 51, 59, 67, 83, 99, 0x83, 0x103, 0x203, 0x403, 0x803, + 0x1003, 0x2003, 0x4003, 0x8003, 0x10003 }; + + static const U32 OF_base[MaxOff+1] = { + 0, 1, 1, 5, 0xD, 0x1D, 0x3D, 0x7D, + 0xFD, 0x1FD, 0x3FD, 0x7FD, 0xFFD, 0x1FFD, 0x3FFD, 0x7FFD, + 0xFFFD, 0x1FFFD, 0x3FFFD, 0x7FFFD, 0xFFFFD, 0x1FFFFD, 0x3FFFFD, 0x7FFFFD, + 0xFFFFFD, 0x1FFFFFD, 0x3FFFFFD, 0x7FFFFFD, 0xFFFFFFD }; + + /* sequence */ + { size_t offset; + if (!ofCode) + offset = 0; + else { + offset = OF_base[ofCode] + BIT_readBitsFast(&seqState->DStream, ofBits); /* <= (ZSTD_WINDOWLOG_MAX-1) bits */ + if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); + } + + if (ofCode <= 1) { + offset += (llCode==0); + if (offset) { + size_t temp = (offset==3) ? seqState->prevOffset[0] - 1 : seqState->prevOffset[offset]; + temp += !temp; /* 0 is not valid; input is corrupted; force offset to 1 */ + if (offset != 1) seqState->prevOffset[2] = seqState->prevOffset[1]; + seqState->prevOffset[1] = seqState->prevOffset[0]; + seqState->prevOffset[0] = offset = temp; + } else { + offset = seqState->prevOffset[0]; + } + } else { + seqState->prevOffset[2] = seqState->prevOffset[1]; + seqState->prevOffset[1] = seqState->prevOffset[0]; + seqState->prevOffset[0] = offset; + } + seq.offset = offset; + } + + seq.matchLength = ML_base[mlCode] + ((mlCode>31) ? BIT_readBitsFast(&seqState->DStream, mlBits) : 0); /* <= 16 bits */ + if (MEM_32bits() && (mlBits+llBits>24)) BIT_reloadDStream(&seqState->DStream); + + seq.litLength = LL_base[llCode] + ((llCode>15) ? BIT_readBitsFast(&seqState->DStream, llBits) : 0); /* <= 16 bits */ + if (MEM_32bits() || + (totalBits > 64 - 7 - (LLFSELog+MLFSELog+OffFSELog)) ) BIT_reloadDStream(&seqState->DStream); + + /* ANS state update */ + FSE_updateState(&seqState->stateLL, &seqState->DStream); /* <= 9 bits */ + FSE_updateState(&seqState->stateML, &seqState->DStream); /* <= 9 bits */ + if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); /* <= 18 bits */ + FSE_updateState(&seqState->stateOffb, &seqState->DStream); /* <= 8 bits */ + + return seq; +} + + +FORCE_INLINE +size_t ZSTD_execSequence(BYTE* op, + BYTE* const oend, seq_t sequence, + const BYTE** litPtr, const BYTE* const litLimit, + const BYTE* const base, const BYTE* const vBase, const BYTE* const dictEnd) +{ + BYTE* const oLitEnd = op + sequence.litLength; + size_t const sequenceLength = sequence.litLength + sequence.matchLength; + BYTE* const oMatchEnd = op + sequenceLength; /* risk : address space overflow (32-bits) */ + BYTE* const oend_w = oend - WILDCOPY_OVERLENGTH; + const BYTE* const iLitEnd = *litPtr + sequence.litLength; + const BYTE* match = oLitEnd - sequence.offset; + + /* check */ + if (oMatchEnd>oend) return ERROR(dstSize_tooSmall); /* last match must start at a minimum distance of WILDCOPY_OVERLENGTH from oend */ + if (iLitEnd > litLimit) return ERROR(corruption_detected); /* over-read beyond lit buffer */ + if (oLitEnd>oend_w) return ZSTD_execSequenceLast7(op, oend, sequence, litPtr, litLimit, base, vBase, dictEnd); + + /* copy Literals */ + ZSTD_copy8(op, *litPtr); + if (sequence.litLength > 8) + ZSTD_wildcopy(op+8, (*litPtr)+8, sequence.litLength - 8); /* note : since oLitEnd <= oend-WILDCOPY_OVERLENGTH, no risk of overwrite beyond oend */ + op = oLitEnd; + *litPtr = iLitEnd; /* update for next sequence */ + + /* copy Match */ + if (sequence.offset > (size_t)(oLitEnd - base)) { + /* offset beyond prefix */ + if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); + match = dictEnd + (match - base); + if (match + sequence.matchLength <= dictEnd) { + memmove(oLitEnd, match, sequence.matchLength); + return sequenceLength; + } + /* span extDict & currentPrefixSegment */ + { size_t const length1 = dictEnd - match; + memmove(oLitEnd, match, length1); + op = oLitEnd + length1; + sequence.matchLength -= length1; + match = base; + if (op > oend_w || sequence.matchLength < MINMATCH) { + U32 i; + for (i = 0; i < sequence.matchLength; ++i) op[i] = match[i]; + return sequenceLength; + } + } } + /* Requirement: op <= oend_w && sequence.matchLength >= MINMATCH */ + + /* match within prefix */ + if (sequence.offset < 8) { + /* close range match, overlap */ + static const U32 dec32table[] = { 0, 1, 2, 1, 4, 4, 4, 4 }; /* added */ + static const int dec64table[] = { 8, 8, 8, 7, 8, 9,10,11 }; /* subtracted */ + int const sub2 = dec64table[sequence.offset]; + op[0] = match[0]; + op[1] = match[1]; + op[2] = match[2]; + op[3] = match[3]; + match += dec32table[sequence.offset]; + ZSTD_copy4(op+4, match); + match -= sub2; + } else { + ZSTD_copy8(op, match); + } + op += 8; match += 8; + + if (oMatchEnd > oend-(16-MINMATCH)) { + if (op < oend_w) { + ZSTD_wildcopy(op, match, oend_w - op); + match += oend_w - op; + op = oend_w; + } + while (op < oMatchEnd) *op++ = *match++; + } else { + ZSTD_wildcopy(op, match, (ptrdiff_t)sequence.matchLength-8); /* works even if matchLength < 8 */ + } + return sequenceLength; +} + + +static size_t ZSTD_decompressSequences( + ZSTD_DCtx* dctx, + void* dst, size_t maxDstSize, + const void* seqStart, size_t seqSize) +{ + const BYTE* ip = (const BYTE*)seqStart; + const BYTE* const iend = ip + seqSize; + BYTE* const ostart = (BYTE* const)dst; + BYTE* const oend = ostart + maxDstSize; + BYTE* op = ostart; + const BYTE* litPtr = dctx->litPtr; + const BYTE* const litEnd = litPtr + dctx->litSize; + const BYTE* const base = (const BYTE*) (dctx->base); + const BYTE* const vBase = (const BYTE*) (dctx->vBase); + const BYTE* const dictEnd = (const BYTE*) (dctx->dictEnd); + int nbSeq; + + /* Build Decoding Tables */ + { size_t const seqHSize = ZSTD_decodeSeqHeaders(dctx, &nbSeq, ip, seqSize); + if (ZSTD_isError(seqHSize)) return seqHSize; + ip += seqHSize; + } + + /* Regen sequences */ + if (nbSeq) { + seqState_t seqState; + dctx->fseEntropy = 1; + { U32 i; for (i=0; ientropy.rep[i]; } + CHECK_E(BIT_initDStream(&seqState.DStream, ip, iend-ip), corruption_detected); + FSE_initDState(&seqState.stateLL, &seqState.DStream, dctx->LLTptr); + FSE_initDState(&seqState.stateOffb, &seqState.DStream, dctx->OFTptr); + FSE_initDState(&seqState.stateML, &seqState.DStream, dctx->MLTptr); + + for ( ; (BIT_reloadDStream(&(seqState.DStream)) <= BIT_DStream_completed) && nbSeq ; ) { + nbSeq--; + { seq_t const sequence = ZSTD_decodeSequence(&seqState); + size_t const oneSeqSize = ZSTD_execSequence(op, oend, sequence, &litPtr, litEnd, base, vBase, dictEnd); + if (ZSTD_isError(oneSeqSize)) return oneSeqSize; + op += oneSeqSize; + } } + + /* check if reached exact end */ + if (nbSeq) return ERROR(corruption_detected); + /* save reps for next block */ + { U32 i; for (i=0; ientropy.rep[i] = (U32)(seqState.prevOffset[i]); } + } + + /* last literal segment */ + { size_t const lastLLSize = litEnd - litPtr; + if (lastLLSize > (size_t)(oend-op)) return ERROR(dstSize_tooSmall); + memcpy(op, litPtr, lastLLSize); + op += lastLLSize; + } + + return op-ostart; +} + + +FORCE_INLINE seq_t ZSTD_decodeSequenceLong_generic(seqState_t* seqState, int const longOffsets) +{ + seq_t seq; + + U32 const llCode = FSE_peekSymbol(&seqState->stateLL); + U32 const mlCode = FSE_peekSymbol(&seqState->stateML); + U32 const ofCode = FSE_peekSymbol(&seqState->stateOffb); /* <= maxOff, by table construction */ + + U32 const llBits = LL_bits[llCode]; + U32 const mlBits = ML_bits[mlCode]; + U32 const ofBits = ofCode; + U32 const totalBits = llBits+mlBits+ofBits; + + static const U32 LL_base[MaxLL+1] = { + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, + 16, 18, 20, 22, 24, 28, 32, 40, 48, 64, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000, + 0x2000, 0x4000, 0x8000, 0x10000 }; + + static const U32 ML_base[MaxML+1] = { + 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, 37, 39, 41, 43, 47, 51, 59, 67, 83, 99, 0x83, 0x103, 0x203, 0x403, 0x803, + 0x1003, 0x2003, 0x4003, 0x8003, 0x10003 }; + + static const U32 OF_base[MaxOff+1] = { + 0, 1, 1, 5, 0xD, 0x1D, 0x3D, 0x7D, + 0xFD, 0x1FD, 0x3FD, 0x7FD, 0xFFD, 0x1FFD, 0x3FFD, 0x7FFD, + 0xFFFD, 0x1FFFD, 0x3FFFD, 0x7FFFD, 0xFFFFD, 0x1FFFFD, 0x3FFFFD, 0x7FFFFD, + 0xFFFFFD, 0x1FFFFFD, 0x3FFFFFD, 0x7FFFFFD, 0xFFFFFFD }; + + /* sequence */ + { size_t offset; + if (!ofCode) + offset = 0; + else { + if (longOffsets) { + int const extraBits = ofBits - MIN(ofBits, STREAM_ACCUMULATOR_MIN); + offset = OF_base[ofCode] + (BIT_readBitsFast(&seqState->DStream, ofBits - extraBits) << extraBits); + if (MEM_32bits() || extraBits) BIT_reloadDStream(&seqState->DStream); + if (extraBits) offset += BIT_readBitsFast(&seqState->DStream, extraBits); + } else { + offset = OF_base[ofCode] + BIT_readBitsFast(&seqState->DStream, ofBits); /* <= (ZSTD_WINDOWLOG_MAX-1) bits */ + if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); + } + } + + if (ofCode <= 1) { + offset += (llCode==0); + if (offset) { + size_t temp = (offset==3) ? seqState->prevOffset[0] - 1 : seqState->prevOffset[offset]; + temp += !temp; /* 0 is not valid; input is corrupted; force offset to 1 */ + if (offset != 1) seqState->prevOffset[2] = seqState->prevOffset[1]; + seqState->prevOffset[1] = seqState->prevOffset[0]; + seqState->prevOffset[0] = offset = temp; + } else { + offset = seqState->prevOffset[0]; + } + } else { + seqState->prevOffset[2] = seqState->prevOffset[1]; + seqState->prevOffset[1] = seqState->prevOffset[0]; + seqState->prevOffset[0] = offset; + } + seq.offset = offset; + } + + seq.matchLength = ML_base[mlCode] + ((mlCode>31) ? BIT_readBitsFast(&seqState->DStream, mlBits) : 0); /* <= 16 bits */ + if (MEM_32bits() && (mlBits+llBits>24)) BIT_reloadDStream(&seqState->DStream); + + seq.litLength = LL_base[llCode] + ((llCode>15) ? BIT_readBitsFast(&seqState->DStream, llBits) : 0); /* <= 16 bits */ + if (MEM_32bits() || + (totalBits > 64 - 7 - (LLFSELog+MLFSELog+OffFSELog)) ) BIT_reloadDStream(&seqState->DStream); + + { size_t const pos = seqState->pos + seq.litLength; + seq.match = seqState->base + pos - seq.offset; /* single memory segment */ + if (seq.offset > pos) seq.match += seqState->gotoDict; /* separate memory segment */ + seqState->pos = pos + seq.matchLength; + } + + /* ANS state update */ + FSE_updateState(&seqState->stateLL, &seqState->DStream); /* <= 9 bits */ + FSE_updateState(&seqState->stateML, &seqState->DStream); /* <= 9 bits */ + if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); /* <= 18 bits */ + FSE_updateState(&seqState->stateOffb, &seqState->DStream); /* <= 8 bits */ + + return seq; +} + +static seq_t ZSTD_decodeSequenceLong(seqState_t* seqState, unsigned const windowSize) { + if (ZSTD_highbit32(windowSize) > STREAM_ACCUMULATOR_MIN) { + return ZSTD_decodeSequenceLong_generic(seqState, 1); + } else { + return ZSTD_decodeSequenceLong_generic(seqState, 0); + } +} + +FORCE_INLINE +size_t ZSTD_execSequenceLong(BYTE* op, + BYTE* const oend, seq_t sequence, + const BYTE** litPtr, const BYTE* const litLimit, + const BYTE* const base, const BYTE* const vBase, const BYTE* const dictEnd) +{ + BYTE* const oLitEnd = op + sequence.litLength; + size_t const sequenceLength = sequence.litLength + sequence.matchLength; + BYTE* const oMatchEnd = op + sequenceLength; /* risk : address space overflow (32-bits) */ + BYTE* const oend_w = oend - WILDCOPY_OVERLENGTH; + const BYTE* const iLitEnd = *litPtr + sequence.litLength; + const BYTE* match = sequence.match; + + /* check */ +#if 1 + if (oMatchEnd>oend) return ERROR(dstSize_tooSmall); /* last match must start at a minimum distance of WILDCOPY_OVERLENGTH from oend */ + if (iLitEnd > litLimit) return ERROR(corruption_detected); /* over-read beyond lit buffer */ + if (oLitEnd>oend_w) return ZSTD_execSequenceLast7(op, oend, sequence, litPtr, litLimit, base, vBase, dictEnd); +#endif + + /* copy Literals */ + ZSTD_copy8(op, *litPtr); + if (sequence.litLength > 8) + ZSTD_wildcopy(op+8, (*litPtr)+8, sequence.litLength - 8); /* note : since oLitEnd <= oend-WILDCOPY_OVERLENGTH, no risk of overwrite beyond oend */ + op = oLitEnd; + *litPtr = iLitEnd; /* update for next sequence */ + + /* copy Match */ +#if 1 + if (sequence.offset > (size_t)(oLitEnd - base)) { + /* offset beyond prefix */ + if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); + if (match + sequence.matchLength <= dictEnd) { + memmove(oLitEnd, match, sequence.matchLength); + return sequenceLength; + } + /* span extDict & currentPrefixSegment */ + { size_t const length1 = dictEnd - match; + memmove(oLitEnd, match, length1); + op = oLitEnd + length1; + sequence.matchLength -= length1; + match = base; + if (op > oend_w || sequence.matchLength < MINMATCH) { + U32 i; + for (i = 0; i < sequence.matchLength; ++i) op[i] = match[i]; + return sequenceLength; + } + } } + /* Requirement: op <= oend_w && sequence.matchLength >= MINMATCH */ +#endif + + /* match within prefix */ + if (sequence.offset < 8) { + /* close range match, overlap */ + static const U32 dec32table[] = { 0, 1, 2, 1, 4, 4, 4, 4 }; /* added */ + static const int dec64table[] = { 8, 8, 8, 7, 8, 9,10,11 }; /* subtracted */ + int const sub2 = dec64table[sequence.offset]; + op[0] = match[0]; + op[1] = match[1]; + op[2] = match[2]; + op[3] = match[3]; + match += dec32table[sequence.offset]; + ZSTD_copy4(op+4, match); + match -= sub2; + } else { + ZSTD_copy8(op, match); + } + op += 8; match += 8; + + if (oMatchEnd > oend-(16-MINMATCH)) { + if (op < oend_w) { + ZSTD_wildcopy(op, match, oend_w - op); + match += oend_w - op; + op = oend_w; + } + while (op < oMatchEnd) *op++ = *match++; + } else { + ZSTD_wildcopy(op, match, (ptrdiff_t)sequence.matchLength-8); /* works even if matchLength < 8 */ + } + return sequenceLength; +} + +static size_t ZSTD_decompressSequencesLong( + ZSTD_DCtx* dctx, + void* dst, size_t maxDstSize, + const void* seqStart, size_t seqSize) +{ + const BYTE* ip = (const BYTE*)seqStart; + const BYTE* const iend = ip + seqSize; + BYTE* const ostart = (BYTE* const)dst; + BYTE* const oend = ostart + maxDstSize; + BYTE* op = ostart; + const BYTE* litPtr = dctx->litPtr; + const BYTE* const litEnd = litPtr + dctx->litSize; + const BYTE* const base = (const BYTE*) (dctx->base); + const BYTE* const vBase = (const BYTE*) (dctx->vBase); + const BYTE* const dictEnd = (const BYTE*) (dctx->dictEnd); + unsigned const windowSize = dctx->fParams.windowSize; + int nbSeq; + + /* Build Decoding Tables */ + { size_t const seqHSize = ZSTD_decodeSeqHeaders(dctx, &nbSeq, ip, seqSize); + if (ZSTD_isError(seqHSize)) return seqHSize; + ip += seqHSize; + } + + /* Regen sequences */ + if (nbSeq) { +#define STORED_SEQS 4 +#define STOSEQ_MASK (STORED_SEQS-1) +#define ADVANCED_SEQS 4 + seq_t sequences[STORED_SEQS]; + int const seqAdvance = MIN(nbSeq, ADVANCED_SEQS); + seqState_t seqState; + int seqNb; + dctx->fseEntropy = 1; + { U32 i; for (i=0; ientropy.rep[i]; } + seqState.base = base; + seqState.pos = (size_t)(op-base); + seqState.gotoDict = (uPtrDiff)dictEnd - (uPtrDiff)base; /* cast to avoid undefined behaviour */ + CHECK_E(BIT_initDStream(&seqState.DStream, ip, iend-ip), corruption_detected); + FSE_initDState(&seqState.stateLL, &seqState.DStream, dctx->LLTptr); + FSE_initDState(&seqState.stateOffb, &seqState.DStream, dctx->OFTptr); + FSE_initDState(&seqState.stateML, &seqState.DStream, dctx->MLTptr); + + /* prepare in advance */ + for (seqNb=0; (BIT_reloadDStream(&seqState.DStream) <= BIT_DStream_completed) && seqNbentropy.rep[i] = (U32)(seqState.prevOffset[i]); } + } + + /* last literal segment */ + { size_t const lastLLSize = litEnd - litPtr; + if (lastLLSize > (size_t)(oend-op)) return ERROR(dstSize_tooSmall); + memcpy(op, litPtr, lastLLSize); + op += lastLLSize; + } + + return op-ostart; +} + + +static size_t ZSTD_decompressBlock_internal(ZSTD_DCtx* dctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize) +{ /* blockType == blockCompressed */ + const BYTE* ip = (const BYTE*)src; + + if (srcSize >= ZSTD_BLOCKSIZE_ABSOLUTEMAX) return ERROR(srcSize_wrong); + + /* Decode literals section */ + { size_t const litCSize = ZSTD_decodeLiteralsBlock(dctx, src, srcSize); + if (ZSTD_isError(litCSize)) return litCSize; + ip += litCSize; + srcSize -= litCSize; + } + if (sizeof(size_t) > 4) /* do not enable prefetching on 32-bits x86, as it's performance detrimental */ + /* likely because of register pressure */ + /* if that's the correct cause, then 32-bits ARM should be affected differently */ + /* it would be good to test this on ARM real hardware, to see if prefetch version improves speed */ + if (dctx->fParams.windowSize > (1<<23)) + return ZSTD_decompressSequencesLong(dctx, dst, dstCapacity, ip, srcSize); + return ZSTD_decompressSequences(dctx, dst, dstCapacity, ip, srcSize); +} + + +static void ZSTD_checkContinuity(ZSTD_DCtx* dctx, const void* dst) +{ + if (dst != dctx->previousDstEnd) { /* not contiguous */ + dctx->dictEnd = dctx->previousDstEnd; + dctx->vBase = (const char*)dst - ((const char*)(dctx->previousDstEnd) - (const char*)(dctx->base)); + dctx->base = dst; + dctx->previousDstEnd = dst; + } +} + +size_t ZSTD_decompressBlock(ZSTD_DCtx* dctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize) +{ + size_t dSize; + ZSTD_checkContinuity(dctx, dst); + dSize = ZSTD_decompressBlock_internal(dctx, dst, dstCapacity, src, srcSize); + dctx->previousDstEnd = (char*)dst + dSize; + return dSize; +} + + +/** ZSTD_insertBlock() : + insert `src` block into `dctx` history. Useful to track uncompressed blocks. */ +ZSTDLIB_API size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize) +{ + ZSTD_checkContinuity(dctx, blockStart); + dctx->previousDstEnd = (const char*)blockStart + blockSize; + return blockSize; +} + + +size_t ZSTD_generateNxBytes(void* dst, size_t dstCapacity, BYTE byte, size_t length) +{ + if (length > dstCapacity) return ERROR(dstSize_tooSmall); + memset(dst, byte, length); + return length; +} + +/** ZSTD_findFrameCompressedSize() : + * compatible with legacy mode + * `src` must point to the start of a ZSTD frame, ZSTD legacy frame, or skippable frame + * `srcSize` must be at least as large as the frame contained + * @return : the compressed size of the frame starting at `src` */ +size_t ZSTD_findFrameCompressedSize(const void *src, size_t srcSize) +{ +#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) + if (ZSTD_isLegacy(src, srcSize)) return ZSTD_findFrameCompressedSizeLegacy(src, srcSize); +#endif + if (srcSize >= ZSTD_skippableHeaderSize && + (MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { + return ZSTD_skippableHeaderSize + MEM_readLE32((const BYTE*)src + 4); + } else { + const BYTE* ip = (const BYTE*)src; + const BYTE* const ipstart = ip; + size_t remainingSize = srcSize; + ZSTD_frameParams fParams; + + size_t const headerSize = ZSTD_frameHeaderSize(ip, remainingSize); + if (ZSTD_isError(headerSize)) return headerSize; + + /* Frame Header */ + { size_t const ret = ZSTD_getFrameParams(&fParams, ip, remainingSize); + if (ZSTD_isError(ret)) return ret; + if (ret > 0) return ERROR(srcSize_wrong); + } + + ip += headerSize; + remainingSize -= headerSize; + + /* Loop on each block */ + while (1) { + blockProperties_t blockProperties; + size_t const cBlockSize = ZSTD_getcBlockSize(ip, remainingSize, &blockProperties); + if (ZSTD_isError(cBlockSize)) return cBlockSize; + + if (ZSTD_blockHeaderSize + cBlockSize > remainingSize) return ERROR(srcSize_wrong); + + ip += ZSTD_blockHeaderSize + cBlockSize; + remainingSize -= ZSTD_blockHeaderSize + cBlockSize; + + if (blockProperties.lastBlock) break; + } + + if (fParams.checksumFlag) { /* Frame content checksum */ + if (remainingSize < 4) return ERROR(srcSize_wrong); + ip += 4; + remainingSize -= 4; + } + + return ip - ipstart; + } +} + +/*! ZSTD_decompressFrame() : +* @dctx must be properly initialized */ +static size_t ZSTD_decompressFrame(ZSTD_DCtx* dctx, + void* dst, size_t dstCapacity, + const void** srcPtr, size_t *srcSizePtr) +{ + const BYTE* ip = (const BYTE*)(*srcPtr); + BYTE* const ostart = (BYTE* const)dst; + BYTE* const oend = ostart + dstCapacity; + BYTE* op = ostart; + size_t remainingSize = *srcSizePtr; + + /* check */ + if (remainingSize < ZSTD_frameHeaderSize_min+ZSTD_blockHeaderSize) return ERROR(srcSize_wrong); + + /* Frame Header */ + { size_t const frameHeaderSize = ZSTD_frameHeaderSize(ip, ZSTD_frameHeaderSize_prefix); + if (ZSTD_isError(frameHeaderSize)) return frameHeaderSize; + if (remainingSize < frameHeaderSize+ZSTD_blockHeaderSize) return ERROR(srcSize_wrong); + CHECK_F(ZSTD_decodeFrameHeader(dctx, ip, frameHeaderSize)); + ip += frameHeaderSize; remainingSize -= frameHeaderSize; + } + + /* Loop on each block */ + while (1) { + size_t decodedSize; + blockProperties_t blockProperties; + size_t const cBlockSize = ZSTD_getcBlockSize(ip, remainingSize, &blockProperties); + if (ZSTD_isError(cBlockSize)) return cBlockSize; + + ip += ZSTD_blockHeaderSize; + remainingSize -= ZSTD_blockHeaderSize; + if (cBlockSize > remainingSize) return ERROR(srcSize_wrong); + + switch(blockProperties.blockType) + { + case bt_compressed: + decodedSize = ZSTD_decompressBlock_internal(dctx, op, oend-op, ip, cBlockSize); + break; + case bt_raw : + decodedSize = ZSTD_copyRawBlock(op, oend-op, ip, cBlockSize); + break; + case bt_rle : + decodedSize = ZSTD_generateNxBytes(op, oend-op, *ip, blockProperties.origSize); + break; + case bt_reserved : + default: + return ERROR(corruption_detected); + } + + if (ZSTD_isError(decodedSize)) return decodedSize; + if (dctx->fParams.checksumFlag) XXH64_update(&dctx->xxhState, op, decodedSize); + op += decodedSize; + ip += cBlockSize; + remainingSize -= cBlockSize; + if (blockProperties.lastBlock) break; + } + + if (dctx->fParams.checksumFlag) { /* Frame content checksum verification */ + U32 const checkCalc = (U32)XXH64_digest(&dctx->xxhState); + U32 checkRead; + if (remainingSize<4) return ERROR(checksum_wrong); + checkRead = MEM_readLE32(ip); + if (checkRead != checkCalc) return ERROR(checksum_wrong); + ip += 4; + remainingSize -= 4; + } + + /* Allow caller to get size read */ + *srcPtr = ip; + *srcSizePtr = remainingSize; + return op-ostart; +} + +static const void* ZSTD_DDictDictContent(const ZSTD_DDict* ddict); +static size_t ZSTD_DDictDictSize(const ZSTD_DDict* ddict); + +static size_t ZSTD_decompressMultiFrame(ZSTD_DCtx* dctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void *dict, size_t dictSize, + const ZSTD_DDict* ddict) +{ + void* const dststart = dst; + + if (ddict) { + if (dict) { + /* programmer error, these two cases should be mutually exclusive */ + return ERROR(GENERIC); + } + + dict = ZSTD_DDictDictContent(ddict); + dictSize = ZSTD_DDictDictSize(ddict); + } + + while (srcSize >= ZSTD_frameHeaderSize_prefix) { + U32 magicNumber; + +#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) + if (ZSTD_isLegacy(src, srcSize)) { + size_t decodedSize; + size_t const frameSize = ZSTD_findFrameCompressedSizeLegacy(src, srcSize); + if (ZSTD_isError(frameSize)) return frameSize; + + decodedSize = ZSTD_decompressLegacy(dst, dstCapacity, src, frameSize, dict, dictSize); + + dst = (BYTE*)dst + decodedSize; + dstCapacity -= decodedSize; + + src = (const BYTE*)src + frameSize; + srcSize -= frameSize; + + continue; + } +#endif + + magicNumber = MEM_readLE32(src); + if (magicNumber != ZSTD_MAGICNUMBER) { + if ((magicNumber & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { + size_t skippableSize; + if (srcSize < ZSTD_skippableHeaderSize) + return ERROR(srcSize_wrong); + skippableSize = MEM_readLE32((const BYTE *)src + 4) + + ZSTD_skippableHeaderSize; + if (srcSize < skippableSize) { + return ERROR(srcSize_wrong); + } + + src = (const BYTE *)src + skippableSize; + srcSize -= skippableSize; + continue; + } else { + return ERROR(prefix_unknown); + } + } + + if (ddict) { + /* we were called from ZSTD_decompress_usingDDict */ + ZSTD_refDDict(dctx, ddict); + } else { + /* this will initialize correctly with no dict if dict == NULL, so + * use this in all cases but ddict */ + CHECK_F(ZSTD_decompressBegin_usingDict(dctx, dict, dictSize)); + } + ZSTD_checkContinuity(dctx, dst); + + { const size_t res = ZSTD_decompressFrame(dctx, dst, dstCapacity, + &src, &srcSize); + if (ZSTD_isError(res)) return res; + /* don't need to bounds check this, ZSTD_decompressFrame will have + * already */ + dst = (BYTE*)dst + res; + dstCapacity -= res; + } + } + + if (srcSize) return ERROR(srcSize_wrong); /* input not entirely consumed */ + + return (BYTE*)dst - (BYTE*)dststart; +} + +size_t ZSTD_decompress_usingDict(ZSTD_DCtx* dctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict, size_t dictSize) +{ + return ZSTD_decompressMultiFrame(dctx, dst, dstCapacity, src, srcSize, dict, dictSize, NULL); +} + + +size_t ZSTD_decompressDCtx(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + return ZSTD_decompress_usingDict(dctx, dst, dstCapacity, src, srcSize, NULL, 0); +} + + +size_t ZSTD_decompress(void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ +#if defined(ZSTD_HEAPMODE) && (ZSTD_HEAPMODE==1) + size_t regenSize; + ZSTD_DCtx* const dctx = ZSTD_createDCtx(); + if (dctx==NULL) return ERROR(memory_allocation); + regenSize = ZSTD_decompressDCtx(dctx, dst, dstCapacity, src, srcSize); + ZSTD_freeDCtx(dctx); + return regenSize; +#else /* stack mode */ + ZSTD_DCtx dctx; + return ZSTD_decompressDCtx(&dctx, dst, dstCapacity, src, srcSize); +#endif +} + + +/*-************************************** +* Advanced Streaming Decompression API +* Bufferless and synchronous +****************************************/ +size_t ZSTD_nextSrcSizeToDecompress(ZSTD_DCtx* dctx) { return dctx->expected; } + +ZSTD_nextInputType_e ZSTD_nextInputType(ZSTD_DCtx* dctx) { + switch(dctx->stage) + { + default: /* should not happen */ + case ZSTDds_getFrameHeaderSize: + case ZSTDds_decodeFrameHeader: + return ZSTDnit_frameHeader; + case ZSTDds_decodeBlockHeader: + return ZSTDnit_blockHeader; + case ZSTDds_decompressBlock: + return ZSTDnit_block; + case ZSTDds_decompressLastBlock: + return ZSTDnit_lastBlock; + case ZSTDds_checkChecksum: + return ZSTDnit_checksum; + case ZSTDds_decodeSkippableHeader: + case ZSTDds_skipFrame: + return ZSTDnit_skippableFrame; + } +} + +int ZSTD_isSkipFrame(ZSTD_DCtx* dctx) { return dctx->stage == ZSTDds_skipFrame; } /* for zbuff */ + +/** ZSTD_decompressContinue() : +* @return : nb of bytes generated into `dst` (necessarily <= `dstCapacity) +* or an error code, which can be tested using ZSTD_isError() */ +size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + /* Sanity check */ + if (srcSize != dctx->expected) return ERROR(srcSize_wrong); + if (dstCapacity) ZSTD_checkContinuity(dctx, dst); + + switch (dctx->stage) + { + case ZSTDds_getFrameHeaderSize : + if (srcSize != ZSTD_frameHeaderSize_prefix) return ERROR(srcSize_wrong); /* impossible */ + if ((MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { /* skippable frame */ + memcpy(dctx->headerBuffer, src, ZSTD_frameHeaderSize_prefix); + dctx->expected = ZSTD_skippableHeaderSize - ZSTD_frameHeaderSize_prefix; /* magic number + skippable frame length */ + dctx->stage = ZSTDds_decodeSkippableHeader; + return 0; + } + dctx->headerSize = ZSTD_frameHeaderSize(src, ZSTD_frameHeaderSize_prefix); + if (ZSTD_isError(dctx->headerSize)) return dctx->headerSize; + memcpy(dctx->headerBuffer, src, ZSTD_frameHeaderSize_prefix); + if (dctx->headerSize > ZSTD_frameHeaderSize_prefix) { + dctx->expected = dctx->headerSize - ZSTD_frameHeaderSize_prefix; + dctx->stage = ZSTDds_decodeFrameHeader; + return 0; + } + dctx->expected = 0; /* not necessary to copy more */ + + case ZSTDds_decodeFrameHeader: + memcpy(dctx->headerBuffer + ZSTD_frameHeaderSize_prefix, src, dctx->expected); + CHECK_F(ZSTD_decodeFrameHeader(dctx, dctx->headerBuffer, dctx->headerSize)); + dctx->expected = ZSTD_blockHeaderSize; + dctx->stage = ZSTDds_decodeBlockHeader; + return 0; + + case ZSTDds_decodeBlockHeader: + { blockProperties_t bp; + size_t const cBlockSize = ZSTD_getcBlockSize(src, ZSTD_blockHeaderSize, &bp); + if (ZSTD_isError(cBlockSize)) return cBlockSize; + dctx->expected = cBlockSize; + dctx->bType = bp.blockType; + dctx->rleSize = bp.origSize; + if (cBlockSize) { + dctx->stage = bp.lastBlock ? ZSTDds_decompressLastBlock : ZSTDds_decompressBlock; + return 0; + } + /* empty block */ + if (bp.lastBlock) { + if (dctx->fParams.checksumFlag) { + dctx->expected = 4; + dctx->stage = ZSTDds_checkChecksum; + } else { + dctx->expected = 0; /* end of frame */ + dctx->stage = ZSTDds_getFrameHeaderSize; + } + } else { + dctx->expected = 3; /* go directly to next header */ + dctx->stage = ZSTDds_decodeBlockHeader; + } + return 0; + } + case ZSTDds_decompressLastBlock: + case ZSTDds_decompressBlock: + { size_t rSize; + switch(dctx->bType) + { + case bt_compressed: + rSize = ZSTD_decompressBlock_internal(dctx, dst, dstCapacity, src, srcSize); + break; + case bt_raw : + rSize = ZSTD_copyRawBlock(dst, dstCapacity, src, srcSize); + break; + case bt_rle : + rSize = ZSTD_setRleBlock(dst, dstCapacity, src, srcSize, dctx->rleSize); + break; + case bt_reserved : /* should never happen */ + default: + return ERROR(corruption_detected); + } + if (ZSTD_isError(rSize)) return rSize; + if (dctx->fParams.checksumFlag) XXH64_update(&dctx->xxhState, dst, rSize); + + if (dctx->stage == ZSTDds_decompressLastBlock) { /* end of frame */ + if (dctx->fParams.checksumFlag) { /* another round for frame checksum */ + dctx->expected = 4; + dctx->stage = ZSTDds_checkChecksum; + } else { + dctx->expected = 0; /* ends here */ + dctx->stage = ZSTDds_getFrameHeaderSize; + } + } else { + dctx->stage = ZSTDds_decodeBlockHeader; + dctx->expected = ZSTD_blockHeaderSize; + dctx->previousDstEnd = (char*)dst + rSize; + } + return rSize; + } + case ZSTDds_checkChecksum: + { U32 const h32 = (U32)XXH64_digest(&dctx->xxhState); + U32 const check32 = MEM_readLE32(src); /* srcSize == 4, guaranteed by dctx->expected */ + if (check32 != h32) return ERROR(checksum_wrong); + dctx->expected = 0; + dctx->stage = ZSTDds_getFrameHeaderSize; + return 0; + } + case ZSTDds_decodeSkippableHeader: + { memcpy(dctx->headerBuffer + ZSTD_frameHeaderSize_prefix, src, dctx->expected); + dctx->expected = MEM_readLE32(dctx->headerBuffer + 4); + dctx->stage = ZSTDds_skipFrame; + return 0; + } + case ZSTDds_skipFrame: + { dctx->expected = 0; + dctx->stage = ZSTDds_getFrameHeaderSize; + return 0; + } + default: + return ERROR(GENERIC); /* impossible */ + } +} + + +static size_t ZSTD_refDictContent(ZSTD_DCtx* dctx, const void* dict, size_t dictSize) +{ + dctx->dictEnd = dctx->previousDstEnd; + dctx->vBase = (const char*)dict - ((const char*)(dctx->previousDstEnd) - (const char*)(dctx->base)); + dctx->base = dict; + dctx->previousDstEnd = (const char*)dict + dictSize; + return 0; +} + +/* ZSTD_loadEntropy() : + * dict : must point at beginning of a valid zstd dictionary + * @return : size of entropy tables read */ +static size_t ZSTD_loadEntropy(ZSTD_entropyTables_t* entropy, const void* const dict, size_t const dictSize) +{ + const BYTE* dictPtr = (const BYTE*)dict; + const BYTE* const dictEnd = dictPtr + dictSize; + + if (dictSize <= 8) return ERROR(dictionary_corrupted); + dictPtr += 8; /* skip header = magic + dictID */ + + + { size_t const hSize = HUF_readDTableX4(entropy->hufTable, dictPtr, dictEnd-dictPtr); + if (HUF_isError(hSize)) return ERROR(dictionary_corrupted); + dictPtr += hSize; + } + + { short offcodeNCount[MaxOff+1]; + U32 offcodeMaxValue = MaxOff, offcodeLog; + size_t const offcodeHeaderSize = FSE_readNCount(offcodeNCount, &offcodeMaxValue, &offcodeLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(offcodeHeaderSize)) return ERROR(dictionary_corrupted); + if (offcodeLog > OffFSELog) return ERROR(dictionary_corrupted); + CHECK_E(FSE_buildDTable(entropy->OFTable, offcodeNCount, offcodeMaxValue, offcodeLog), dictionary_corrupted); + dictPtr += offcodeHeaderSize; + } + + { short matchlengthNCount[MaxML+1]; + unsigned matchlengthMaxValue = MaxML, matchlengthLog; + size_t const matchlengthHeaderSize = FSE_readNCount(matchlengthNCount, &matchlengthMaxValue, &matchlengthLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(matchlengthHeaderSize)) return ERROR(dictionary_corrupted); + if (matchlengthLog > MLFSELog) return ERROR(dictionary_corrupted); + CHECK_E(FSE_buildDTable(entropy->MLTable, matchlengthNCount, matchlengthMaxValue, matchlengthLog), dictionary_corrupted); + dictPtr += matchlengthHeaderSize; + } + + { short litlengthNCount[MaxLL+1]; + unsigned litlengthMaxValue = MaxLL, litlengthLog; + size_t const litlengthHeaderSize = FSE_readNCount(litlengthNCount, &litlengthMaxValue, &litlengthLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(litlengthHeaderSize)) return ERROR(dictionary_corrupted); + if (litlengthLog > LLFSELog) return ERROR(dictionary_corrupted); + CHECK_E(FSE_buildDTable(entropy->LLTable, litlengthNCount, litlengthMaxValue, litlengthLog), dictionary_corrupted); + dictPtr += litlengthHeaderSize; + } + + if (dictPtr+12 > dictEnd) return ERROR(dictionary_corrupted); + { int i; + size_t const dictContentSize = (size_t)(dictEnd - (dictPtr+12)); + for (i=0; i<3; i++) { + U32 const rep = MEM_readLE32(dictPtr); dictPtr += 4; + if (rep==0 || rep >= dictContentSize) return ERROR(dictionary_corrupted); + entropy->rep[i] = rep; + } } + + return dictPtr - (const BYTE*)dict; +} + +static size_t ZSTD_decompress_insertDictionary(ZSTD_DCtx* dctx, const void* dict, size_t dictSize) +{ + if (dictSize < 8) return ZSTD_refDictContent(dctx, dict, dictSize); + { U32 const magic = MEM_readLE32(dict); + if (magic != ZSTD_DICT_MAGIC) { + return ZSTD_refDictContent(dctx, dict, dictSize); /* pure content mode */ + } } + dctx->dictID = MEM_readLE32((const char*)dict + 4); + + /* load entropy tables */ + { size_t const eSize = ZSTD_loadEntropy(&dctx->entropy, dict, dictSize); + if (ZSTD_isError(eSize)) return ERROR(dictionary_corrupted); + dict = (const char*)dict + eSize; + dictSize -= eSize; + } + dctx->litEntropy = dctx->fseEntropy = 1; + + /* reference dictionary content */ + return ZSTD_refDictContent(dctx, dict, dictSize); +} + +size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx* dctx, const void* dict, size_t dictSize) +{ + CHECK_F(ZSTD_decompressBegin(dctx)); + if (dict && dictSize) CHECK_E(ZSTD_decompress_insertDictionary(dctx, dict, dictSize), dictionary_corrupted); + return 0; +} + + +/* ====== ZSTD_DDict ====== */ + +struct ZSTD_DDict_s { + void* dictBuffer; + const void* dictContent; + size_t dictSize; + ZSTD_entropyTables_t entropy; + U32 dictID; + U32 entropyPresent; + ZSTD_customMem cMem; +}; /* typedef'd to ZSTD_DDict within "zstd.h" */ + +static const void* ZSTD_DDictDictContent(const ZSTD_DDict* ddict) +{ + return ddict->dictContent; +} + +static size_t ZSTD_DDictDictSize(const ZSTD_DDict* ddict) +{ + return ddict->dictSize; +} + +static void ZSTD_refDDict(ZSTD_DCtx* dstDCtx, const ZSTD_DDict* ddict) +{ + ZSTD_decompressBegin(dstDCtx); /* init */ + if (ddict) { /* support refDDict on NULL */ + dstDCtx->dictID = ddict->dictID; + dstDCtx->base = ddict->dictContent; + dstDCtx->vBase = ddict->dictContent; + dstDCtx->dictEnd = (const BYTE*)ddict->dictContent + ddict->dictSize; + dstDCtx->previousDstEnd = dstDCtx->dictEnd; + if (ddict->entropyPresent) { + dstDCtx->litEntropy = 1; + dstDCtx->fseEntropy = 1; + dstDCtx->LLTptr = ddict->entropy.LLTable; + dstDCtx->MLTptr = ddict->entropy.MLTable; + dstDCtx->OFTptr = ddict->entropy.OFTable; + dstDCtx->HUFptr = ddict->entropy.hufTable; + dstDCtx->entropy.rep[0] = ddict->entropy.rep[0]; + dstDCtx->entropy.rep[1] = ddict->entropy.rep[1]; + dstDCtx->entropy.rep[2] = ddict->entropy.rep[2]; + } else { + dstDCtx->litEntropy = 0; + dstDCtx->fseEntropy = 0; + } + } +} + +static size_t ZSTD_loadEntropy_inDDict(ZSTD_DDict* ddict) +{ + ddict->dictID = 0; + ddict->entropyPresent = 0; + if (ddict->dictSize < 8) return 0; + { U32 const magic = MEM_readLE32(ddict->dictContent); + if (magic != ZSTD_DICT_MAGIC) return 0; /* pure content mode */ + } + ddict->dictID = MEM_readLE32((const char*)ddict->dictContent + 4); + + /* load entropy tables */ + CHECK_E( ZSTD_loadEntropy(&ddict->entropy, ddict->dictContent, ddict->dictSize), dictionary_corrupted ); + ddict->entropyPresent = 1; + return 0; +} + + +ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize, unsigned byReference, ZSTD_customMem customMem) +{ + if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; + if (!customMem.customAlloc || !customMem.customFree) return NULL; + + { ZSTD_DDict* const ddict = (ZSTD_DDict*) ZSTD_malloc(sizeof(ZSTD_DDict), customMem); + if (!ddict) return NULL; + ddict->cMem = customMem; + + if ((byReference) || (!dict) || (!dictSize)) { + ddict->dictBuffer = NULL; + ddict->dictContent = dict; + } else { + void* const internalBuffer = ZSTD_malloc(dictSize, customMem); + if (!internalBuffer) { ZSTD_freeDDict(ddict); return NULL; } + memcpy(internalBuffer, dict, dictSize); + ddict->dictBuffer = internalBuffer; + ddict->dictContent = internalBuffer; + } + ddict->dictSize = dictSize; + ddict->entropy.hufTable[0] = (HUF_DTable)((HufLog)*0x1000001); /* cover both little and big endian */ + /* parse dictionary content */ + { size_t const errorCode = ZSTD_loadEntropy_inDDict(ddict); + if (ZSTD_isError(errorCode)) { + ZSTD_freeDDict(ddict); + return NULL; + } } + + return ddict; + } +} + +/*! ZSTD_createDDict() : +* Create a digested dictionary, to start decompression without startup delay. +* `dict` content is copied inside DDict. +* Consequently, `dict` can be released after `ZSTD_DDict` creation */ +ZSTD_DDict* ZSTD_createDDict(const void* dict, size_t dictSize) +{ + ZSTD_customMem const allocator = { NULL, NULL, NULL }; + return ZSTD_createDDict_advanced(dict, dictSize, 0, allocator); +} + + +/*! ZSTD_createDDict_byReference() : + * Create a digested dictionary, to start decompression without startup delay. + * Dictionary content is simply referenced, it will be accessed during decompression. + * Warning : dictBuffer must outlive DDict (DDict must be freed before dictBuffer) */ +ZSTD_DDict* ZSTD_createDDict_byReference(const void* dictBuffer, size_t dictSize) +{ + ZSTD_customMem const allocator = { NULL, NULL, NULL }; + return ZSTD_createDDict_advanced(dictBuffer, dictSize, 1, allocator); +} + + +size_t ZSTD_freeDDict(ZSTD_DDict* ddict) +{ + if (ddict==NULL) return 0; /* support free on NULL */ + { ZSTD_customMem const cMem = ddict->cMem; + ZSTD_free(ddict->dictBuffer, cMem); + ZSTD_free(ddict, cMem); + return 0; + } +} + +size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict) +{ + if (ddict==NULL) return 0; /* support sizeof on NULL */ + return sizeof(*ddict) + (ddict->dictBuffer ? ddict->dictSize : 0) ; +} + +/*! ZSTD_getDictID_fromDict() : + * Provides the dictID stored within dictionary. + * if @return == 0, the dictionary is not conformant with Zstandard specification. + * It can still be loaded, but as a content-only dictionary. */ +unsigned ZSTD_getDictID_fromDict(const void* dict, size_t dictSize) +{ + if (dictSize < 8) return 0; + if (MEM_readLE32(dict) != ZSTD_DICT_MAGIC) return 0; + return MEM_readLE32((const char*)dict + 4); +} + +/*! ZSTD_getDictID_fromDDict() : + * Provides the dictID of the dictionary loaded into `ddict`. + * If @return == 0, the dictionary is not conformant to Zstandard specification, or empty. + * Non-conformant dictionaries can still be loaded, but as content-only dictionaries. */ +unsigned ZSTD_getDictID_fromDDict(const ZSTD_DDict* ddict) +{ + if (ddict==NULL) return 0; + return ZSTD_getDictID_fromDict(ddict->dictContent, ddict->dictSize); +} + +/*! ZSTD_getDictID_fromFrame() : + * Provides the dictID required to decompressed the frame stored within `src`. + * If @return == 0, the dictID could not be decoded. + * This could for one of the following reasons : + * - The frame does not require a dictionary to be decoded (most common case). + * - The frame was built with dictID intentionally removed. Whatever dictionary is necessary is a hidden information. + * Note : this use case also happens when using a non-conformant dictionary. + * - `srcSize` is too small, and as a result, the frame header could not be decoded (only possible if `srcSize < ZSTD_FRAMEHEADERSIZE_MAX`). + * - This is not a Zstandard frame. + * When identifying the exact failure cause, it's possible to used ZSTD_getFrameParams(), which will provide a more precise error code. */ +unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize) +{ + ZSTD_frameParams zfp = { 0 , 0 , 0 , 0 }; + size_t const hError = ZSTD_getFrameParams(&zfp, src, srcSize); + if (ZSTD_isError(hError)) return 0; + return zfp.dictID; +} + + +/*! ZSTD_decompress_usingDDict() : +* Decompression using a pre-digested Dictionary +* Use dictionary without significant overhead. */ +size_t ZSTD_decompress_usingDDict(ZSTD_DCtx* dctx, + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const ZSTD_DDict* ddict) +{ + /* pass content and size in case legacy frames are encountered */ + return ZSTD_decompressMultiFrame(dctx, dst, dstCapacity, src, srcSize, + NULL, 0, + ddict); +} + + +/*===================================== +* Streaming decompression +*====================================*/ + +typedef enum { zdss_init, zdss_loadHeader, + zdss_read, zdss_load, zdss_flush } ZSTD_dStreamStage; + +/* *** Resource management *** */ +struct ZSTD_DStream_s { + ZSTD_DCtx* dctx; + ZSTD_DDict* ddictLocal; + const ZSTD_DDict* ddict; + ZSTD_frameParams fParams; + ZSTD_dStreamStage stage; + char* inBuff; + size_t inBuffSize; + size_t inPos; + size_t maxWindowSize; + char* outBuff; + size_t outBuffSize; + size_t outStart; + size_t outEnd; + size_t blockSize; + BYTE headerBuffer[ZSTD_FRAMEHEADERSIZE_MAX]; /* tmp buffer to store frame header */ + size_t lhSize; + ZSTD_customMem customMem; + void* legacyContext; + U32 previousLegacyVersion; + U32 legacyVersion; + U32 hostageByte; +}; /* typedef'd to ZSTD_DStream within "zstd.h" */ + + +ZSTD_DStream* ZSTD_createDStream(void) +{ + return ZSTD_createDStream_advanced(defaultCustomMem); +} + +ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem) +{ + ZSTD_DStream* zds; + + if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; + if (!customMem.customAlloc || !customMem.customFree) return NULL; + + zds = (ZSTD_DStream*) ZSTD_malloc(sizeof(ZSTD_DStream), customMem); + if (zds==NULL) return NULL; + memset(zds, 0, sizeof(ZSTD_DStream)); + memcpy(&zds->customMem, &customMem, sizeof(ZSTD_customMem)); + zds->dctx = ZSTD_createDCtx_advanced(customMem); + if (zds->dctx == NULL) { ZSTD_freeDStream(zds); return NULL; } + zds->stage = zdss_init; + zds->maxWindowSize = ZSTD_MAXWINDOWSIZE_DEFAULT; + return zds; +} + +size_t ZSTD_freeDStream(ZSTD_DStream* zds) +{ + if (zds==NULL) return 0; /* support free on null */ + { ZSTD_customMem const cMem = zds->customMem; + ZSTD_freeDCtx(zds->dctx); + zds->dctx = NULL; + ZSTD_freeDDict(zds->ddictLocal); + zds->ddictLocal = NULL; + ZSTD_free(zds->inBuff, cMem); + zds->inBuff = NULL; + ZSTD_free(zds->outBuff, cMem); + zds->outBuff = NULL; +#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) + if (zds->legacyContext) + ZSTD_freeLegacyStreamContext(zds->legacyContext, zds->previousLegacyVersion); +#endif + ZSTD_free(zds, cMem); + return 0; + } +} + + +/* *** Initialization *** */ + +size_t ZSTD_DStreamInSize(void) { return ZSTD_BLOCKSIZE_ABSOLUTEMAX + ZSTD_blockHeaderSize; } +size_t ZSTD_DStreamOutSize(void) { return ZSTD_BLOCKSIZE_ABSOLUTEMAX; } + +size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize) +{ + zds->stage = zdss_loadHeader; + zds->lhSize = zds->inPos = zds->outStart = zds->outEnd = 0; + ZSTD_freeDDict(zds->ddictLocal); + if (dict && dictSize >= 8) { + zds->ddictLocal = ZSTD_createDDict(dict, dictSize); + if (zds->ddictLocal == NULL) return ERROR(memory_allocation); + } else zds->ddictLocal = NULL; + zds->ddict = zds->ddictLocal; + zds->legacyVersion = 0; + zds->hostageByte = 0; + return ZSTD_frameHeaderSize_prefix; +} + +size_t ZSTD_initDStream(ZSTD_DStream* zds) +{ + return ZSTD_initDStream_usingDict(zds, NULL, 0); +} + +size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, const ZSTD_DDict* ddict) /**< note : ddict will just be referenced, and must outlive decompression session */ +{ + size_t const initResult = ZSTD_initDStream(zds); + zds->ddict = ddict; + return initResult; +} + +size_t ZSTD_resetDStream(ZSTD_DStream* zds) +{ + zds->stage = zdss_loadHeader; + zds->lhSize = zds->inPos = zds->outStart = zds->outEnd = 0; + zds->legacyVersion = 0; + zds->hostageByte = 0; + return ZSTD_frameHeaderSize_prefix; +} + +size_t ZSTD_setDStreamParameter(ZSTD_DStream* zds, + ZSTD_DStreamParameter_e paramType, unsigned paramValue) +{ + switch(paramType) + { + default : return ERROR(parameter_unknown); + case DStream_p_maxWindowSize : zds->maxWindowSize = paramValue ? paramValue : (U32)(-1); break; + } + return 0; +} + + +size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds) +{ + if (zds==NULL) return 0; /* support sizeof on NULL */ + return sizeof(*zds) + ZSTD_sizeof_DCtx(zds->dctx) + ZSTD_sizeof_DDict(zds->ddictLocal) + zds->inBuffSize + zds->outBuffSize; +} + + +/* ***** Decompression ***** */ + +MEM_STATIC size_t ZSTD_limitCopy(void* dst, size_t dstCapacity, const void* src, size_t srcSize) +{ + size_t const length = MIN(dstCapacity, srcSize); + memcpy(dst, src, length); + return length; +} + + +size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input) +{ + const char* const istart = (const char*)(input->src) + input->pos; + const char* const iend = (const char*)(input->src) + input->size; + const char* ip = istart; + char* const ostart = (char*)(output->dst) + output->pos; + char* const oend = (char*)(output->dst) + output->size; + char* op = ostart; + U32 someMoreWork = 1; + +#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1) + if (zds->legacyVersion) + return ZSTD_decompressLegacyStream(zds->legacyContext, zds->legacyVersion, output, input); +#endif + + while (someMoreWork) { + switch(zds->stage) + { + case zdss_init : + ZSTD_resetDStream(zds); /* transparent reset on starting decoding a new frame */ + /* fall-through */ + + case zdss_loadHeader : + { size_t const hSize = ZSTD_getFrameParams(&zds->fParams, zds->headerBuffer, zds->lhSize); + if (ZSTD_isError(hSize)) +#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1) + { U32 const legacyVersion = ZSTD_isLegacy(istart, iend-istart); + if (legacyVersion) { + const void* const dict = zds->ddict ? zds->ddict->dictContent : NULL; + size_t const dictSize = zds->ddict ? zds->ddict->dictSize : 0; + CHECK_F(ZSTD_initLegacyStream(&zds->legacyContext, zds->previousLegacyVersion, legacyVersion, + dict, dictSize)); + zds->legacyVersion = zds->previousLegacyVersion = legacyVersion; + return ZSTD_decompressLegacyStream(zds->legacyContext, zds->legacyVersion, output, input); + } else { + return hSize; /* error */ + } } +#else + return hSize; +#endif + if (hSize != 0) { /* need more input */ + size_t const toLoad = hSize - zds->lhSize; /* if hSize!=0, hSize > zds->lhSize */ + if (toLoad > (size_t)(iend-ip)) { /* not enough input to load full header */ + memcpy(zds->headerBuffer + zds->lhSize, ip, iend-ip); + zds->lhSize += iend-ip; + input->pos = input->size; + return (MAX(ZSTD_frameHeaderSize_min, hSize) - zds->lhSize) + ZSTD_blockHeaderSize; /* remaining header bytes + next block header */ + } + memcpy(zds->headerBuffer + zds->lhSize, ip, toLoad); zds->lhSize = hSize; ip += toLoad; + break; + } } + + /* check for single-pass mode opportunity */ + if (zds->fParams.frameContentSize && zds->fParams.windowSize /* skippable frame if == 0 */ + && (U64)(size_t)(oend-op) >= zds->fParams.frameContentSize) { + size_t const cSize = ZSTD_findFrameCompressedSize(istart, iend-istart); + if (cSize <= (size_t)(iend-istart)) { + size_t const decompressedSize = ZSTD_decompress_usingDDict(zds->dctx, op, oend-op, istart, cSize, zds->ddict); + if (ZSTD_isError(decompressedSize)) return decompressedSize; + ip = istart + cSize; + op += decompressedSize; + zds->dctx->expected = 0; + zds->stage = zdss_init; + someMoreWork = 0; + break; + } } + + /* Consume header */ + ZSTD_refDDict(zds->dctx, zds->ddict); + { size_t const h1Size = ZSTD_nextSrcSizeToDecompress(zds->dctx); /* == ZSTD_frameHeaderSize_prefix */ + CHECK_F(ZSTD_decompressContinue(zds->dctx, NULL, 0, zds->headerBuffer, h1Size)); + { size_t const h2Size = ZSTD_nextSrcSizeToDecompress(zds->dctx); + CHECK_F(ZSTD_decompressContinue(zds->dctx, NULL, 0, zds->headerBuffer+h1Size, h2Size)); + } } + + zds->fParams.windowSize = MAX(zds->fParams.windowSize, 1U << ZSTD_WINDOWLOG_ABSOLUTEMIN); + if (zds->fParams.windowSize > zds->maxWindowSize) return ERROR(frameParameter_windowTooLarge); + + /* Adapt buffer sizes to frame header instructions */ + { size_t const blockSize = MIN(zds->fParams.windowSize, ZSTD_BLOCKSIZE_ABSOLUTEMAX); + size_t const neededOutSize = zds->fParams.windowSize + blockSize + WILDCOPY_OVERLENGTH * 2; + zds->blockSize = blockSize; + if (zds->inBuffSize < blockSize) { + ZSTD_free(zds->inBuff, zds->customMem); + zds->inBuffSize = blockSize; + zds->inBuff = (char*)ZSTD_malloc(blockSize, zds->customMem); + if (zds->inBuff == NULL) return ERROR(memory_allocation); + } + if (zds->outBuffSize < neededOutSize) { + ZSTD_free(zds->outBuff, zds->customMem); + zds->outBuffSize = neededOutSize; + zds->outBuff = (char*)ZSTD_malloc(neededOutSize, zds->customMem); + if (zds->outBuff == NULL) return ERROR(memory_allocation); + } } + zds->stage = zdss_read; + /* pass-through */ + + case zdss_read: + { size_t const neededInSize = ZSTD_nextSrcSizeToDecompress(zds->dctx); + if (neededInSize==0) { /* end of frame */ + zds->stage = zdss_init; + someMoreWork = 0; + break; + } + if ((size_t)(iend-ip) >= neededInSize) { /* decode directly from src */ + const int isSkipFrame = ZSTD_isSkipFrame(zds->dctx); + size_t const decodedSize = ZSTD_decompressContinue(zds->dctx, + zds->outBuff + zds->outStart, (isSkipFrame ? 0 : zds->outBuffSize - zds->outStart), + ip, neededInSize); + if (ZSTD_isError(decodedSize)) return decodedSize; + ip += neededInSize; + if (!decodedSize && !isSkipFrame) break; /* this was just a header */ + zds->outEnd = zds->outStart + decodedSize; + zds->stage = zdss_flush; + break; + } + if (ip==iend) { someMoreWork = 0; break; } /* no more input */ + zds->stage = zdss_load; + /* pass-through */ + } + + case zdss_load: + { size_t const neededInSize = ZSTD_nextSrcSizeToDecompress(zds->dctx); + size_t const toLoad = neededInSize - zds->inPos; /* should always be <= remaining space within inBuff */ + size_t loadedSize; + if (toLoad > zds->inBuffSize - zds->inPos) return ERROR(corruption_detected); /* should never happen */ + loadedSize = ZSTD_limitCopy(zds->inBuff + zds->inPos, toLoad, ip, iend-ip); + ip += loadedSize; + zds->inPos += loadedSize; + if (loadedSize < toLoad) { someMoreWork = 0; break; } /* not enough input, wait for more */ + + /* decode loaded input */ + { const int isSkipFrame = ZSTD_isSkipFrame(zds->dctx); + size_t const decodedSize = ZSTD_decompressContinue(zds->dctx, + zds->outBuff + zds->outStart, zds->outBuffSize - zds->outStart, + zds->inBuff, neededInSize); + if (ZSTD_isError(decodedSize)) return decodedSize; + zds->inPos = 0; /* input is consumed */ + if (!decodedSize && !isSkipFrame) { zds->stage = zdss_read; break; } /* this was just a header */ + zds->outEnd = zds->outStart + decodedSize; + zds->stage = zdss_flush; + /* pass-through */ + } } + + case zdss_flush: + { size_t const toFlushSize = zds->outEnd - zds->outStart; + size_t const flushedSize = ZSTD_limitCopy(op, oend-op, zds->outBuff + zds->outStart, toFlushSize); + op += flushedSize; + zds->outStart += flushedSize; + if (flushedSize == toFlushSize) { /* flush completed */ + zds->stage = zdss_read; + if (zds->outStart + zds->blockSize > zds->outBuffSize) + zds->outStart = zds->outEnd = 0; + break; + } + /* cannot complete flush */ + someMoreWork = 0; + break; + } + default: return ERROR(GENERIC); /* impossible */ + } } + + /* result */ + input->pos += (size_t)(ip-istart); + output->pos += (size_t)(op-ostart); + { size_t nextSrcSizeHint = ZSTD_nextSrcSizeToDecompress(zds->dctx); + if (!nextSrcSizeHint) { /* frame fully decoded */ + if (zds->outEnd == zds->outStart) { /* output fully flushed */ + if (zds->hostageByte) { + if (input->pos >= input->size) { zds->stage = zdss_read; return 1; } /* can't release hostage (not present) */ + input->pos++; /* release hostage */ + } + return 0; + } + if (!zds->hostageByte) { /* output not fully flushed; keep last byte as hostage; will be released when all output is flushed */ + input->pos--; /* note : pos > 0, otherwise, impossible to finish reading last block */ + zds->hostageByte=1; + } + return 1; + } + nextSrcSizeHint += ZSTD_blockHeaderSize * (ZSTD_nextInputType(zds->dctx) == ZSTDnit_block); /* preload header of next block */ + if (zds->inPos > nextSrcSizeHint) return ERROR(GENERIC); /* should never happen */ + nextSrcSizeHint -= zds->inPos; /* already loaded*/ + return nextSrcSizeHint; + } +} diff --git a/contrib/linux-kernel/lib/zstd_errors.h b/contrib/linux-kernel/lib/zstd_errors.h new file mode 100644 index 000000000..3d579d969 --- /dev/null +++ b/contrib/linux-kernel/lib/zstd_errors.h @@ -0,0 +1,75 @@ +/** + * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + +#ifndef ZSTD_ERRORS_H_398273423 +#define ZSTD_ERRORS_H_398273423 + +#if defined (__cplusplus) +extern "C" { +#endif + +/*===== dependency =====*/ +#include /* size_t */ + + +/* ===== ZSTDERRORLIB_API : control library symbols visibility ===== */ +#if defined(__GNUC__) && (__GNUC__ >= 4) +# define ZSTDERRORLIB_VISIBILITY __attribute__ ((visibility ("default"))) +#else +# define ZSTDERRORLIB_VISIBILITY +#endif +#if defined(ZSTD_DLL_EXPORT) && (ZSTD_DLL_EXPORT==1) +# define ZSTDERRORLIB_API __declspec(dllexport) ZSTDERRORLIB_VISIBILITY +#elif defined(ZSTD_DLL_IMPORT) && (ZSTD_DLL_IMPORT==1) +# define ZSTDERRORLIB_API __declspec(dllimport) ZSTDERRORLIB_VISIBILITY /* It isn't required but allows to generate better code, saving a function pointer load from the IAT and an indirect jump.*/ +#else +# define ZSTDERRORLIB_API ZSTDERRORLIB_VISIBILITY +#endif + +/*-**************************************** +* error codes list +******************************************/ +typedef enum { + ZSTD_error_no_error, + ZSTD_error_GENERIC, + ZSTD_error_prefix_unknown, + ZSTD_error_version_unsupported, + ZSTD_error_parameter_unknown, + ZSTD_error_frameParameter_unsupported, + ZSTD_error_frameParameter_unsupportedBy32bits, + ZSTD_error_frameParameter_windowTooLarge, + ZSTD_error_compressionParameter_unsupported, + ZSTD_error_init_missing, + ZSTD_error_memory_allocation, + ZSTD_error_stage_wrong, + ZSTD_error_dstSize_tooSmall, + ZSTD_error_srcSize_wrong, + ZSTD_error_corruption_detected, + ZSTD_error_checksum_wrong, + ZSTD_error_tableLog_tooLarge, + ZSTD_error_maxSymbolValue_tooLarge, + ZSTD_error_maxSymbolValue_tooSmall, + ZSTD_error_dictionary_corrupted, + ZSTD_error_dictionary_wrong, + ZSTD_error_dictionaryCreation_failed, + ZSTD_error_maxCode +} ZSTD_ErrorCode; + +/*! ZSTD_getErrorCode() : + convert a `size_t` function result into a `ZSTD_ErrorCode` enum type, + which can be used to compare directly with enum list published into "error_public.h" */ +ZSTDERRORLIB_API ZSTD_ErrorCode ZSTD_getErrorCode(size_t functionResult); +ZSTDERRORLIB_API const char* ZSTD_getErrorString(ZSTD_ErrorCode code); + + +#if defined (__cplusplus) +} +#endif + +#endif /* ZSTD_ERRORS_H_398273423 */ diff --git a/contrib/linux-kernel/lib/zstd_internal.h b/contrib/linux-kernel/lib/zstd_internal.h new file mode 100644 index 000000000..5c5b28732 --- /dev/null +++ b/contrib/linux-kernel/lib/zstd_internal.h @@ -0,0 +1,283 @@ +/** + * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + +#ifndef ZSTD_CCOMMON_H_MODULE +#define ZSTD_CCOMMON_H_MODULE + +/*-******************************************************* +* Compiler specifics +*********************************************************/ +#ifdef _MSC_VER /* Visual Studio */ +# define FORCE_INLINE static __forceinline +# include /* For Visual 2005 */ +# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ +# pragma warning(disable : 4324) /* disable: C4324: padded structure */ +# pragma warning(disable : 4100) /* disable: C4100: unreferenced formal parameter */ +#else +# if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ +# ifdef __GNUC__ +# define FORCE_INLINE static inline __attribute__((always_inline)) +# else +# define FORCE_INLINE static inline +# endif +# else +# define FORCE_INLINE static +# endif /* __STDC_VERSION__ */ +#endif + +#ifdef _MSC_VER +# define FORCE_NOINLINE static __declspec(noinline) +#else +# ifdef __GNUC__ +# define FORCE_NOINLINE static __attribute__((__noinline__)) +# else +# define FORCE_NOINLINE static +# endif +#endif + + +/*-************************************* +* Dependencies +***************************************/ +#include "mem.h" +#include "error_private.h" +#define ZSTD_STATIC_LINKING_ONLY +#include "zstd.h" +#ifndef XXH_STATIC_LINKING_ONLY +# define XXH_STATIC_LINKING_ONLY /* XXH64_state_t */ +#endif +#include "xxhash.h" /* XXH_reset, update, digest */ + + +/*-************************************* +* shared macros +***************************************/ +#define MIN(a,b) ((a)<(b) ? (a) : (b)) +#define MAX(a,b) ((a)>(b) ? (a) : (b)) +#define CHECK_F(f) { size_t const errcod = f; if (ERR_isError(errcod)) return errcod; } /* check and Forward error code */ +#define CHECK_E(f, e) { size_t const errcod = f; if (ERR_isError(errcod)) return ERROR(e); } /* check and send Error code */ + + +/*-************************************* +* Common constants +***************************************/ +#define ZSTD_OPT_NUM (1<<12) +#define ZSTD_DICT_MAGIC 0xEC30A437 /* v0.7+ */ + +#define ZSTD_REP_NUM 3 /* number of repcodes */ +#define ZSTD_REP_CHECK (ZSTD_REP_NUM) /* number of repcodes to check by the optimal parser */ +#define ZSTD_REP_MOVE (ZSTD_REP_NUM-1) +#define ZSTD_REP_MOVE_OPT (ZSTD_REP_NUM) +static const U32 repStartValue[ZSTD_REP_NUM] = { 1, 4, 8 }; + +#define KB *(1 <<10) +#define MB *(1 <<20) +#define GB *(1U<<30) + +#define BIT7 128 +#define BIT6 64 +#define BIT5 32 +#define BIT4 16 +#define BIT1 2 +#define BIT0 1 + +#define ZSTD_WINDOWLOG_ABSOLUTEMIN 10 +static const size_t ZSTD_fcs_fieldSize[4] = { 0, 2, 4, 8 }; +static const size_t ZSTD_did_fieldSize[4] = { 0, 1, 2, 4 }; + +#define ZSTD_BLOCKHEADERSIZE 3 /* C standard doesn't allow `static const` variable to be init using another `static const` variable */ +static const size_t ZSTD_blockHeaderSize = ZSTD_BLOCKHEADERSIZE; +typedef enum { bt_raw, bt_rle, bt_compressed, bt_reserved } blockType_e; + +#define MIN_SEQUENCES_SIZE 1 /* nbSeq==0 */ +#define MIN_CBLOCK_SIZE (1 /*litCSize*/ + 1 /* RLE or RAW */ + MIN_SEQUENCES_SIZE /* nbSeq==0 */) /* for a non-null block */ + +#define HufLog 12 +typedef enum { set_basic, set_rle, set_compressed, set_repeat } symbolEncodingType_e; + +#define LONGNBSEQ 0x7F00 + +#define MINMATCH 3 +#define EQUAL_READ32 4 + +#define Litbits 8 +#define MaxLit ((1<= 3) /* GCC Intrinsic */ + return 31 - __builtin_clz(val); +# else /* Software version */ + static const int DeBruijnClz[32] = { 0, 9, 1, 10, 13, 21, 2, 29, 11, 14, 16, 18, 22, 25, 3, 30, 8, 12, 20, 28, 15, 17, 24, 7, 19, 27, 23, 6, 26, 5, 4, 31 }; + U32 v = val; + int r; + v |= v >> 1; + v |= v >> 2; + v |= v >> 4; + v |= v >> 8; + v |= v >> 16; + r = DeBruijnClz[(U32)(v * 0x07C4ACDDU) >> 27]; + return r; +# endif +} + + +/* hidden functions */ + +/* ZSTD_invalidateRepCodes() : + * ensures next compression will not use repcodes from previous block. + * Note : only works with regular variant; + * do not use with extDict variant ! */ +void ZSTD_invalidateRepCodes(ZSTD_CCtx* cctx); + + +#endif /* ZSTD_CCOMMON_H_MODULE */ diff --git a/contrib/linux-kernel/lib/zstd_opt.h b/contrib/linux-kernel/lib/zstd_opt.h new file mode 100644 index 000000000..543761191 --- /dev/null +++ b/contrib/linux-kernel/lib/zstd_opt.h @@ -0,0 +1,921 @@ +/** + * Copyright (c) 2016-present, Przemyslaw Skibinski, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + + +/* Note : this file is intended to be included within zstd_compress.c */ + + +#ifndef ZSTD_OPT_H_91842398743 +#define ZSTD_OPT_H_91842398743 + + +#define ZSTD_LITFREQ_ADD 2 +#define ZSTD_FREQ_DIV 4 +#define ZSTD_MAX_PRICE (1<<30) + +/*-************************************* +* Price functions for optimal parser +***************************************/ +FORCE_INLINE void ZSTD_setLog2Prices(seqStore_t* ssPtr) +{ + ssPtr->log2matchLengthSum = ZSTD_highbit32(ssPtr->matchLengthSum+1); + ssPtr->log2litLengthSum = ZSTD_highbit32(ssPtr->litLengthSum+1); + ssPtr->log2litSum = ZSTD_highbit32(ssPtr->litSum+1); + ssPtr->log2offCodeSum = ZSTD_highbit32(ssPtr->offCodeSum+1); + ssPtr->factor = 1 + ((ssPtr->litSum>>5) / ssPtr->litLengthSum) + ((ssPtr->litSum<<1) / (ssPtr->litSum + ssPtr->matchSum)); +} + + +MEM_STATIC void ZSTD_rescaleFreqs(seqStore_t* ssPtr, const BYTE* src, size_t srcSize) +{ + unsigned u; + + ssPtr->cachedLiterals = NULL; + ssPtr->cachedPrice = ssPtr->cachedLitLength = 0; + ssPtr->staticPrices = 0; + + if (ssPtr->litLengthSum == 0) { + if (srcSize <= 1024) ssPtr->staticPrices = 1; + + for (u=0; u<=MaxLit; u++) + ssPtr->litFreq[u] = 0; + for (u=0; ulitFreq[src[u]]++; + + ssPtr->litSum = 0; + ssPtr->litLengthSum = MaxLL+1; + ssPtr->matchLengthSum = MaxML+1; + ssPtr->offCodeSum = (MaxOff+1); + ssPtr->matchSum = (ZSTD_LITFREQ_ADD<litFreq[u] = 1 + (ssPtr->litFreq[u]>>ZSTD_FREQ_DIV); + ssPtr->litSum += ssPtr->litFreq[u]; + } + for (u=0; u<=MaxLL; u++) + ssPtr->litLengthFreq[u] = 1; + for (u=0; u<=MaxML; u++) + ssPtr->matchLengthFreq[u] = 1; + for (u=0; u<=MaxOff; u++) + ssPtr->offCodeFreq[u] = 1; + } else { + ssPtr->matchLengthSum = 0; + ssPtr->litLengthSum = 0; + ssPtr->offCodeSum = 0; + ssPtr->matchSum = 0; + ssPtr->litSum = 0; + + for (u=0; u<=MaxLit; u++) { + ssPtr->litFreq[u] = 1 + (ssPtr->litFreq[u]>>(ZSTD_FREQ_DIV+1)); + ssPtr->litSum += ssPtr->litFreq[u]; + } + for (u=0; u<=MaxLL; u++) { + ssPtr->litLengthFreq[u] = 1 + (ssPtr->litLengthFreq[u]>>(ZSTD_FREQ_DIV+1)); + ssPtr->litLengthSum += ssPtr->litLengthFreq[u]; + } + for (u=0; u<=MaxML; u++) { + ssPtr->matchLengthFreq[u] = 1 + (ssPtr->matchLengthFreq[u]>>ZSTD_FREQ_DIV); + ssPtr->matchLengthSum += ssPtr->matchLengthFreq[u]; + ssPtr->matchSum += ssPtr->matchLengthFreq[u] * (u + 3); + } + ssPtr->matchSum *= ZSTD_LITFREQ_ADD; + for (u=0; u<=MaxOff; u++) { + ssPtr->offCodeFreq[u] = 1 + (ssPtr->offCodeFreq[u]>>ZSTD_FREQ_DIV); + ssPtr->offCodeSum += ssPtr->offCodeFreq[u]; + } + } + + ZSTD_setLog2Prices(ssPtr); +} + + +FORCE_INLINE U32 ZSTD_getLiteralPrice(seqStore_t* ssPtr, U32 litLength, const BYTE* literals) +{ + U32 price, u; + + if (ssPtr->staticPrices) + return ZSTD_highbit32((U32)litLength+1) + (litLength*6); + + if (litLength == 0) + return ssPtr->log2litLengthSum - ZSTD_highbit32(ssPtr->litLengthFreq[0]+1); + + /* literals */ + if (ssPtr->cachedLiterals == literals) { + U32 const additional = litLength - ssPtr->cachedLitLength; + const BYTE* literals2 = ssPtr->cachedLiterals + ssPtr->cachedLitLength; + price = ssPtr->cachedPrice + additional * ssPtr->log2litSum; + for (u=0; u < additional; u++) + price -= ZSTD_highbit32(ssPtr->litFreq[literals2[u]]+1); + ssPtr->cachedPrice = price; + ssPtr->cachedLitLength = litLength; + } else { + price = litLength * ssPtr->log2litSum; + for (u=0; u < litLength; u++) + price -= ZSTD_highbit32(ssPtr->litFreq[literals[u]]+1); + + if (litLength >= 12) { + ssPtr->cachedLiterals = literals; + ssPtr->cachedPrice = price; + ssPtr->cachedLitLength = litLength; + } + } + + /* literal Length */ + { const BYTE LL_deltaCode = 19; + const BYTE llCode = (litLength>63) ? (BYTE)ZSTD_highbit32(litLength) + LL_deltaCode : LL_Code[litLength]; + price += LL_bits[llCode] + ssPtr->log2litLengthSum - ZSTD_highbit32(ssPtr->litLengthFreq[llCode]+1); + } + + return price; +} + + +FORCE_INLINE U32 ZSTD_getPrice(seqStore_t* seqStorePtr, U32 litLength, const BYTE* literals, U32 offset, U32 matchLength, const int ultra) +{ + /* offset */ + U32 price; + BYTE const offCode = (BYTE)ZSTD_highbit32(offset+1); + + if (seqStorePtr->staticPrices) + return ZSTD_getLiteralPrice(seqStorePtr, litLength, literals) + ZSTD_highbit32((U32)matchLength+1) + 16 + offCode; + + price = offCode + seqStorePtr->log2offCodeSum - ZSTD_highbit32(seqStorePtr->offCodeFreq[offCode]+1); + if (!ultra && offCode >= 20) price += (offCode-19)*2; + + /* match Length */ + { const BYTE ML_deltaCode = 36; + const BYTE mlCode = (matchLength>127) ? (BYTE)ZSTD_highbit32(matchLength) + ML_deltaCode : ML_Code[matchLength]; + price += ML_bits[mlCode] + seqStorePtr->log2matchLengthSum - ZSTD_highbit32(seqStorePtr->matchLengthFreq[mlCode]+1); + } + + return price + ZSTD_getLiteralPrice(seqStorePtr, litLength, literals) + seqStorePtr->factor; +} + + +MEM_STATIC void ZSTD_updatePrice(seqStore_t* seqStorePtr, U32 litLength, const BYTE* literals, U32 offset, U32 matchLength) +{ + U32 u; + + /* literals */ + seqStorePtr->litSum += litLength*ZSTD_LITFREQ_ADD; + for (u=0; u < litLength; u++) + seqStorePtr->litFreq[literals[u]] += ZSTD_LITFREQ_ADD; + + /* literal Length */ + { const BYTE LL_deltaCode = 19; + const BYTE llCode = (litLength>63) ? (BYTE)ZSTD_highbit32(litLength) + LL_deltaCode : LL_Code[litLength]; + seqStorePtr->litLengthFreq[llCode]++; + seqStorePtr->litLengthSum++; + } + + /* match offset */ + { BYTE const offCode = (BYTE)ZSTD_highbit32(offset+1); + seqStorePtr->offCodeSum++; + seqStorePtr->offCodeFreq[offCode]++; + } + + /* match Length */ + { const BYTE ML_deltaCode = 36; + const BYTE mlCode = (matchLength>127) ? (BYTE)ZSTD_highbit32(matchLength) + ML_deltaCode : ML_Code[matchLength]; + seqStorePtr->matchLengthFreq[mlCode]++; + seqStorePtr->matchLengthSum++; + } + + ZSTD_setLog2Prices(seqStorePtr); +} + + +#define SET_PRICE(pos, mlen_, offset_, litlen_, price_) \ + { \ + while (last_pos < pos) { opt[last_pos+1].price = ZSTD_MAX_PRICE; last_pos++; } \ + opt[pos].mlen = mlen_; \ + opt[pos].off = offset_; \ + opt[pos].litlen = litlen_; \ + opt[pos].price = price_; \ + } + + + +/* Update hashTable3 up to ip (excluded) + Assumption : always within prefix (i.e. not within extDict) */ +FORCE_INLINE +U32 ZSTD_insertAndFindFirstIndexHash3 (ZSTD_CCtx* zc, const BYTE* ip) +{ + U32* const hashTable3 = zc->hashTable3; + U32 const hashLog3 = zc->hashLog3; + const BYTE* const base = zc->base; + U32 idx = zc->nextToUpdate3; + const U32 target = zc->nextToUpdate3 = (U32)(ip - base); + const size_t hash3 = ZSTD_hash3Ptr(ip, hashLog3); + + while(idx < target) { + hashTable3[ZSTD_hash3Ptr(base+idx, hashLog3)] = idx; + idx++; + } + + return hashTable3[hash3]; +} + + +/*-************************************* +* Binary Tree search +***************************************/ +static U32 ZSTD_insertBtAndGetAllMatches ( + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iLimit, + U32 nbCompares, const U32 mls, + U32 extDict, ZSTD_match_t* matches, const U32 minMatchLen) +{ + const BYTE* const base = zc->base; + const U32 current = (U32)(ip-base); + const U32 hashLog = zc->params.cParams.hashLog; + const size_t h = ZSTD_hashPtr(ip, hashLog, mls); + U32* const hashTable = zc->hashTable; + U32 matchIndex = hashTable[h]; + U32* const bt = zc->chainTable; + const U32 btLog = zc->params.cParams.chainLog - 1; + const U32 btMask= (1U << btLog) - 1; + size_t commonLengthSmaller=0, commonLengthLarger=0; + const BYTE* const dictBase = zc->dictBase; + const U32 dictLimit = zc->dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const prefixStart = base + dictLimit; + const U32 btLow = btMask >= current ? 0 : current - btMask; + const U32 windowLow = zc->lowLimit; + U32* smallerPtr = bt + 2*(current&btMask); + U32* largerPtr = bt + 2*(current&btMask) + 1; + U32 matchEndIdx = current+8; + U32 dummy32; /* to be nullified at the end */ + U32 mnum = 0; + + const U32 minMatch = (mls == 3) ? 3 : 4; + size_t bestLength = minMatchLen-1; + + if (minMatch == 3) { /* HC3 match finder */ + U32 const matchIndex3 = ZSTD_insertAndFindFirstIndexHash3 (zc, ip); + if (matchIndex3>windowLow && (current - matchIndex3 < (1<<18))) { + const BYTE* match; + size_t currentMl=0; + if ((!extDict) || matchIndex3 >= dictLimit) { + match = base + matchIndex3; + if (match[bestLength] == ip[bestLength]) currentMl = ZSTD_count(ip, match, iLimit); + } else { + match = dictBase + matchIndex3; + if (MEM_readMINMATCH(match, MINMATCH) == MEM_readMINMATCH(ip, MINMATCH)) /* assumption : matchIndex3 <= dictLimit-4 (by table construction) */ + currentMl = ZSTD_count_2segments(ip+MINMATCH, match+MINMATCH, iLimit, dictEnd, prefixStart) + MINMATCH; + } + + /* save best solution */ + if (currentMl > bestLength) { + bestLength = currentMl; + matches[mnum].off = ZSTD_REP_MOVE_OPT + current - matchIndex3; + matches[mnum].len = (U32)currentMl; + mnum++; + if (currentMl > ZSTD_OPT_NUM) goto update; + if (ip+currentMl == iLimit) goto update; /* best possible, and avoid read overflow*/ + } + } + } + + hashTable[h] = current; /* Update Hash Table */ + + while (nbCompares-- && (matchIndex > windowLow)) { + U32* nextPtr = bt + 2*(matchIndex & btMask); + size_t matchLength = MIN(commonLengthSmaller, commonLengthLarger); /* guaranteed minimum nb of common bytes */ + const BYTE* match; + + if ((!extDict) || (matchIndex+matchLength >= dictLimit)) { + match = base + matchIndex; + if (match[matchLength] == ip[matchLength]) { + matchLength += ZSTD_count(ip+matchLength+1, match+matchLength+1, iLimit) +1; + } + } else { + match = dictBase + matchIndex; + matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iLimit, dictEnd, prefixStart); + if (matchIndex+matchLength >= dictLimit) + match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ + } + + if (matchLength > bestLength) { + if (matchLength > matchEndIdx - matchIndex) matchEndIdx = matchIndex + (U32)matchLength; + bestLength = matchLength; + matches[mnum].off = ZSTD_REP_MOVE_OPT + current - matchIndex; + matches[mnum].len = (U32)matchLength; + mnum++; + if (matchLength > ZSTD_OPT_NUM) break; + if (ip+matchLength == iLimit) /* equal : no way to know if inf or sup */ + break; /* drop, to guarantee consistency (miss a little bit of compression) */ + } + + if (match[matchLength] < ip[matchLength]) { + /* match is smaller than current */ + *smallerPtr = matchIndex; /* update smaller idx */ + commonLengthSmaller = matchLength; /* all smaller will now have at least this guaranteed common length */ + if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ + matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ + } else { + /* match is larger than current */ + *largerPtr = matchIndex; + commonLengthLarger = matchLength; + if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + largerPtr = nextPtr; + matchIndex = nextPtr[0]; + } } + + *smallerPtr = *largerPtr = 0; + +update: + zc->nextToUpdate = (matchEndIdx > current + 8) ? matchEndIdx - 8 : current+1; + return mnum; +} + + +/** Tree updater, providing best match */ +static U32 ZSTD_BtGetAllMatches ( + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iLimit, + const U32 maxNbAttempts, const U32 mls, ZSTD_match_t* matches, const U32 minMatchLen) +{ + if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ + ZSTD_updateTree(zc, ip, iLimit, maxNbAttempts, mls); + return ZSTD_insertBtAndGetAllMatches(zc, ip, iLimit, maxNbAttempts, mls, 0, matches, minMatchLen); +} + + +static U32 ZSTD_BtGetAllMatches_selectMLS ( + ZSTD_CCtx* zc, /* Index table will be updated */ + const BYTE* ip, const BYTE* const iHighLimit, + const U32 maxNbAttempts, const U32 matchLengthSearch, ZSTD_match_t* matches, const U32 minMatchLen) +{ + switch(matchLengthSearch) + { + case 3 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 3, matches, minMatchLen); + default : + case 4 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 4, matches, minMatchLen); + case 5 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 5, matches, minMatchLen); + case 7 : + case 6 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 6, matches, minMatchLen); + } +} + +/** Tree updater, providing best match */ +static U32 ZSTD_BtGetAllMatches_extDict ( + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iLimit, + const U32 maxNbAttempts, const U32 mls, ZSTD_match_t* matches, const U32 minMatchLen) +{ + if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ + ZSTD_updateTree_extDict(zc, ip, iLimit, maxNbAttempts, mls); + return ZSTD_insertBtAndGetAllMatches(zc, ip, iLimit, maxNbAttempts, mls, 1, matches, minMatchLen); +} + + +static U32 ZSTD_BtGetAllMatches_selectMLS_extDict ( + ZSTD_CCtx* zc, /* Index table will be updated */ + const BYTE* ip, const BYTE* const iHighLimit, + const U32 maxNbAttempts, const U32 matchLengthSearch, ZSTD_match_t* matches, const U32 minMatchLen) +{ + switch(matchLengthSearch) + { + case 3 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 3, matches, minMatchLen); + default : + case 4 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 4, matches, minMatchLen); + case 5 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 5, matches, minMatchLen); + case 7 : + case 6 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 6, matches, minMatchLen); + } +} + + +/*-******************************* +* Optimal parser +*********************************/ +FORCE_INLINE +void ZSTD_compressBlock_opt_generic(ZSTD_CCtx* ctx, + const void* src, size_t srcSize, const int ultra) +{ + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + const BYTE* const base = ctx->base; + const BYTE* const prefixStart = base + ctx->dictLimit; + + const U32 maxSearches = 1U << ctx->params.cParams.searchLog; + const U32 sufficient_len = ctx->params.cParams.targetLength; + const U32 mls = ctx->params.cParams.searchLength; + const U32 minMatch = (ctx->params.cParams.searchLength == 3) ? 3 : 4; + + ZSTD_optimal_t* opt = seqStorePtr->priceTable; + ZSTD_match_t* matches = seqStorePtr->matchTable; + const BYTE* inr; + U32 offset, rep[ZSTD_REP_NUM]; + + /* init */ + ctx->nextToUpdate3 = ctx->nextToUpdate; + ZSTD_rescaleFreqs(seqStorePtr, (const BYTE*)src, srcSize); + ip += (ip==prefixStart); + { U32 i; for (i=0; irep[i]; } + + /* Match Loop */ + while (ip < ilimit) { + U32 cur, match_num, last_pos, litlen, price; + U32 u, mlen, best_mlen, best_off, litLength; + memset(opt, 0, sizeof(ZSTD_optimal_t)); + last_pos = 0; + litlen = (U32)(ip - anchor); + + /* check repCode */ + { U32 i, last_i = ZSTD_REP_CHECK + (ip==anchor); + for (i=(ip == anchor); i 0) && (repCur < (S32)(ip-prefixStart)) + && (MEM_readMINMATCH(ip, minMatch) == MEM_readMINMATCH(ip - repCur, minMatch))) { + mlen = (U32)ZSTD_count(ip+minMatch, ip+minMatch-repCur, iend) + minMatch; + if (mlen > sufficient_len || mlen >= ZSTD_OPT_NUM) { + best_mlen = mlen; best_off = i; cur = 0; last_pos = 1; + goto _storeSequence; + } + best_off = i - (ip == anchor); + do { + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); + if (mlen > last_pos || price < opt[mlen].price) + SET_PRICE(mlen, mlen, i, litlen, price); /* note : macro modifies last_pos */ + mlen--; + } while (mlen >= minMatch); + } } } + + match_num = ZSTD_BtGetAllMatches_selectMLS(ctx, ip, iend, maxSearches, mls, matches, minMatch); + + if (!last_pos && !match_num) { ip++; continue; } + + if (match_num && (matches[match_num-1].len > sufficient_len || matches[match_num-1].len >= ZSTD_OPT_NUM)) { + best_mlen = matches[match_num-1].len; + best_off = matches[match_num-1].off; + cur = 0; + last_pos = 1; + goto _storeSequence; + } + + /* set prices using matches at position = 0 */ + best_mlen = (last_pos) ? last_pos : minMatch; + for (u = 0; u < match_num; u++) { + mlen = (u>0) ? matches[u-1].len+1 : best_mlen; + best_mlen = matches[u].len; + while (mlen <= best_mlen) { + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); + if (mlen > last_pos || price < opt[mlen].price) + SET_PRICE(mlen, mlen, matches[u].off, litlen, price); /* note : macro modifies last_pos */ + mlen++; + } } + + if (last_pos < minMatch) { ip++; continue; } + + /* initialize opt[0] */ + { U32 i ; for (i=0; i litlen) { + price = opt[cur - litlen].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-litlen); + } else + price = ZSTD_getLiteralPrice(seqStorePtr, litlen, anchor); + } else { + litlen = 1; + price = opt[cur - 1].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-1); + } + + if (cur > last_pos || price <= opt[cur].price) + SET_PRICE(cur, 1, 0, litlen, price); + + if (cur == last_pos) break; + + if (inr > ilimit) /* last match must start at a minimum distance of 8 from oend */ + continue; + + mlen = opt[cur].mlen; + if (opt[cur].off > ZSTD_REP_MOVE_OPT) { + opt[cur].rep[2] = opt[cur-mlen].rep[1]; + opt[cur].rep[1] = opt[cur-mlen].rep[0]; + opt[cur].rep[0] = opt[cur].off - ZSTD_REP_MOVE_OPT; + } else { + opt[cur].rep[2] = (opt[cur].off > 1) ? opt[cur-mlen].rep[1] : opt[cur-mlen].rep[2]; + opt[cur].rep[1] = (opt[cur].off > 0) ? opt[cur-mlen].rep[0] : opt[cur-mlen].rep[1]; + opt[cur].rep[0] = ((opt[cur].off==ZSTD_REP_MOVE_OPT) && (mlen != 1)) ? (opt[cur-mlen].rep[0] - 1) : (opt[cur-mlen].rep[opt[cur].off]); + } + + best_mlen = minMatch; + { U32 i, last_i = ZSTD_REP_CHECK + (mlen != 1); + for (i=(opt[cur].mlen != 1); i 0) && (repCur < (S32)(inr-prefixStart)) + && (MEM_readMINMATCH(inr, minMatch) == MEM_readMINMATCH(inr - repCur, minMatch))) { + mlen = (U32)ZSTD_count(inr+minMatch, inr+minMatch - repCur, iend) + minMatch; + + if (mlen > sufficient_len || cur + mlen >= ZSTD_OPT_NUM) { + best_mlen = mlen; best_off = i; last_pos = cur + 1; + goto _storeSequence; + } + + best_off = i - (opt[cur].mlen != 1); + if (mlen > best_mlen) best_mlen = mlen; + + do { + if (opt[cur].mlen == 1) { + litlen = opt[cur].litlen; + if (cur > litlen) { + price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, inr-litlen, best_off, mlen - MINMATCH, ultra); + } else + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); + } else { + litlen = 0; + price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, best_off, mlen - MINMATCH, ultra); + } + + if (cur + mlen > last_pos || price <= opt[cur + mlen].price) + SET_PRICE(cur + mlen, mlen, i, litlen, price); + mlen--; + } while (mlen >= minMatch); + } } } + + match_num = ZSTD_BtGetAllMatches_selectMLS(ctx, inr, iend, maxSearches, mls, matches, best_mlen); + + if (match_num > 0 && (matches[match_num-1].len > sufficient_len || cur + matches[match_num-1].len >= ZSTD_OPT_NUM)) { + best_mlen = matches[match_num-1].len; + best_off = matches[match_num-1].off; + last_pos = cur + 1; + goto _storeSequence; + } + + /* set prices using matches at position = cur */ + for (u = 0; u < match_num; u++) { + mlen = (u>0) ? matches[u-1].len+1 : best_mlen; + best_mlen = matches[u].len; + + while (mlen <= best_mlen) { + if (opt[cur].mlen == 1) { + litlen = opt[cur].litlen; + if (cur > litlen) + price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, ip+cur-litlen, matches[u].off-1, mlen - MINMATCH, ultra); + else + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); + } else { + litlen = 0; + price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, matches[u].off-1, mlen - MINMATCH, ultra); + } + + if (cur + mlen > last_pos || (price < opt[cur + mlen].price)) + SET_PRICE(cur + mlen, mlen, matches[u].off, litlen, price); + + mlen++; + } } } + + best_mlen = opt[last_pos].mlen; + best_off = opt[last_pos].off; + cur = last_pos - best_mlen; + + /* store sequence */ +_storeSequence: /* cur, last_pos, best_mlen, best_off have to be set */ + opt[0].mlen = 1; + + while (1) { + mlen = opt[cur].mlen; + offset = opt[cur].off; + opt[cur].mlen = best_mlen; + opt[cur].off = best_off; + best_mlen = mlen; + best_off = offset; + if (mlen > cur) break; + cur -= mlen; + } + + for (u = 0; u <= last_pos;) { + u += opt[u].mlen; + } + + for (cur=0; cur < last_pos; ) { + mlen = opt[cur].mlen; + if (mlen == 1) { ip++; cur++; continue; } + offset = opt[cur].off; + cur += mlen; + litLength = (U32)(ip - anchor); + + if (offset > ZSTD_REP_MOVE_OPT) { + rep[2] = rep[1]; + rep[1] = rep[0]; + rep[0] = offset - ZSTD_REP_MOVE_OPT; + offset--; + } else { + if (offset != 0) { + best_off = (offset==ZSTD_REP_MOVE_OPT) ? (rep[0] - 1) : (rep[offset]); + if (offset != 1) rep[2] = rep[1]; + rep[1] = rep[0]; + rep[0] = best_off; + } + if (litLength==0) offset--; + } + + ZSTD_updatePrice(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); + ZSTD_storeSeq(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); + anchor = ip = ip + mlen; + } } /* for (cur=0; cur < last_pos; ) */ + + /* Save reps for next block */ + { int i; for (i=0; irepToConfirm[i] = rep[i]; } + + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } +} + + +FORCE_INLINE +void ZSTD_compressBlock_opt_extDict_generic(ZSTD_CCtx* ctx, + const void* src, size_t srcSize, const int ultra) +{ + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + const BYTE* const base = ctx->base; + const U32 lowestIndex = ctx->lowLimit; + const U32 dictLimit = ctx->dictLimit; + const BYTE* const prefixStart = base + dictLimit; + const BYTE* const dictBase = ctx->dictBase; + const BYTE* const dictEnd = dictBase + dictLimit; + + const U32 maxSearches = 1U << ctx->params.cParams.searchLog; + const U32 sufficient_len = ctx->params.cParams.targetLength; + const U32 mls = ctx->params.cParams.searchLength; + const U32 minMatch = (ctx->params.cParams.searchLength == 3) ? 3 : 4; + + ZSTD_optimal_t* opt = seqStorePtr->priceTable; + ZSTD_match_t* matches = seqStorePtr->matchTable; + const BYTE* inr; + + /* init */ + U32 offset, rep[ZSTD_REP_NUM]; + { U32 i; for (i=0; irep[i]; } + + ctx->nextToUpdate3 = ctx->nextToUpdate; + ZSTD_rescaleFreqs(seqStorePtr, (const BYTE*)src, srcSize); + ip += (ip==prefixStart); + + /* Match Loop */ + while (ip < ilimit) { + U32 cur, match_num, last_pos, litlen, price; + U32 u, mlen, best_mlen, best_off, litLength; + U32 current = (U32)(ip-base); + memset(opt, 0, sizeof(ZSTD_optimal_t)); + last_pos = 0; + opt[0].litlen = (U32)(ip - anchor); + + /* check repCode */ + { U32 i, last_i = ZSTD_REP_CHECK + (ip==anchor); + for (i = (ip==anchor); i 0 && repCur <= (S32)current) + && (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex>lowestIndex)) /* intentional overflow */ + && (MEM_readMINMATCH(ip, minMatch) == MEM_readMINMATCH(repMatch, minMatch)) ) { + /* repcode detected we should take it */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + mlen = (U32)ZSTD_count_2segments(ip+minMatch, repMatch+minMatch, iend, repEnd, prefixStart) + minMatch; + + if (mlen > sufficient_len || mlen >= ZSTD_OPT_NUM) { + best_mlen = mlen; best_off = i; cur = 0; last_pos = 1; + goto _storeSequence; + } + + best_off = i - (ip==anchor); + litlen = opt[0].litlen; + do { + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); + if (mlen > last_pos || price < opt[mlen].price) + SET_PRICE(mlen, mlen, i, litlen, price); /* note : macro modifies last_pos */ + mlen--; + } while (mlen >= minMatch); + } } } + + match_num = ZSTD_BtGetAllMatches_selectMLS_extDict(ctx, ip, iend, maxSearches, mls, matches, minMatch); /* first search (depth 0) */ + + if (!last_pos && !match_num) { ip++; continue; } + + { U32 i; for (i=0; i sufficient_len || matches[match_num-1].len >= ZSTD_OPT_NUM)) { + best_mlen = matches[match_num-1].len; + best_off = matches[match_num-1].off; + cur = 0; + last_pos = 1; + goto _storeSequence; + } + + best_mlen = (last_pos) ? last_pos : minMatch; + + /* set prices using matches at position = 0 */ + for (u = 0; u < match_num; u++) { + mlen = (u>0) ? matches[u-1].len+1 : best_mlen; + best_mlen = matches[u].len; + litlen = opt[0].litlen; + while (mlen <= best_mlen) { + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); + if (mlen > last_pos || price < opt[mlen].price) + SET_PRICE(mlen, mlen, matches[u].off, litlen, price); + mlen++; + } } + + if (last_pos < minMatch) { + ip++; continue; + } + + /* check further positions */ + for (cur = 1; cur <= last_pos; cur++) { + inr = ip + cur; + + if (opt[cur-1].mlen == 1) { + litlen = opt[cur-1].litlen + 1; + if (cur > litlen) { + price = opt[cur - litlen].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-litlen); + } else + price = ZSTD_getLiteralPrice(seqStorePtr, litlen, anchor); + } else { + litlen = 1; + price = opt[cur - 1].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-1); + } + + if (cur > last_pos || price <= opt[cur].price) + SET_PRICE(cur, 1, 0, litlen, price); + + if (cur == last_pos) break; + + if (inr > ilimit) /* last match must start at a minimum distance of 8 from oend */ + continue; + + mlen = opt[cur].mlen; + if (opt[cur].off > ZSTD_REP_MOVE_OPT) { + opt[cur].rep[2] = opt[cur-mlen].rep[1]; + opt[cur].rep[1] = opt[cur-mlen].rep[0]; + opt[cur].rep[0] = opt[cur].off - ZSTD_REP_MOVE_OPT; + } else { + opt[cur].rep[2] = (opt[cur].off > 1) ? opt[cur-mlen].rep[1] : opt[cur-mlen].rep[2]; + opt[cur].rep[1] = (opt[cur].off > 0) ? opt[cur-mlen].rep[0] : opt[cur-mlen].rep[1]; + opt[cur].rep[0] = ((opt[cur].off==ZSTD_REP_MOVE_OPT) && (mlen != 1)) ? (opt[cur-mlen].rep[0] - 1) : (opt[cur-mlen].rep[opt[cur].off]); + } + + best_mlen = minMatch; + { U32 i, last_i = ZSTD_REP_CHECK + (mlen != 1); + for (i = (mlen != 1); i 0 && repCur <= (S32)(current+cur)) + && (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex>lowestIndex)) /* intentional overflow */ + && (MEM_readMINMATCH(inr, minMatch) == MEM_readMINMATCH(repMatch, minMatch)) ) { + /* repcode detected */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + mlen = (U32)ZSTD_count_2segments(inr+minMatch, repMatch+minMatch, iend, repEnd, prefixStart) + minMatch; + + if (mlen > sufficient_len || cur + mlen >= ZSTD_OPT_NUM) { + best_mlen = mlen; best_off = i; last_pos = cur + 1; + goto _storeSequence; + } + + best_off = i - (opt[cur].mlen != 1); + if (mlen > best_mlen) best_mlen = mlen; + + do { + if (opt[cur].mlen == 1) { + litlen = opt[cur].litlen; + if (cur > litlen) { + price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, inr-litlen, best_off, mlen - MINMATCH, ultra); + } else + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); + } else { + litlen = 0; + price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, best_off, mlen - MINMATCH, ultra); + } + + if (cur + mlen > last_pos || price <= opt[cur + mlen].price) + SET_PRICE(cur + mlen, mlen, i, litlen, price); + mlen--; + } while (mlen >= minMatch); + } } } + + match_num = ZSTD_BtGetAllMatches_selectMLS_extDict(ctx, inr, iend, maxSearches, mls, matches, minMatch); + + if (match_num > 0 && (matches[match_num-1].len > sufficient_len || cur + matches[match_num-1].len >= ZSTD_OPT_NUM)) { + best_mlen = matches[match_num-1].len; + best_off = matches[match_num-1].off; + last_pos = cur + 1; + goto _storeSequence; + } + + /* set prices using matches at position = cur */ + for (u = 0; u < match_num; u++) { + mlen = (u>0) ? matches[u-1].len+1 : best_mlen; + best_mlen = matches[u].len; + + while (mlen <= best_mlen) { + if (opt[cur].mlen == 1) { + litlen = opt[cur].litlen; + if (cur > litlen) + price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, ip+cur-litlen, matches[u].off-1, mlen - MINMATCH, ultra); + else + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); + } else { + litlen = 0; + price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, matches[u].off-1, mlen - MINMATCH, ultra); + } + + if (cur + mlen > last_pos || (price < opt[cur + mlen].price)) + SET_PRICE(cur + mlen, mlen, matches[u].off, litlen, price); + + mlen++; + } } } /* for (cur = 1; cur <= last_pos; cur++) */ + + best_mlen = opt[last_pos].mlen; + best_off = opt[last_pos].off; + cur = last_pos - best_mlen; + + /* store sequence */ +_storeSequence: /* cur, last_pos, best_mlen, best_off have to be set */ + opt[0].mlen = 1; + + while (1) { + mlen = opt[cur].mlen; + offset = opt[cur].off; + opt[cur].mlen = best_mlen; + opt[cur].off = best_off; + best_mlen = mlen; + best_off = offset; + if (mlen > cur) break; + cur -= mlen; + } + + for (u = 0; u <= last_pos; ) { + u += opt[u].mlen; + } + + for (cur=0; cur < last_pos; ) { + mlen = opt[cur].mlen; + if (mlen == 1) { ip++; cur++; continue; } + offset = opt[cur].off; + cur += mlen; + litLength = (U32)(ip - anchor); + + if (offset > ZSTD_REP_MOVE_OPT) { + rep[2] = rep[1]; + rep[1] = rep[0]; + rep[0] = offset - ZSTD_REP_MOVE_OPT; + offset--; + } else { + if (offset != 0) { + best_off = (offset==ZSTD_REP_MOVE_OPT) ? (rep[0] - 1) : (rep[offset]); + if (offset != 1) rep[2] = rep[1]; + rep[1] = rep[0]; + rep[0] = best_off; + } + + if (litLength==0) offset--; + } + + ZSTD_updatePrice(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); + ZSTD_storeSeq(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); + anchor = ip = ip + mlen; + } } /* for (cur=0; cur < last_pos; ) */ + + /* Save reps for next block */ + { int i; for (i=0; irepToConfirm[i] = rep[i]; } + + /* Last Literals */ + { size_t lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } +} + +#endif /* ZSTD_OPT_H_91842398743 */ From 78063ca2bd6964d76654527eb8cf4971e4d93d50 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Wed, 29 Mar 2017 18:47:23 -0700 Subject: [PATCH 095/305] spaces to tabs --- contrib/linux-kernel/spaces_to_tabs.sh | 28 ++++++++++++++++++++++++++ 1 file changed, 28 insertions(+) create mode 100755 contrib/linux-kernel/spaces_to_tabs.sh diff --git a/contrib/linux-kernel/spaces_to_tabs.sh b/contrib/linux-kernel/spaces_to_tabs.sh new file mode 100755 index 000000000..ebde5fbaa --- /dev/null +++ b/contrib/linux-kernel/spaces_to_tabs.sh @@ -0,0 +1,28 @@ +#!/bin/sh +set -e + +# Constants +INCLUDE='include/' +LIB='lib/' +SPACES=' ' +TAB=$'\t' +TMP="replacements.tmp" + +echo "Files: " $INCLUDE* $LIB* + +# Check files for existing tabs +grep "$TAB" $INCLUDE* $LIB* && exit 1 || true +# Replace the first tab on every line +sed -i '' "s/^$SPACES/$TAB/" $INCLUDE* $LIB* + +# Execute once and then execute as long as replacements are happening +more_work="yes" +while [ ! -z "$more_work" ] +do + rm -f $TMP + # Replaces $SPACES that directly follow a $TAB with a $TAB. + # $TMP will be non-empty if any replacements took place. + sed -i '' "s/$TAB$SPACES/$TAB$TAB/w $TMP" $INCLUDE* $LIB* + more_work=$(cat "$TMP") +done +rm -f $TMP From 1075c12078692d628851f70dd2fe41ba8764d0a7 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Wed, 29 Mar 2017 19:57:22 -0700 Subject: [PATCH 096/305] Convert all spaces to tabs --- contrib/linux-kernel/include/zstd.h | 172 +- contrib/linux-kernel/lib/bitstream.h | 258 +- contrib/linux-kernel/lib/entropy_common.c | 288 +- contrib/linux-kernel/lib/error_private.c | 56 +- contrib/linux-kernel/lib/error_private.h | 2 +- contrib/linux-kernel/lib/fse.h | 268 +- contrib/linux-kernel/lib/fse_compress.c | 1046 ++--- contrib/linux-kernel/lib/fse_decompress.c | 302 +- contrib/linux-kernel/lib/huf.h | 62 +- contrib/linux-kernel/lib/huf_compress.c | 860 ++-- contrib/linux-kernel/lib/huf_decompress.c | 1114 ++--- contrib/linux-kernel/lib/mem.h | 212 +- contrib/linux-kernel/lib/xxhash.c | 752 +-- contrib/linux-kernel/lib/xxhash.h | 52 +- contrib/linux-kernel/lib/zstd_common.c | 16 +- contrib/linux-kernel/lib/zstd_compress.c | 4766 ++++++++++---------- contrib/linux-kernel/lib/zstd_decompress.c | 3552 +++++++-------- contrib/linux-kernel/lib/zstd_errors.h | 4 +- contrib/linux-kernel/lib/zstd_internal.h | 156 +- contrib/linux-kernel/lib/zstd_opt.h | 1400 +++--- 20 files changed, 7669 insertions(+), 7669 deletions(-) diff --git a/contrib/linux-kernel/include/zstd.h b/contrib/linux-kernel/include/zstd.h index 4531a84bf..d7f4f5a09 100644 --- a/contrib/linux-kernel/include/zstd.h +++ b/contrib/linux-kernel/include/zstd.h @@ -41,12 +41,12 @@ extern "C" { decompression functions. The library supports compression levels from 1 up to ZSTD_maxCLevel() which is 22. Levels >= 20, labeled `--ultra`, should be used with caution, as they require more memory. Compression can be done in: - - a single step (described as Simple API) - - a single step, reusing a context (described as Explicit memory management) - - unbounded multiple steps (described as Streaming compression) + - a single step (described as Simple API) + - a single step, reusing a context (described as Explicit memory management) + - unbounded multiple steps (described as Streaming compression) The compression ratio achievable on small data can be highly improved using compression with a dictionary in: - - a single step (described as Simple dictionary API) - - a single step, reusing a dictionary (described as Fast dictionary API) + - a single step (described as Simple dictionary API) + - a single step, reusing a dictionary (described as Fast dictionary API) Advanced experimental functions can be accessed using #define ZSTD_STATIC_LINKING_ONLY before including zstd.h. These APIs shall never be used with a dynamic library. @@ -71,22 +71,22 @@ ZSTDLIB_API unsigned ZSTD_versionNumber(void); /**< library version number; to * Simple API ***************************************/ /*! ZSTD_compress() : - Compresses `src` content as a single zstd compressed frame into already allocated `dst`. - Hint : compression runs faster if `dstCapacity` >= `ZSTD_compressBound(srcSize)`. - @return : compressed size written into `dst` (<= `dstCapacity), - or an error code if it fails (which can be tested using ZSTD_isError()). */ + Compresses `src` content as a single zstd compressed frame into already allocated `dst`. + Hint : compression runs faster if `dstCapacity` >= `ZSTD_compressBound(srcSize)`. + @return : compressed size written into `dst` (<= `dstCapacity), + or an error code if it fails (which can be tested using ZSTD_isError()). */ ZSTDLIB_API size_t ZSTD_compress( void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - int compressionLevel); + const void* src, size_t srcSize, + int compressionLevel); /*! ZSTD_decompress() : - `compressedSize` : must be the _exact_ size of some number of compressed and/or skippable frames. - `dstCapacity` is an upper bound of originalSize. - If user cannot imply a maximum upper bound, it's better to use streaming mode to decompress data. - @return : the number of bytes decompressed into `dst` (<= `dstCapacity`), - or an errorCode if it fails (which can be tested using ZSTD_isError()). */ + `compressedSize` : must be the _exact_ size of some number of compressed and/or skippable frames. + `dstCapacity` is an upper bound of originalSize. + If user cannot imply a maximum upper bound, it's better to use streaming mode to decompress data. + @return : the number of bytes decompressed into `dst` (<= `dstCapacity`), + or an errorCode if it fails (which can be tested using ZSTD_isError()). */ ZSTDLIB_API size_t ZSTD_decompress( void* dst, size_t dstCapacity, - const void* src, size_t compressedSize); + const void* src, size_t compressedSize); /*! ZSTD_getDecompressedSize() : * NOTE: This function is planned to be obsolete, in favour of ZSTD_getFrameContentSize. @@ -136,7 +136,7 @@ ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx(void); ZSTDLIB_API size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx); /*! ZSTD_compressCCtx() : - Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). */ + Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). */ ZSTDLIB_API size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel); /*= Decompression context @@ -161,10 +161,10 @@ ZSTDLIB_API size_t ZSTD_decompressDCtx(ZSTD_DCtx* ctx, void* dst, size_t dstCapa * Note : This function loads the dictionary, resulting in significant startup delay. * Note : When `dict == NULL || dictSize < 8` no dictionary is used. */ ZSTDLIB_API size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict,size_t dictSize, - int compressionLevel); + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict,size_t dictSize, + int compressionLevel); /*! ZSTD_decompress_usingDict() : * Decompression using a predefined Dictionary (see dictBuilder/zdict.h). @@ -172,9 +172,9 @@ ZSTDLIB_API size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, * Note : This function loads the dictionary, resulting in significant startup delay. * Note : When `dict == NULL || dictSize < 8` no dictionary is used. */ ZSTDLIB_API size_t ZSTD_decompress_usingDict(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict,size_t dictSize); + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict,size_t dictSize); /**************************** @@ -198,9 +198,9 @@ ZSTDLIB_API size_t ZSTD_freeCDict(ZSTD_CDict* CDict); * Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. * Note that compression level is decided during dictionary creation. */ ZSTDLIB_API size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const ZSTD_CDict* cdict); + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const ZSTD_CDict* cdict); typedef struct ZSTD_DDict_s ZSTD_DDict; @@ -218,9 +218,9 @@ ZSTDLIB_API size_t ZSTD_freeDDict(ZSTD_DDict* ddict); * Decompression using a digested Dictionary. * Faster startup than ZSTD_decompress_usingDict(), recommended when same dictionary is used multiple times. */ ZSTDLIB_API size_t ZSTD_decompress_usingDDict(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const ZSTD_DDict* ddict); + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const ZSTD_DDict* ddict); /**************************** @@ -379,24 +379,24 @@ static const size_t ZSTD_skippableHeaderSize = 8; /* magic number + skippable f typedef enum { ZSTD_fast, ZSTD_dfast, ZSTD_greedy, ZSTD_lazy, ZSTD_lazy2, ZSTD_btlazy2, ZSTD_btopt, ZSTD_btopt2 } ZSTD_strategy; /* from faster to stronger */ typedef struct { - unsigned windowLog; /**< largest match distance : larger == more compression, more memory needed during decompression */ - unsigned chainLog; /**< fully searched segment : larger == more compression, slower, more memory (useless for fast) */ - unsigned hashLog; /**< dispatch table : larger == faster, more memory */ - unsigned searchLog; /**< nb of searches : larger == more compression, slower */ - unsigned searchLength; /**< match length searched : larger == faster decompression, sometimes less compression */ - unsigned targetLength; /**< acceptable match size for optimal parser (only) : larger == more compression, slower */ - ZSTD_strategy strategy; + unsigned windowLog; /**< largest match distance : larger == more compression, more memory needed during decompression */ + unsigned chainLog; /**< fully searched segment : larger == more compression, slower, more memory (useless for fast) */ + unsigned hashLog; /**< dispatch table : larger == faster, more memory */ + unsigned searchLog; /**< nb of searches : larger == more compression, slower */ + unsigned searchLength; /**< match length searched : larger == faster decompression, sometimes less compression */ + unsigned targetLength; /**< acceptable match size for optimal parser (only) : larger == more compression, slower */ + ZSTD_strategy strategy; } ZSTD_compressionParameters; typedef struct { - unsigned contentSizeFlag; /**< 1: content size will be in frame header (when known) */ - unsigned checksumFlag; /**< 1: generate a 32-bits checksum at end of frame, for error detection */ - unsigned noDictIDFlag; /**< 1: no dictID will be saved into frame header (if dictionary compression) */ + unsigned contentSizeFlag; /**< 1: content size will be in frame header (when known) */ + unsigned checksumFlag; /**< 1: generate a 32-bits checksum at end of frame, for error detection */ + unsigned noDictIDFlag; /**< 1: no dictID will be saved into frame header (if dictionary compression) */ } ZSTD_frameParameters; typedef struct { - ZSTD_compressionParameters cParams; - ZSTD_frameParameters fParams; + ZSTD_compressionParameters cParams; + ZSTD_frameParameters fParams; } ZSTD_parameters; /*= Custom memory allocation functions */ @@ -470,8 +470,8 @@ ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx_advanced(ZSTD_customMem customMem); ZSTDLIB_API size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx); typedef enum { - ZSTD_p_forceWindow, /* Force back-references to remain < windowSize, even when referencing Dictionary content (default:0) */ - ZSTD_p_forceRawDict /* Force loading dictionary in "content-only" mode (no header analysis) */ + ZSTD_p_forceWindow, /* Force back-references to remain < windowSize, even when referencing Dictionary content (default:0) */ + ZSTD_p_forceRawDict /* Force loading dictionary in "content-only" mode (no header analysis) */ } ZSTD_CCtxParameter; /*! ZSTD_setCCtxParameter() : * Set advanced parameters, selected through enum ZSTD_CCtxParameter @@ -487,7 +487,7 @@ ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_byReference(const void* dictBuffer, siz /*! ZSTD_createCDict_advanced() : * Create a ZSTD_CDict using external alloc and free, and customized compression parameters */ ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_advanced(const void* dict, size_t dictSize, unsigned byReference, - ZSTD_parameters params, ZSTD_customMem customMem); + ZSTD_parameters params, ZSTD_customMem customMem); /*! ZSTD_sizeof_CDict() : * Gives the amount of memory used by a given ZSTD_sizeof_CDict */ @@ -515,10 +515,10 @@ ZSTDLIB_API ZSTD_compressionParameters ZSTD_adjustCParams(ZSTD_compressionParame /*! ZSTD_compress_advanced() : * Same as ZSTD_compress_usingDict(), with fine-tune control of each compression parameter */ ZSTDLIB_API size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict,size_t dictSize, - ZSTD_parameters params); + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict,size_t dictSize, + ZSTD_parameters params); /*--- Advanced decompression functions ---*/ @@ -551,7 +551,7 @@ ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_byReference(const void* dictBuffer, siz /*! ZSTD_createDDict_advanced() : * Create a ZSTD_DDict using external alloc and free, optionally by reference */ ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize, - unsigned byReference, ZSTD_customMem customMem); + unsigned byReference, ZSTD_customMem customMem); /*! ZSTD_sizeof_DDict() : * Gives the amount of memory used by a given ZSTD_DDict */ @@ -591,7 +591,7 @@ ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem); ZSTDLIB_API size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize); /**< pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ ZSTDLIB_API size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */ ZSTDLIB_API size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize, - ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */ + ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */ ZSTDLIB_API size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict); /**< note : cdict will just be referenced, and must outlive compression session */ ZSTDLIB_API size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize); /**< re-use compression parameters from previous init; skip dictionary loading stage; zcs must be init at least once before. note: pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs); @@ -632,12 +632,12 @@ ZSTDLIB_API size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds); - ZSTD_compressContinue() has no internal buffer. It uses externally provided buffer only. - Interface is synchronous : input is consumed entirely and produce 1+ (or more) compressed blocks. - Caller must ensure there is enough space in `dst` to store compressed data under worst case scenario. - Worst case evaluation is provided by ZSTD_compressBound(). - ZSTD_compressContinue() doesn't guarantee recover after a failed compression. + Worst case evaluation is provided by ZSTD_compressBound(). + ZSTD_compressContinue() doesn't guarantee recover after a failed compression. - ZSTD_compressContinue() presumes prior input ***is still accessible and unmodified*** (up to maximum distance size, see WindowLog). - It remembers all previous contiguous blocks, plus one separated memory segment (which can itself consists of multiple contiguous blocks) + It remembers all previous contiguous blocks, plus one separated memory segment (which can itself consists of multiple contiguous blocks) - ZSTD_compressContinue() detects that prior input has been overwritten when `src` buffer overlaps. - In which case, it will "discard" the relevant memory section from its history. + In which case, it will "discard" the relevant memory section from its history. Finish a frame with ZSTD_compressEnd(), which will write the last block(s) and optional checksum. It's possible to use srcSize==0, in which case, it will write a final empty block to end the frame. @@ -675,8 +675,8 @@ ZSTDLIB_API size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapaci Frame parameters are extracted from the beginning of the compressed frame. Data fragment must be large enough to ensure successful decoding, typically `ZSTD_frameHeaderSize_max` bytes. @result : 0 : successful decoding, the `ZSTD_frameParams` structure is correctly filled. - >0 : `srcSize` is too small, please provide at least @result bytes on next attempt. - errorCode, which can be tested using ZSTD_isError(). + >0 : `srcSize` is too small, please provide at least @result bytes on next attempt. + errorCode, which can be tested using ZSTD_isError(). Start decompression, with ZSTD_decompressBegin() or ZSTD_decompressBegin_usingDict(). Alternatively, you can copy a prepared context, using ZSTD_copyDCtx(). @@ -711,17 +711,17 @@ ZSTDLIB_API size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapaci c) Frame Content - any content (User Data) of length equal to Frame Size For skippable frames ZSTD_decompressContinue() always returns 0. For skippable frames ZSTD_getFrameParams() returns fparamsPtr->windowLog==0 what means that a frame is skippable. - Note : If fparamsPtr->frameContentSize==0, it is ambiguous: the frame might actually be a Zstd encoded frame with no content. - For purposes of decompression, it is valid in both cases to skip the frame using - ZSTD_findFrameCompressedSize to find its size in bytes. + Note : If fparamsPtr->frameContentSize==0, it is ambiguous: the frame might actually be a Zstd encoded frame with no content. + For purposes of decompression, it is valid in both cases to skip the frame using + ZSTD_findFrameCompressedSize to find its size in bytes. It also returns Frame Size as fparamsPtr->frameContentSize. */ typedef struct { - unsigned long long frameContentSize; - unsigned windowSize; - unsigned dictID; - unsigned checksumFlag; + unsigned long long frameContentSize; + unsigned windowSize; + unsigned dictID; + unsigned checksumFlag; } ZSTD_frameParams; /*===== Buffer-less streaming decompression functions =====*/ @@ -735,29 +735,29 @@ typedef enum { ZSTDnit_frameHeader, ZSTDnit_blockHeader, ZSTDnit_block, ZSTDnit_ ZSTDLIB_API ZSTD_nextInputType_e ZSTD_nextInputType(ZSTD_DCtx* dctx); /** - Block functions + Block functions - Block functions produce and decode raw zstd blocks, without frame metadata. - Frame metadata cost is typically ~18 bytes, which can be non-negligible for very small blocks (< 100 bytes). - User will have to take in charge required information to regenerate data, such as compressed and content sizes. + Block functions produce and decode raw zstd blocks, without frame metadata. + Frame metadata cost is typically ~18 bytes, which can be non-negligible for very small blocks (< 100 bytes). + User will have to take in charge required information to regenerate data, such as compressed and content sizes. - A few rules to respect : - - Compressing and decompressing require a context structure - + Use ZSTD_createCCtx() and ZSTD_createDCtx() - - It is necessary to init context before starting - + compression : ZSTD_compressBegin() - + decompression : ZSTD_decompressBegin() - + variants _usingDict() are also allowed - + copyCCtx() and copyDCtx() work too - - Block size is limited, it must be <= ZSTD_getBlockSizeMax() - + If you need to compress more, cut data into multiple blocks - + Consider using the regular ZSTD_compress() instead, as frame metadata costs become negligible when source size is large. - - When a block is considered not compressible enough, ZSTD_compressBlock() result will be zero. - In which case, nothing is produced into `dst`. - + User must test for such outcome and deal directly with uncompressed data - + ZSTD_decompressBlock() doesn't accept uncompressed data as input !!! - + In case of multiple successive blocks, decoder must be informed of uncompressed block existence to follow proper history. - Use ZSTD_insertBlock() in such a case. + A few rules to respect : + - Compressing and decompressing require a context structure + + Use ZSTD_createCCtx() and ZSTD_createDCtx() + - It is necessary to init context before starting + + compression : ZSTD_compressBegin() + + decompression : ZSTD_decompressBegin() + + variants _usingDict() are also allowed + + copyCCtx() and copyDCtx() work too + - Block size is limited, it must be <= ZSTD_getBlockSizeMax() + + If you need to compress more, cut data into multiple blocks + + Consider using the regular ZSTD_compress() instead, as frame metadata costs become negligible when source size is large. + - When a block is considered not compressible enough, ZSTD_compressBlock() result will be zero. + In which case, nothing is produced into `dst`. + + User must test for such outcome and deal directly with uncompressed data + + ZSTD_decompressBlock() doesn't accept uncompressed data as input !!! + + In case of multiple successive blocks, decoder must be informed of uncompressed block existence to follow proper history. + Use ZSTD_insertBlock() in such a case. */ #define ZSTD_BLOCKSIZE_ABSOLUTEMAX (128 * 1024) /* define, for static allocation */ diff --git a/contrib/linux-kernel/lib/bitstream.h b/contrib/linux-kernel/lib/bitstream.h index 0e3d2fc55..546f6582b 100644 --- a/contrib/linux-kernel/lib/bitstream.h +++ b/contrib/linux-kernel/lib/bitstream.h @@ -10,9 +10,9 @@ modification, are permitted provided that the following conditions are met: - * Redistributions of source code must retain the above copyright + * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - * Redistributions in binary form must reproduce the above + * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. @@ -73,11 +73,11 @@ extern "C" { */ typedef struct { - size_t bitContainer; - int bitPos; - char* startPtr; - char* ptr; - char* endPtr; + size_t bitContainer; + int bitPos; + char* startPtr; + char* ptr; + char* endPtr; } BIT_CStream_t; MEM_STATIC size_t BIT_initCStream(BIT_CStream_t* bitC, void* dstBuffer, size_t dstCapacity); @@ -108,17 +108,17 @@ MEM_STATIC size_t BIT_closeCStream(BIT_CStream_t* bitC); **********************************************/ typedef struct { - size_t bitContainer; - unsigned bitsConsumed; - const char* ptr; - const char* start; + size_t bitContainer; + unsigned bitsConsumed; + const char* ptr; + const char* start; } BIT_DStream_t; typedef enum { BIT_DStream_unfinished = 0, - BIT_DStream_endOfBuffer = 1, - BIT_DStream_completed = 2, - BIT_DStream_overflow = 3 } BIT_DStream_status; /* result of BIT_reloadDStream() */ - /* 1,2,4,8 would be better for bitmap combinations, but slows down performance a bit ... :( */ + BIT_DStream_endOfBuffer = 1, + BIT_DStream_completed = 2, + BIT_DStream_overflow = 3 } BIT_DStream_status; /* result of BIT_reloadDStream() */ + /* 1,2,4,8 would be better for bitmap combinations, but slows down performance a bit ... :( */ MEM_STATIC size_t BIT_initDStream(BIT_DStream_t* bitD, const void* srcBuffer, size_t srcSize); MEM_STATIC size_t BIT_readBits(BIT_DStream_t* bitD, unsigned nbBits); @@ -157,20 +157,20 @@ MEM_STATIC size_t BIT_readBitsFast(BIT_DStream_t* bitD, unsigned nbBits); MEM_STATIC unsigned BIT_highbit32 (register U32 val) { # if defined(_MSC_VER) /* Visual */ - unsigned long r=0; - _BitScanReverse ( &r, val ); - return (unsigned) r; + unsigned long r=0; + _BitScanReverse ( &r, val ); + return (unsigned) r; # elif defined(__GNUC__) && (__GNUC__ >= 3) /* Use GCC Intrinsic */ - return 31 - __builtin_clz (val); + return 31 - __builtin_clz (val); # else /* Software version */ - static const unsigned DeBruijnClz[32] = { 0, 9, 1, 10, 13, 21, 2, 29, 11, 14, 16, 18, 22, 25, 3, 30, 8, 12, 20, 28, 15, 17, 24, 7, 19, 27, 23, 6, 26, 5, 4, 31 }; - U32 v = val; - v |= v >> 1; - v |= v >> 2; - v |= v >> 4; - v |= v >> 8; - v |= v >> 16; - return DeBruijnClz[ (U32) (v * 0x07C4ACDDU) >> 27]; + static const unsigned DeBruijnClz[32] = { 0, 9, 1, 10, 13, 21, 2, 29, 11, 14, 16, 18, 22, 25, 3, 30, 8, 12, 20, 28, 15, 17, 24, 7, 19, 27, 23, 6, 26, 5, 4, 31 }; + U32 v = val; + v |= v >> 1; + v |= v >> 2; + v |= v >> 4; + v |= v >> 8; + v |= v >> 16; + return DeBruijnClz[ (U32) (v * 0x07C4ACDDU) >> 27]; # endif } @@ -184,44 +184,44 @@ static const unsigned BIT_mask[] = { 0, 1, 3, 7, 0xF, 0x1F, 0x3F, 0x7F, 0xFF, 0x /*! BIT_initCStream() : * `dstCapacity` must be > sizeof(void*) * @return : 0 if success, - otherwise an error code (can be tested using ERR_isError() ) */ + otherwise an error code (can be tested using ERR_isError() ) */ MEM_STATIC size_t BIT_initCStream(BIT_CStream_t* bitC, void* startPtr, size_t dstCapacity) { - bitC->bitContainer = 0; - bitC->bitPos = 0; - bitC->startPtr = (char*)startPtr; - bitC->ptr = bitC->startPtr; - bitC->endPtr = bitC->startPtr + dstCapacity - sizeof(bitC->ptr); - if (dstCapacity <= sizeof(bitC->ptr)) return ERROR(dstSize_tooSmall); - return 0; + bitC->bitContainer = 0; + bitC->bitPos = 0; + bitC->startPtr = (char*)startPtr; + bitC->ptr = bitC->startPtr; + bitC->endPtr = bitC->startPtr + dstCapacity - sizeof(bitC->ptr); + if (dstCapacity <= sizeof(bitC->ptr)) return ERROR(dstSize_tooSmall); + return 0; } /*! BIT_addBits() : - can add up to 26 bits into `bitC`. - Does not check for register overflow ! */ + can add up to 26 bits into `bitC`. + Does not check for register overflow ! */ MEM_STATIC void BIT_addBits(BIT_CStream_t* bitC, size_t value, unsigned nbBits) { - bitC->bitContainer |= (value & BIT_mask[nbBits]) << bitC->bitPos; - bitC->bitPos += nbBits; + bitC->bitContainer |= (value & BIT_mask[nbBits]) << bitC->bitPos; + bitC->bitPos += nbBits; } /*! BIT_addBitsFast() : * works only if `value` is _clean_, meaning all high bits above nbBits are 0 */ MEM_STATIC void BIT_addBitsFast(BIT_CStream_t* bitC, size_t value, unsigned nbBits) { - bitC->bitContainer |= value << bitC->bitPos; - bitC->bitPos += nbBits; + bitC->bitContainer |= value << bitC->bitPos; + bitC->bitPos += nbBits; } /*! BIT_flushBitsFast() : * unsafe version; does not check buffer overflow */ MEM_STATIC void BIT_flushBitsFast(BIT_CStream_t* bitC) { - size_t const nbBytes = bitC->bitPos >> 3; - MEM_writeLEST(bitC->ptr, bitC->bitContainer); - bitC->ptr += nbBytes; - bitC->bitPos &= 7; - bitC->bitContainer >>= nbBytes*8; /* if bitPos >= sizeof(bitContainer)*8 --> undefined behavior */ + size_t const nbBytes = bitC->bitPos >> 3; + MEM_writeLEST(bitC->ptr, bitC->bitContainer); + bitC->ptr += nbBytes; + bitC->bitPos &= 7; + bitC->bitContainer >>= nbBytes*8; /* if bitPos >= sizeof(bitContainer)*8 --> undefined behavior */ } /*! BIT_flushBits() : @@ -229,25 +229,25 @@ MEM_STATIC void BIT_flushBitsFast(BIT_CStream_t* bitC) * note : does not signal buffer overflow. This will be revealed later on using BIT_closeCStream() */ MEM_STATIC void BIT_flushBits(BIT_CStream_t* bitC) { - size_t const nbBytes = bitC->bitPos >> 3; - MEM_writeLEST(bitC->ptr, bitC->bitContainer); - bitC->ptr += nbBytes; - if (bitC->ptr > bitC->endPtr) bitC->ptr = bitC->endPtr; - bitC->bitPos &= 7; - bitC->bitContainer >>= nbBytes*8; /* if bitPos >= sizeof(bitContainer)*8 --> undefined behavior */ + size_t const nbBytes = bitC->bitPos >> 3; + MEM_writeLEST(bitC->ptr, bitC->bitContainer); + bitC->ptr += nbBytes; + if (bitC->ptr > bitC->endPtr) bitC->ptr = bitC->endPtr; + bitC->bitPos &= 7; + bitC->bitContainer >>= nbBytes*8; /* if bitPos >= sizeof(bitContainer)*8 --> undefined behavior */ } /*! BIT_closeCStream() : * @return : size of CStream, in bytes, - or 0 if it could not fit into dstBuffer */ + or 0 if it could not fit into dstBuffer */ MEM_STATIC size_t BIT_closeCStream(BIT_CStream_t* bitC) { - BIT_addBitsFast(bitC, 1, 1); /* endMark */ - BIT_flushBits(bitC); + BIT_addBitsFast(bitC, 1, 1); /* endMark */ + BIT_flushBits(bitC); - if (bitC->ptr >= bitC->endPtr) return 0; /* doesn't fit within authorized budget : cancel */ + if (bitC->ptr >= bitC->endPtr) return 0; /* doesn't fit within authorized budget : cancel */ - return (bitC->ptr - bitC->startPtr) + (bitC->bitPos > 0); + return (bitC->ptr - bitC->startPtr) + (bitC->bitPos > 0); } @@ -262,60 +262,60 @@ MEM_STATIC size_t BIT_closeCStream(BIT_CStream_t* bitC) */ MEM_STATIC size_t BIT_initDStream(BIT_DStream_t* bitD, const void* srcBuffer, size_t srcSize) { - if (srcSize < 1) { memset(bitD, 0, sizeof(*bitD)); return ERROR(srcSize_wrong); } + if (srcSize < 1) { memset(bitD, 0, sizeof(*bitD)); return ERROR(srcSize_wrong); } - if (srcSize >= sizeof(bitD->bitContainer)) { /* normal case */ - bitD->start = (const char*)srcBuffer; - bitD->ptr = (const char*)srcBuffer + srcSize - sizeof(bitD->bitContainer); - bitD->bitContainer = MEM_readLEST(bitD->ptr); - { BYTE const lastByte = ((const BYTE*)srcBuffer)[srcSize-1]; - bitD->bitsConsumed = lastByte ? 8 - BIT_highbit32(lastByte) : 0; /* ensures bitsConsumed is always set */ - if (lastByte == 0) return ERROR(GENERIC); /* endMark not present */ } - } else { - bitD->start = (const char*)srcBuffer; - bitD->ptr = bitD->start; - bitD->bitContainer = *(const BYTE*)(bitD->start); - switch(srcSize) - { - case 7: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[6]) << (sizeof(bitD->bitContainer)*8 - 16); - case 6: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[5]) << (sizeof(bitD->bitContainer)*8 - 24); - case 5: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[4]) << (sizeof(bitD->bitContainer)*8 - 32); - case 4: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[3]) << 24; - case 3: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[2]) << 16; - case 2: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[1]) << 8; - default:; - } - { BYTE const lastByte = ((const BYTE*)srcBuffer)[srcSize-1]; - bitD->bitsConsumed = lastByte ? 8 - BIT_highbit32(lastByte) : 0; - if (lastByte == 0) return ERROR(GENERIC); /* endMark not present */ } - bitD->bitsConsumed += (U32)(sizeof(bitD->bitContainer) - srcSize)*8; - } + if (srcSize >= sizeof(bitD->bitContainer)) { /* normal case */ + bitD->start = (const char*)srcBuffer; + bitD->ptr = (const char*)srcBuffer + srcSize - sizeof(bitD->bitContainer); + bitD->bitContainer = MEM_readLEST(bitD->ptr); + { BYTE const lastByte = ((const BYTE*)srcBuffer)[srcSize-1]; + bitD->bitsConsumed = lastByte ? 8 - BIT_highbit32(lastByte) : 0; /* ensures bitsConsumed is always set */ + if (lastByte == 0) return ERROR(GENERIC); /* endMark not present */ } + } else { + bitD->start = (const char*)srcBuffer; + bitD->ptr = bitD->start; + bitD->bitContainer = *(const BYTE*)(bitD->start); + switch(srcSize) + { + case 7: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[6]) << (sizeof(bitD->bitContainer)*8 - 16); + case 6: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[5]) << (sizeof(bitD->bitContainer)*8 - 24); + case 5: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[4]) << (sizeof(bitD->bitContainer)*8 - 32); + case 4: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[3]) << 24; + case 3: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[2]) << 16; + case 2: bitD->bitContainer += (size_t)(((const BYTE*)(srcBuffer))[1]) << 8; + default:; + } + { BYTE const lastByte = ((const BYTE*)srcBuffer)[srcSize-1]; + bitD->bitsConsumed = lastByte ? 8 - BIT_highbit32(lastByte) : 0; + if (lastByte == 0) return ERROR(GENERIC); /* endMark not present */ } + bitD->bitsConsumed += (U32)(sizeof(bitD->bitContainer) - srcSize)*8; + } - return srcSize; + return srcSize; } MEM_STATIC size_t BIT_getUpperBits(size_t bitContainer, U32 const start) { - return bitContainer >> start; + return bitContainer >> start; } MEM_STATIC size_t BIT_getMiddleBits(size_t bitContainer, U32 const start, U32 const nbBits) { #if defined(__BMI__) && defined(__GNUC__) && __GNUC__*1000+__GNUC_MINOR__ >= 4008 /* experimental */ # if defined(__x86_64__) - if (sizeof(bitContainer)==8) - return _bextr_u64(bitContainer, start, nbBits); - else + if (sizeof(bitContainer)==8) + return _bextr_u64(bitContainer, start, nbBits); + else # endif - return _bextr_u32(bitContainer, start, nbBits); + return _bextr_u32(bitContainer, start, nbBits); #else - return (bitContainer >> start) & BIT_mask[nbBits]; + return (bitContainer >> start) & BIT_mask[nbBits]; #endif } MEM_STATIC size_t BIT_getLowerBits(size_t bitContainer, U32 const nbBits) { - return bitContainer & BIT_mask[nbBits]; + return bitContainer & BIT_mask[nbBits]; } /*! BIT_lookBits() : @@ -328,10 +328,10 @@ MEM_STATIC size_t BIT_getLowerBits(size_t bitContainer, U32 const nbBits) MEM_STATIC size_t BIT_lookBits(const BIT_DStream_t* bitD, U32 nbBits) { #if defined(__BMI__) && defined(__GNUC__) /* experimental; fails if bitD->bitsConsumed + nbBits > sizeof(bitD->bitContainer)*8 */ - return BIT_getMiddleBits(bitD->bitContainer, (sizeof(bitD->bitContainer)*8) - bitD->bitsConsumed - nbBits, nbBits); + return BIT_getMiddleBits(bitD->bitContainer, (sizeof(bitD->bitContainer)*8) - bitD->bitsConsumed - nbBits, nbBits); #else - U32 const bitMask = sizeof(bitD->bitContainer)*8 - 1; - return ((bitD->bitContainer << (bitD->bitsConsumed & bitMask)) >> 1) >> ((bitMask-nbBits) & bitMask); + U32 const bitMask = sizeof(bitD->bitContainer)*8 - 1; + return ((bitD->bitContainer << (bitD->bitsConsumed & bitMask)) >> 1) >> ((bitMask-nbBits) & bitMask); #endif } @@ -339,13 +339,13 @@ MEM_STATIC size_t BIT_getLowerBits(size_t bitContainer, U32 const nbBits) * unsafe version; only works only if nbBits >= 1 */ MEM_STATIC size_t BIT_lookBitsFast(const BIT_DStream_t* bitD, U32 nbBits) { - U32 const bitMask = sizeof(bitD->bitContainer)*8 - 1; - return (bitD->bitContainer << (bitD->bitsConsumed & bitMask)) >> (((bitMask+1)-nbBits) & bitMask); + U32 const bitMask = sizeof(bitD->bitContainer)*8 - 1; + return (bitD->bitContainer << (bitD->bitsConsumed & bitMask)) >> (((bitMask+1)-nbBits) & bitMask); } MEM_STATIC void BIT_skipBits(BIT_DStream_t* bitD, U32 nbBits) { - bitD->bitsConsumed += nbBits; + bitD->bitsConsumed += nbBits; } /*! BIT_readBits() : @@ -355,51 +355,51 @@ MEM_STATIC void BIT_skipBits(BIT_DStream_t* bitD, U32 nbBits) */ MEM_STATIC size_t BIT_readBits(BIT_DStream_t* bitD, U32 nbBits) { - size_t const value = BIT_lookBits(bitD, nbBits); - BIT_skipBits(bitD, nbBits); - return value; + size_t const value = BIT_lookBits(bitD, nbBits); + BIT_skipBits(bitD, nbBits); + return value; } /*! BIT_readBitsFast() : * unsafe version; only works only if nbBits >= 1 */ MEM_STATIC size_t BIT_readBitsFast(BIT_DStream_t* bitD, U32 nbBits) { - size_t const value = BIT_lookBitsFast(bitD, nbBits); - BIT_skipBits(bitD, nbBits); - return value; + size_t const value = BIT_lookBitsFast(bitD, nbBits); + BIT_skipBits(bitD, nbBits); + return value; } /*! BIT_reloadDStream() : * Refill `bitD` from buffer previously set in BIT_initDStream() . * This function is safe, it guarantees it will not read beyond src buffer. * @return : status of `BIT_DStream_t` internal register. - if status == BIT_DStream_unfinished, internal register is filled with >= (sizeof(bitD->bitContainer)*8 - 7) bits */ + if status == BIT_DStream_unfinished, internal register is filled with >= (sizeof(bitD->bitContainer)*8 - 7) bits */ MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD) { - if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should not happen => corruption detected */ - return BIT_DStream_overflow; + if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8)) /* should not happen => corruption detected */ + return BIT_DStream_overflow; - if (bitD->ptr >= bitD->start + sizeof(bitD->bitContainer)) { - bitD->ptr -= bitD->bitsConsumed >> 3; - bitD->bitsConsumed &= 7; - bitD->bitContainer = MEM_readLEST(bitD->ptr); - return BIT_DStream_unfinished; - } - if (bitD->ptr == bitD->start) { - if (bitD->bitsConsumed < sizeof(bitD->bitContainer)*8) return BIT_DStream_endOfBuffer; - return BIT_DStream_completed; - } - { U32 nbBytes = bitD->bitsConsumed >> 3; - BIT_DStream_status result = BIT_DStream_unfinished; - if (bitD->ptr - nbBytes < bitD->start) { - nbBytes = (U32)(bitD->ptr - bitD->start); /* ptr > start */ - result = BIT_DStream_endOfBuffer; - } - bitD->ptr -= nbBytes; - bitD->bitsConsumed -= nbBytes*8; - bitD->bitContainer = MEM_readLEST(bitD->ptr); /* reminder : srcSize > sizeof(bitD) */ - return result; - } + if (bitD->ptr >= bitD->start + sizeof(bitD->bitContainer)) { + bitD->ptr -= bitD->bitsConsumed >> 3; + bitD->bitsConsumed &= 7; + bitD->bitContainer = MEM_readLEST(bitD->ptr); + return BIT_DStream_unfinished; + } + if (bitD->ptr == bitD->start) { + if (bitD->bitsConsumed < sizeof(bitD->bitContainer)*8) return BIT_DStream_endOfBuffer; + return BIT_DStream_completed; + } + { U32 nbBytes = bitD->bitsConsumed >> 3; + BIT_DStream_status result = BIT_DStream_unfinished; + if (bitD->ptr - nbBytes < bitD->start) { + nbBytes = (U32)(bitD->ptr - bitD->start); /* ptr > start */ + result = BIT_DStream_endOfBuffer; + } + bitD->ptr -= nbBytes; + bitD->bitsConsumed -= nbBytes*8; + bitD->bitContainer = MEM_readLEST(bitD->ptr); /* reminder : srcSize > sizeof(bitD) */ + return result; + } } /*! BIT_endOfDStream() : @@ -407,7 +407,7 @@ MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD) */ MEM_STATIC unsigned BIT_endOfDStream(const BIT_DStream_t* DStream) { - return ((DStream->ptr == DStream->start) && (DStream->bitsConsumed == sizeof(DStream->bitContainer)*8)); + return ((DStream->ptr == DStream->start) && (DStream->bitsConsumed == sizeof(DStream->bitContainer)*8)); } #if defined (__cplusplus) diff --git a/contrib/linux-kernel/lib/entropy_common.c b/contrib/linux-kernel/lib/entropy_common.c index b37a082fe..c9d489bc7 100644 --- a/contrib/linux-kernel/lib/entropy_common.c +++ b/contrib/linux-kernel/lib/entropy_common.c @@ -8,9 +8,9 @@ modification, are permitted provided that the following conditions are met: - * Redistributions of source code must retain the above copyright + * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - * Redistributions in binary form must reproduce the above + * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. @@ -27,9 +27,9 @@ (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - You can contact the author at : - - FSE+HUF source repository : https://github.com/Cyan4973/FiniteStateEntropy - - Public forum : https://groups.google.com/forum/#!forum/lz4c + You can contact the author at : + - FSE+HUF source repository : https://github.com/Cyan4973/FiniteStateEntropy + - Public forum : https://groups.google.com/forum/#!forum/lz4c *************************************************************************** */ /* ************************************* @@ -59,163 +59,163 @@ const char* HUF_getErrorName(size_t code) { return ERR_getErrorName(code); } * FSE NCount encoding-decoding ****************************************************************/ size_t FSE_readNCount (short* normalizedCounter, unsigned* maxSVPtr, unsigned* tableLogPtr, - const void* headerBuffer, size_t hbSize) + const void* headerBuffer, size_t hbSize) { - const BYTE* const istart = (const BYTE*) headerBuffer; - const BYTE* const iend = istart + hbSize; - const BYTE* ip = istart; - int nbBits; - int remaining; - int threshold; - U32 bitStream; - int bitCount; - unsigned charnum = 0; - int previous0 = 0; + const BYTE* const istart = (const BYTE*) headerBuffer; + const BYTE* const iend = istart + hbSize; + const BYTE* ip = istart; + int nbBits; + int remaining; + int threshold; + U32 bitStream; + int bitCount; + unsigned charnum = 0; + int previous0 = 0; - if (hbSize < 4) return ERROR(srcSize_wrong); - bitStream = MEM_readLE32(ip); - nbBits = (bitStream & 0xF) + FSE_MIN_TABLELOG; /* extract tableLog */ - if (nbBits > FSE_TABLELOG_ABSOLUTE_MAX) return ERROR(tableLog_tooLarge); - bitStream >>= 4; - bitCount = 4; - *tableLogPtr = nbBits; - remaining = (1< FSE_TABLELOG_ABSOLUTE_MAX) return ERROR(tableLog_tooLarge); + bitStream >>= 4; + bitCount = 4; + *tableLogPtr = nbBits; + remaining = (1<1) & (charnum<=*maxSVPtr)) { - if (previous0) { - unsigned n0 = charnum; - while ((bitStream & 0xFFFF) == 0xFFFF) { - n0 += 24; - if (ip < iend-5) { - ip += 2; - bitStream = MEM_readLE32(ip) >> bitCount; - } else { - bitStream >>= 16; - bitCount += 16; - } } - while ((bitStream & 3) == 3) { - n0 += 3; - bitStream >>= 2; - bitCount += 2; - } - n0 += bitStream & 3; - bitCount += 2; - if (n0 > *maxSVPtr) return ERROR(maxSymbolValue_tooSmall); - while (charnum < n0) normalizedCounter[charnum++] = 0; - if ((ip <= iend-7) || (ip + (bitCount>>3) <= iend-4)) { - ip += bitCount>>3; - bitCount &= 7; - bitStream = MEM_readLE32(ip) >> bitCount; - } else { - bitStream >>= 2; - } } - { int const max = (2*threshold-1) - remaining; - int count; + while ((remaining>1) & (charnum<=*maxSVPtr)) { + if (previous0) { + unsigned n0 = charnum; + while ((bitStream & 0xFFFF) == 0xFFFF) { + n0 += 24; + if (ip < iend-5) { + ip += 2; + bitStream = MEM_readLE32(ip) >> bitCount; + } else { + bitStream >>= 16; + bitCount += 16; + } } + while ((bitStream & 3) == 3) { + n0 += 3; + bitStream >>= 2; + bitCount += 2; + } + n0 += bitStream & 3; + bitCount += 2; + if (n0 > *maxSVPtr) return ERROR(maxSymbolValue_tooSmall); + while (charnum < n0) normalizedCounter[charnum++] = 0; + if ((ip <= iend-7) || (ip + (bitCount>>3) <= iend-4)) { + ip += bitCount>>3; + bitCount &= 7; + bitStream = MEM_readLE32(ip) >> bitCount; + } else { + bitStream >>= 2; + } } + { int const max = (2*threshold-1) - remaining; + int count; - if ((bitStream & (threshold-1)) < (U32)max) { - count = bitStream & (threshold-1); - bitCount += nbBits-1; - } else { - count = bitStream & (2*threshold-1); - if (count >= threshold) count -= max; - bitCount += nbBits; - } + if ((bitStream & (threshold-1)) < (U32)max) { + count = bitStream & (threshold-1); + bitCount += nbBits-1; + } else { + count = bitStream & (2*threshold-1); + if (count >= threshold) count -= max; + bitCount += nbBits; + } - count--; /* extra accuracy */ - remaining -= count < 0 ? -count : count; /* -1 means +1 */ - normalizedCounter[charnum++] = (short)count; - previous0 = !count; - while (remaining < threshold) { - nbBits--; - threshold >>= 1; - } + count--; /* extra accuracy */ + remaining -= count < 0 ? -count : count; /* -1 means +1 */ + normalizedCounter[charnum++] = (short)count; + previous0 = !count; + while (remaining < threshold) { + nbBits--; + threshold >>= 1; + } - if ((ip <= iend-7) || (ip + (bitCount>>3) <= iend-4)) { - ip += bitCount>>3; - bitCount &= 7; - } else { - bitCount -= (int)(8 * (iend - 4 - ip)); - ip = iend - 4; - } - bitStream = MEM_readLE32(ip) >> (bitCount & 31); - } } /* while ((remaining>1) & (charnum<=*maxSVPtr)) */ - if (remaining != 1) return ERROR(corruption_detected); - if (bitCount > 32) return ERROR(corruption_detected); - *maxSVPtr = charnum-1; + if ((ip <= iend-7) || (ip + (bitCount>>3) <= iend-4)) { + ip += bitCount>>3; + bitCount &= 7; + } else { + bitCount -= (int)(8 * (iend - 4 - ip)); + ip = iend - 4; + } + bitStream = MEM_readLE32(ip) >> (bitCount & 31); + } } /* while ((remaining>1) & (charnum<=*maxSVPtr)) */ + if (remaining != 1) return ERROR(corruption_detected); + if (bitCount > 32) return ERROR(corruption_detected); + *maxSVPtr = charnum-1; - ip += (bitCount+7)>>3; - return ip-istart; + ip += (bitCount+7)>>3; + return ip-istart; } /*! HUF_readStats() : - Read compact Huffman tree, saved by HUF_writeCTable(). - `huffWeight` is destination buffer. - `rankStats` is assumed to be a table of at least HUF_TABLELOG_MAX U32. - @return : size read from `src` , or an error Code . - Note : Needed by HUF_readCTable() and HUF_readDTableX?() . + Read compact Huffman tree, saved by HUF_writeCTable(). + `huffWeight` is destination buffer. + `rankStats` is assumed to be a table of at least HUF_TABLELOG_MAX U32. + @return : size read from `src` , or an error Code . + Note : Needed by HUF_readCTable() and HUF_readDTableX?() . */ size_t HUF_readStats(BYTE* huffWeight, size_t hwSize, U32* rankStats, - U32* nbSymbolsPtr, U32* tableLogPtr, - const void* src, size_t srcSize) + U32* nbSymbolsPtr, U32* tableLogPtr, + const void* src, size_t srcSize) { - U32 weightTotal; - const BYTE* ip = (const BYTE*) src; - size_t iSize; - size_t oSize; + U32 weightTotal; + const BYTE* ip = (const BYTE*) src; + size_t iSize; + size_t oSize; - if (!srcSize) return ERROR(srcSize_wrong); - iSize = ip[0]; - /* memset(huffWeight, 0, hwSize); *//* is not necessary, even though some analyzer complain ... */ + if (!srcSize) return ERROR(srcSize_wrong); + iSize = ip[0]; + /* memset(huffWeight, 0, hwSize); *//* is not necessary, even though some analyzer complain ... */ - if (iSize >= 128) { /* special header */ - oSize = iSize - 127; - iSize = ((oSize+1)/2); - if (iSize+1 > srcSize) return ERROR(srcSize_wrong); - if (oSize >= hwSize) return ERROR(corruption_detected); - ip += 1; - { U32 n; - for (n=0; n> 4; - huffWeight[n+1] = ip[n/2] & 15; - } } } - else { /* header compressed with FSE (normal case) */ - FSE_DTable fseWorkspace[FSE_DTABLE_SIZE_U32(6)]; /* 6 is max possible tableLog for HUF header (maybe even 5, to be tested) */ - if (iSize+1 > srcSize) return ERROR(srcSize_wrong); - oSize = FSE_decompress_wksp(huffWeight, hwSize-1, ip+1, iSize, fseWorkspace, 6); /* max (hwSize-1) values decoded, as last one is implied */ - if (FSE_isError(oSize)) return oSize; - } + if (iSize >= 128) { /* special header */ + oSize = iSize - 127; + iSize = ((oSize+1)/2); + if (iSize+1 > srcSize) return ERROR(srcSize_wrong); + if (oSize >= hwSize) return ERROR(corruption_detected); + ip += 1; + { U32 n; + for (n=0; n> 4; + huffWeight[n+1] = ip[n/2] & 15; + } } } + else { /* header compressed with FSE (normal case) */ + FSE_DTable fseWorkspace[FSE_DTABLE_SIZE_U32(6)]; /* 6 is max possible tableLog for HUF header (maybe even 5, to be tested) */ + if (iSize+1 > srcSize) return ERROR(srcSize_wrong); + oSize = FSE_decompress_wksp(huffWeight, hwSize-1, ip+1, iSize, fseWorkspace, 6); /* max (hwSize-1) values decoded, as last one is implied */ + if (FSE_isError(oSize)) return oSize; + } - /* collect weight stats */ - memset(rankStats, 0, (HUF_TABLELOG_MAX + 1) * sizeof(U32)); - weightTotal = 0; - { U32 n; for (n=0; n= HUF_TABLELOG_MAX) return ERROR(corruption_detected); - rankStats[huffWeight[n]]++; - weightTotal += (1 << huffWeight[n]) >> 1; - } } - if (weightTotal == 0) return ERROR(corruption_detected); + /* collect weight stats */ + memset(rankStats, 0, (HUF_TABLELOG_MAX + 1) * sizeof(U32)); + weightTotal = 0; + { U32 n; for (n=0; n= HUF_TABLELOG_MAX) return ERROR(corruption_detected); + rankStats[huffWeight[n]]++; + weightTotal += (1 << huffWeight[n]) >> 1; + } } + if (weightTotal == 0) return ERROR(corruption_detected); - /* get last non-null symbol weight (implied, total must be 2^n) */ - { U32 const tableLog = BIT_highbit32(weightTotal) + 1; - if (tableLog > HUF_TABLELOG_MAX) return ERROR(corruption_detected); - *tableLogPtr = tableLog; - /* determine last weight */ - { U32 const total = 1 << tableLog; - U32 const rest = total - weightTotal; - U32 const verif = 1 << BIT_highbit32(rest); - U32 const lastWeight = BIT_highbit32(rest) + 1; - if (verif != rest) return ERROR(corruption_detected); /* last value must be a clean power of 2 */ - huffWeight[oSize] = (BYTE)lastWeight; - rankStats[lastWeight]++; - } } + /* get last non-null symbol weight (implied, total must be 2^n) */ + { U32 const tableLog = BIT_highbit32(weightTotal) + 1; + if (tableLog > HUF_TABLELOG_MAX) return ERROR(corruption_detected); + *tableLogPtr = tableLog; + /* determine last weight */ + { U32 const total = 1 << tableLog; + U32 const rest = total - weightTotal; + U32 const verif = 1 << BIT_highbit32(rest); + U32 const lastWeight = BIT_highbit32(rest) + 1; + if (verif != rest) return ERROR(corruption_detected); /* last value must be a clean power of 2 */ + huffWeight[oSize] = (BYTE)lastWeight; + rankStats[lastWeight]++; + } } - /* check tree construction validity */ - if ((rankStats[1] < 2) || (rankStats[1] & 1)) return ERROR(corruption_detected); /* by construction : at least 2 elts of rank 1, must be even */ + /* check tree construction validity */ + if ((rankStats[1] < 2) || (rankStats[1] & 1)) return ERROR(corruption_detected); /* by construction : at least 2 elts of rank 1, must be even */ - /* results */ - *nbSymbolsPtr = (U32)(oSize+1); - return iSize+1; + /* results */ + *nbSymbolsPtr = (U32)(oSize+1); + return iSize+1; } diff --git a/contrib/linux-kernel/lib/error_private.c b/contrib/linux-kernel/lib/error_private.c index 44ae20104..83e27cb49 100644 --- a/contrib/linux-kernel/lib/error_private.c +++ b/contrib/linux-kernel/lib/error_private.c @@ -13,32 +13,32 @@ const char* ERR_getErrorString(ERR_enum code) { - static const char* const notErrorCode = "Unspecified error code"; - switch( code ) - { - case PREFIX(no_error): return "No error detected"; - case PREFIX(GENERIC): return "Error (generic)"; - case PREFIX(prefix_unknown): return "Unknown frame descriptor"; - case PREFIX(version_unsupported): return "Version not supported"; - case PREFIX(parameter_unknown): return "Unknown parameter type"; - case PREFIX(frameParameter_unsupported): return "Unsupported frame parameter"; - case PREFIX(frameParameter_unsupportedBy32bits): return "Frame parameter unsupported in 32-bits mode"; - case PREFIX(frameParameter_windowTooLarge): return "Frame requires too much memory for decoding"; - case PREFIX(compressionParameter_unsupported): return "Compression parameter is out of bound"; - case PREFIX(init_missing): return "Context should be init first"; - case PREFIX(memory_allocation): return "Allocation error : not enough memory"; - case PREFIX(stage_wrong): return "Operation not authorized at current processing stage"; - case PREFIX(dstSize_tooSmall): return "Destination buffer is too small"; - case PREFIX(srcSize_wrong): return "Src size incorrect"; - case PREFIX(corruption_detected): return "Corrupted block detected"; - case PREFIX(checksum_wrong): return "Restored data doesn't match checksum"; - case PREFIX(tableLog_tooLarge): return "tableLog requires too much memory : unsupported"; - case PREFIX(maxSymbolValue_tooLarge): return "Unsupported max Symbol Value : too large"; - case PREFIX(maxSymbolValue_tooSmall): return "Specified maxSymbolValue is too small"; - case PREFIX(dictionary_corrupted): return "Dictionary is corrupted"; - case PREFIX(dictionary_wrong): return "Dictionary mismatch"; - case PREFIX(dictionaryCreation_failed): return "Cannot create Dictionary from provided samples"; - case PREFIX(maxCode): - default: return notErrorCode; - } + static const char* const notErrorCode = "Unspecified error code"; + switch( code ) + { + case PREFIX(no_error): return "No error detected"; + case PREFIX(GENERIC): return "Error (generic)"; + case PREFIX(prefix_unknown): return "Unknown frame descriptor"; + case PREFIX(version_unsupported): return "Version not supported"; + case PREFIX(parameter_unknown): return "Unknown parameter type"; + case PREFIX(frameParameter_unsupported): return "Unsupported frame parameter"; + case PREFIX(frameParameter_unsupportedBy32bits): return "Frame parameter unsupported in 32-bits mode"; + case PREFIX(frameParameter_windowTooLarge): return "Frame requires too much memory for decoding"; + case PREFIX(compressionParameter_unsupported): return "Compression parameter is out of bound"; + case PREFIX(init_missing): return "Context should be init first"; + case PREFIX(memory_allocation): return "Allocation error : not enough memory"; + case PREFIX(stage_wrong): return "Operation not authorized at current processing stage"; + case PREFIX(dstSize_tooSmall): return "Destination buffer is too small"; + case PREFIX(srcSize_wrong): return "Src size incorrect"; + case PREFIX(corruption_detected): return "Corrupted block detected"; + case PREFIX(checksum_wrong): return "Restored data doesn't match checksum"; + case PREFIX(tableLog_tooLarge): return "tableLog requires too much memory : unsupported"; + case PREFIX(maxSymbolValue_tooLarge): return "Unsupported max Symbol Value : too large"; + case PREFIX(maxSymbolValue_tooSmall): return "Specified maxSymbolValue is too small"; + case PREFIX(dictionary_corrupted): return "Dictionary is corrupted"; + case PREFIX(dictionary_wrong): return "Dictionary mismatch"; + case PREFIX(dictionaryCreation_failed): return "Cannot create Dictionary from provided samples"; + case PREFIX(maxCode): + default: return notErrorCode; + } } diff --git a/contrib/linux-kernel/lib/error_private.h b/contrib/linux-kernel/lib/error_private.h index 1bc2e4954..dc160914b 100644 --- a/contrib/linux-kernel/lib/error_private.h +++ b/contrib/linux-kernel/lib/error_private.h @@ -66,7 +66,7 @@ const char* ERR_getErrorString(ERR_enum code); /* error_private.c */ ERR_STATIC const char* ERR_getErrorName(size_t code) { - return ERR_getErrorString(ERR_getErrorCode(code)); + return ERR_getErrorString(ERR_getErrorCode(code)); } #if defined (__cplusplus) diff --git a/contrib/linux-kernel/lib/fse.h b/contrib/linux-kernel/lib/fse.h index baac39032..bcb592e96 100644 --- a/contrib/linux-kernel/lib/fse.h +++ b/contrib/linux-kernel/lib/fse.h @@ -9,9 +9,9 @@ modification, are permitted provided that the following conditions are met: - * Redistributions of source code must retain the above copyright + * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - * Redistributions in binary form must reproduce the above + * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. @@ -75,28 +75,28 @@ FSE_PUBLIC_API unsigned FSE_versionNumber(void); /**< library version number; * FSE simple functions ******************************************/ /*! FSE_compress() : - Compress content of buffer 'src', of size 'srcSize', into destination buffer 'dst'. - 'dst' buffer must be already allocated. Compression runs faster is dstCapacity >= FSE_compressBound(srcSize). - @return : size of compressed data (<= dstCapacity). - Special values : if return == 0, srcData is not compressible => Nothing is stored within dst !!! - if return == 1, srcData is a single byte symbol * srcSize times. Use RLE compression instead. - if FSE_isError(return), compression failed (more details using FSE_getErrorName()) + Compress content of buffer 'src', of size 'srcSize', into destination buffer 'dst'. + 'dst' buffer must be already allocated. Compression runs faster is dstCapacity >= FSE_compressBound(srcSize). + @return : size of compressed data (<= dstCapacity). + Special values : if return == 0, srcData is not compressible => Nothing is stored within dst !!! + if return == 1, srcData is a single byte symbol * srcSize times. Use RLE compression instead. + if FSE_isError(return), compression failed (more details using FSE_getErrorName()) */ FSE_PUBLIC_API size_t FSE_compress(void* dst, size_t dstCapacity, - const void* src, size_t srcSize); + const void* src, size_t srcSize); /*! FSE_decompress(): - Decompress FSE data from buffer 'cSrc', of size 'cSrcSize', - into already allocated destination buffer 'dst', of size 'dstCapacity'. - @return : size of regenerated data (<= maxDstSize), - or an error code, which can be tested using FSE_isError() . + Decompress FSE data from buffer 'cSrc', of size 'cSrcSize', + into already allocated destination buffer 'dst', of size 'dstCapacity'. + @return : size of regenerated data (<= maxDstSize), + or an error code, which can be tested using FSE_isError() . - ** Important ** : FSE_decompress() does not decompress non-compressible nor RLE data !!! - Why ? : making this distinction requires a header. - Header management is intentionally delegated to the user layer, which can better manage special cases. + ** Important ** : FSE_decompress() does not decompress non-compressible nor RLE data !!! + Why ? : making this distinction requires a header. + Header management is intentionally delegated to the user layer, which can better manage special cases. */ FSE_PUBLIC_API size_t FSE_decompress(void* dst, size_t dstCapacity, - const void* cSrc, size_t cSrcSize); + const void* cSrc, size_t cSrcSize); /*-***************************************** @@ -113,12 +113,12 @@ FSE_PUBLIC_API const char* FSE_getErrorName(size_t code); /* provides error co * FSE advanced functions ******************************************/ /*! FSE_compress2() : - Same as FSE_compress(), but allows the selection of 'maxSymbolValue' and 'tableLog' - Both parameters can be defined as '0' to mean : use default value - @return : size of compressed data - Special values : if return == 0, srcData is not compressible => Nothing is stored within cSrc !!! - if return == 1, srcData is a single byte symbol * srcSize times. Use RLE compression. - if FSE_isError(return), it's an error code. + Same as FSE_compress(), but allows the selection of 'maxSymbolValue' and 'tableLog' + Both parameters can be defined as '0' to mean : use default value + @return : size of compressed data + Special values : if return == 0, srcData is not compressible => Nothing is stored within cSrc !!! + if return == 1, srcData is a single byte symbol * srcSize times. Use RLE compression. + if FSE_isError(return), it's an error code. */ FSE_PUBLIC_API size_t FSE_compress2 (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog); @@ -147,55 +147,55 @@ or to save and provide normalized distribution using external method. /* *** COMPRESSION *** */ /*! FSE_count(): - Provides the precise count of each byte within a table 'count'. - 'count' is a table of unsigned int, of minimum size (*maxSymbolValuePtr+1). - *maxSymbolValuePtr will be updated if detected smaller than initial value. - @return : the count of the most frequent symbol (which is not identified). - if return == srcSize, there is only one symbol. - Can also return an error code, which can be tested with FSE_isError(). */ + Provides the precise count of each byte within a table 'count'. + 'count' is a table of unsigned int, of minimum size (*maxSymbolValuePtr+1). + *maxSymbolValuePtr will be updated if detected smaller than initial value. + @return : the count of the most frequent symbol (which is not identified). + if return == srcSize, there is only one symbol. + Can also return an error code, which can be tested with FSE_isError(). */ FSE_PUBLIC_API size_t FSE_count(unsigned* count, unsigned* maxSymbolValuePtr, const void* src, size_t srcSize); /*! FSE_optimalTableLog(): - dynamically downsize 'tableLog' when conditions are met. - It saves CPU time, by using smaller tables, while preserving or even improving compression ratio. - @return : recommended tableLog (necessarily <= 'maxTableLog') */ + dynamically downsize 'tableLog' when conditions are met. + It saves CPU time, by using smaller tables, while preserving or even improving compression ratio. + @return : recommended tableLog (necessarily <= 'maxTableLog') */ FSE_PUBLIC_API unsigned FSE_optimalTableLog(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue); /*! FSE_normalizeCount(): - normalize counts so that sum(count[]) == Power_of_2 (2^tableLog) - 'normalizedCounter' is a table of short, of minimum size (maxSymbolValue+1). - @return : tableLog, - or an errorCode, which can be tested using FSE_isError() */ + normalize counts so that sum(count[]) == Power_of_2 (2^tableLog) + 'normalizedCounter' is a table of short, of minimum size (maxSymbolValue+1). + @return : tableLog, + or an errorCode, which can be tested using FSE_isError() */ FSE_PUBLIC_API size_t FSE_normalizeCount(short* normalizedCounter, unsigned tableLog, const unsigned* count, size_t srcSize, unsigned maxSymbolValue); /*! FSE_NCountWriteBound(): - Provides the maximum possible size of an FSE normalized table, given 'maxSymbolValue' and 'tableLog'. - Typically useful for allocation purpose. */ + Provides the maximum possible size of an FSE normalized table, given 'maxSymbolValue' and 'tableLog'. + Typically useful for allocation purpose. */ FSE_PUBLIC_API size_t FSE_NCountWriteBound(unsigned maxSymbolValue, unsigned tableLog); /*! FSE_writeNCount(): - Compactly save 'normalizedCounter' into 'buffer'. - @return : size of the compressed table, - or an errorCode, which can be tested using FSE_isError(). */ + Compactly save 'normalizedCounter' into 'buffer'. + @return : size of the compressed table, + or an errorCode, which can be tested using FSE_isError(). */ FSE_PUBLIC_API size_t FSE_writeNCount (void* buffer, size_t bufferSize, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog); /*! Constructor and Destructor of FSE_CTable. - Note that FSE_CTable size depends on 'tableLog' and 'maxSymbolValue' */ + Note that FSE_CTable size depends on 'tableLog' and 'maxSymbolValue' */ typedef unsigned FSE_CTable; /* don't allocate that. It's only meant to be more restrictive than void* */ FSE_PUBLIC_API FSE_CTable* FSE_createCTable (unsigned tableLog, unsigned maxSymbolValue); FSE_PUBLIC_API void FSE_freeCTable (FSE_CTable* ct); /*! FSE_buildCTable(): - Builds `ct`, which must be already allocated, using FSE_createCTable(). - @return : 0, or an errorCode, which can be tested using FSE_isError() */ + Builds `ct`, which must be already allocated, using FSE_createCTable(). + @return : 0, or an errorCode, which can be tested using FSE_isError() */ FSE_PUBLIC_API size_t FSE_buildCTable(FSE_CTable* ct, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog); /*! FSE_compress_usingCTable(): - Compress `src` using `ct` into `dst` which must be already allocated. - @return : size of compressed data (<= `dstCapacity`), - or 0 if compressed data could not fit into `dst`, - or an errorCode, which can be tested using FSE_isError() */ + Compress `src` using `ct` into `dst` which must be already allocated. + @return : size of compressed data (<= `dstCapacity`), + or 0 if compressed data could not fit into `dst`, + or an errorCode, which can be tested using FSE_isError() */ FSE_PUBLIC_API size_t FSE_compress_usingCTable (void* dst, size_t dstCapacity, const void* src, size_t srcSize, const FSE_CTable* ct); /*! @@ -245,28 +245,28 @@ If there is an error, the function will return an ErrorCode (which can be tested /* *** DECOMPRESSION *** */ /*! FSE_readNCount(): - Read compactly saved 'normalizedCounter' from 'rBuffer'. - @return : size read from 'rBuffer', - or an errorCode, which can be tested using FSE_isError(). - maxSymbolValuePtr[0] and tableLogPtr[0] will also be updated with their respective values */ + Read compactly saved 'normalizedCounter' from 'rBuffer'. + @return : size read from 'rBuffer', + or an errorCode, which can be tested using FSE_isError(). + maxSymbolValuePtr[0] and tableLogPtr[0] will also be updated with their respective values */ FSE_PUBLIC_API size_t FSE_readNCount (short* normalizedCounter, unsigned* maxSymbolValuePtr, unsigned* tableLogPtr, const void* rBuffer, size_t rBuffSize); /*! Constructor and Destructor of FSE_DTable. - Note that its size depends on 'tableLog' */ + Note that its size depends on 'tableLog' */ typedef unsigned FSE_DTable; /* don't allocate that. It's just a way to be more restrictive than void* */ FSE_PUBLIC_API FSE_DTable* FSE_createDTable(unsigned tableLog); FSE_PUBLIC_API void FSE_freeDTable(FSE_DTable* dt); /*! FSE_buildDTable(): - Builds 'dt', which must be already allocated, using FSE_createDTable(). - return : 0, or an errorCode, which can be tested using FSE_isError() */ + Builds 'dt', which must be already allocated, using FSE_createDTable(). + return : 0, or an errorCode, which can be tested using FSE_isError() */ FSE_PUBLIC_API size_t FSE_buildDTable (FSE_DTable* dt, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog); /*! FSE_decompress_usingDTable(): - Decompress compressed source `cSrc` of size `cSrcSize` using `dt` - into `dst` which must be already allocated. - @return : size of regenerated data (necessarily <= `dstCapacity`), - or an errorCode, which can be tested using FSE_isError() */ + Decompress compressed source `cSrc` of size `cSrcSize` using `dt` + into `dst` which must be already allocated. + @return : size of regenerated data (necessarily <= `dstCapacity`), + or an errorCode, which can be tested using FSE_isError() */ FSE_PUBLIC_API size_t FSE_decompress_usingDTable(void* dst, size_t dstCapacity, const void* cSrc, size_t cSrcSize, const FSE_DTable* dt); /*! @@ -325,7 +325,7 @@ If there is an error, the function will return an error code, which can be teste * `workSpace` size must be table of >= `1024` unsigned */ size_t FSE_count_wksp(unsigned* count, unsigned* maxSymbolValuePtr, - const void* source, size_t sourceSize, unsigned* workSpace); + const void* source, size_t sourceSize, unsigned* workSpace); /** FSE_countFast() : * same as FSE_count(), but blindly trusts that all byte values within src are <= *maxSymbolValuePtr @@ -386,10 +386,10 @@ size_t FSE_decompress_wksp(void* dst, size_t dstCapacity, const void* cSrc, size Hence their body are included in next section. */ typedef struct { - ptrdiff_t value; - const void* stateTable; - const void* symbolTT; - unsigned stateLog; + ptrdiff_t value; + const void* stateTable; + const void* symbolTT; + unsigned stateLog; } FSE_CState_t; static void FSE_initCState(FSE_CState_t* CStatePtr, const FSE_CTable* ct); @@ -413,32 +413,32 @@ FSE_CState_t state; // State tracking structure (can have several) The first thing to do is to init bitStream and state. - size_t errorCode = BIT_initCStream(&bitStream, dstBuffer, maxDstSize); - FSE_initCState(&state, ct); + size_t errorCode = BIT_initCStream(&bitStream, dstBuffer, maxDstSize); + FSE_initCState(&state, ct); Note that BIT_initCStream() can produce an error code, so its result should be tested, using FSE_isError(); You can then encode your input data, byte after byte. FSE_encodeSymbol() outputs a maximum of 'tableLog' bits at a time. Remember decoding will be done in reverse direction. - FSE_encodeByte(&bitStream, &state, symbol); + FSE_encodeByte(&bitStream, &state, symbol); At any time, you can also add any bit sequence. Note : maximum allowed nbBits is 25, for compatibility with 32-bits decoders - BIT_addBits(&bitStream, bitField, nbBits); + BIT_addBits(&bitStream, bitField, nbBits); The above methods don't commit data to memory, they just store it into local register, for speed. Local register size is 64-bits on 64-bits systems, 32-bits on 32-bits systems (size_t). Writing data to memory is a manual operation, performed by the flushBits function. - BIT_flushBits(&bitStream); + BIT_flushBits(&bitStream); Your last FSE encoding operation shall be to flush your last state value(s). - FSE_flushState(&bitStream, &state); + FSE_flushState(&bitStream, &state); Finally, you must close the bitStream. The function returns the size of CStream in bytes. If data couldn't fit into dstBuffer, it will return a 0 ( == not compressible) If there is an error, it returns an errorCode (which can be tested using FSE_isError()). - size_t size = BIT_closeCStream(&bitStream); + size_t size = BIT_closeCStream(&bitStream); */ @@ -446,8 +446,8 @@ If there is an error, it returns an errorCode (which can be tested using FSE_isE * FSE symbol decompression API *******************************************/ typedef struct { - size_t state; - const void* table; /* precise table may vary, depending on U16 */ + size_t state; + const void* table; /* precise table may vary, depending on U16 */ } FSE_DState_t; @@ -469,24 +469,24 @@ FSE_DState_t DState; // State context. Multiple ones are possible FSE_DTable* DTablePtr; // Decoding table, provided by FSE_buildDTable() The first thing to do is to init the bitStream. - errorCode = BIT_initDStream(&DStream, srcBuffer, srcSize); + errorCode = BIT_initDStream(&DStream, srcBuffer, srcSize); You should then retrieve your initial state(s) (in reverse flushing order if you have several ones) : - errorCode = FSE_initDState(&DState, &DStream, DTablePtr); + errorCode = FSE_initDState(&DState, &DStream, DTablePtr); You can then decode your data, symbol after symbol. For information the maximum number of bits read by FSE_decodeSymbol() is 'tableLog'. Keep in mind that symbols are decoded in reverse order, like a LIFO stack (last in, first out). - unsigned char symbol = FSE_decodeSymbol(&DState, &DStream); + unsigned char symbol = FSE_decodeSymbol(&DState, &DStream); You can retrieve any bitfield you eventually stored into the bitStream (in reverse order) Note : maximum allowed nbBits is 25, for 32-bits compatibility - size_t bitField = BIT_readBits(&DStream, nbBits); + size_t bitField = BIT_readBits(&DStream, nbBits); All above operations only read from local register (which size depends on size_t). Refueling the register from memory is manually performed by the reload method. - endSignal = FSE_reloadDStream(&DStream); + endSignal = FSE_reloadDStream(&DStream); BIT_reloadDStream() result tells if there is still some more data to read from DStream. BIT_DStream_unfinished : there is still some data left into the DStream. @@ -497,13 +497,13 @@ BIT_DStream_tooFar : Dstream went too far. Decompression result is corrupted. When reaching end of buffer (BIT_DStream_endOfBuffer), progress slowly, notably if you decode multiple symbols per loop, to properly detect the exact end of stream. After each decoded symbol, check if DStream is fully consumed using this simple test : - BIT_reloadDStream(&DStream) >= BIT_DStream_completed + BIT_reloadDStream(&DStream) >= BIT_DStream_completed When it's done, verify decompression is fully completed, by checking both DStream and the relevant states. Checking if DStream has reached its end is performed by : - BIT_endOfDStream(&DStream); + BIT_endOfDStream(&DStream); Check also the states. There might be some symbols left there, if some high probability ones (>50%) are possible. - FSE_endOfDState(&DState); + FSE_endOfDState(&DState); */ @@ -518,19 +518,19 @@ static unsigned char FSE_decodeSymbolFast(FSE_DState_t* DStatePtr, BIT_DStream_t * Implementation of inlined functions *******************************************/ typedef struct { - int deltaFindState; - U32 deltaNbBits; + int deltaFindState; + U32 deltaNbBits; } FSE_symbolCompressionTransform; /* total 8 bytes */ MEM_STATIC void FSE_initCState(FSE_CState_t* statePtr, const FSE_CTable* ct) { - const void* ptr = ct; - const U16* u16ptr = (const U16*) ptr; - const U32 tableLog = MEM_read16(ptr); - statePtr->value = (ptrdiff_t)1<stateTable = u16ptr+2; - statePtr->symbolTT = ((const U32*)ct + 1 + (tableLog ? (1<<(tableLog-1)) : 1)); - statePtr->stateLog = tableLog; + const void* ptr = ct; + const U16* u16ptr = (const U16*) ptr; + const U32 tableLog = MEM_read16(ptr); + statePtr->value = (ptrdiff_t)1<stateTable = u16ptr+2; + statePtr->symbolTT = ((const U32*)ct + 1 + (tableLog ? (1<<(tableLog-1)) : 1)); + statePtr->stateLog = tableLog; } @@ -539,95 +539,95 @@ MEM_STATIC void FSE_initCState(FSE_CState_t* statePtr, const FSE_CTable* ct) * uses the smallest state value possible, saving the cost of this symbol */ MEM_STATIC void FSE_initCState2(FSE_CState_t* statePtr, const FSE_CTable* ct, U32 symbol) { - FSE_initCState(statePtr, ct); - { const FSE_symbolCompressionTransform symbolTT = ((const FSE_symbolCompressionTransform*)(statePtr->symbolTT))[symbol]; - const U16* stateTable = (const U16*)(statePtr->stateTable); - U32 nbBitsOut = (U32)((symbolTT.deltaNbBits + (1<<15)) >> 16); - statePtr->value = (nbBitsOut << 16) - symbolTT.deltaNbBits; - statePtr->value = stateTable[(statePtr->value >> nbBitsOut) + symbolTT.deltaFindState]; - } + FSE_initCState(statePtr, ct); + { const FSE_symbolCompressionTransform symbolTT = ((const FSE_symbolCompressionTransform*)(statePtr->symbolTT))[symbol]; + const U16* stateTable = (const U16*)(statePtr->stateTable); + U32 nbBitsOut = (U32)((symbolTT.deltaNbBits + (1<<15)) >> 16); + statePtr->value = (nbBitsOut << 16) - symbolTT.deltaNbBits; + statePtr->value = stateTable[(statePtr->value >> nbBitsOut) + symbolTT.deltaFindState]; + } } MEM_STATIC void FSE_encodeSymbol(BIT_CStream_t* bitC, FSE_CState_t* statePtr, U32 symbol) { - const FSE_symbolCompressionTransform symbolTT = ((const FSE_symbolCompressionTransform*)(statePtr->symbolTT))[symbol]; - const U16* const stateTable = (const U16*)(statePtr->stateTable); - U32 nbBitsOut = (U32)((statePtr->value + symbolTT.deltaNbBits) >> 16); - BIT_addBits(bitC, statePtr->value, nbBitsOut); - statePtr->value = stateTable[ (statePtr->value >> nbBitsOut) + symbolTT.deltaFindState]; + const FSE_symbolCompressionTransform symbolTT = ((const FSE_symbolCompressionTransform*)(statePtr->symbolTT))[symbol]; + const U16* const stateTable = (const U16*)(statePtr->stateTable); + U32 nbBitsOut = (U32)((statePtr->value + symbolTT.deltaNbBits) >> 16); + BIT_addBits(bitC, statePtr->value, nbBitsOut); + statePtr->value = stateTable[ (statePtr->value >> nbBitsOut) + symbolTT.deltaFindState]; } MEM_STATIC void FSE_flushCState(BIT_CStream_t* bitC, const FSE_CState_t* statePtr) { - BIT_addBits(bitC, statePtr->value, statePtr->stateLog); - BIT_flushBits(bitC); + BIT_addBits(bitC, statePtr->value, statePtr->stateLog); + BIT_flushBits(bitC); } /* ====== Decompression ====== */ typedef struct { - U16 tableLog; - U16 fastMode; + U16 tableLog; + U16 fastMode; } FSE_DTableHeader; /* sizeof U32 */ typedef struct { - unsigned short newState; - unsigned char symbol; - unsigned char nbBits; + unsigned short newState; + unsigned char symbol; + unsigned char nbBits; } FSE_decode_t; /* size == U32 */ MEM_STATIC void FSE_initDState(FSE_DState_t* DStatePtr, BIT_DStream_t* bitD, const FSE_DTable* dt) { - const void* ptr = dt; - const FSE_DTableHeader* const DTableH = (const FSE_DTableHeader*)ptr; - DStatePtr->state = BIT_readBits(bitD, DTableH->tableLog); - BIT_reloadDStream(bitD); - DStatePtr->table = dt + 1; + const void* ptr = dt; + const FSE_DTableHeader* const DTableH = (const FSE_DTableHeader*)ptr; + DStatePtr->state = BIT_readBits(bitD, DTableH->tableLog); + BIT_reloadDStream(bitD); + DStatePtr->table = dt + 1; } MEM_STATIC BYTE FSE_peekSymbol(const FSE_DState_t* DStatePtr) { - FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; - return DInfo.symbol; + FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; + return DInfo.symbol; } MEM_STATIC void FSE_updateState(FSE_DState_t* DStatePtr, BIT_DStream_t* bitD) { - FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; - U32 const nbBits = DInfo.nbBits; - size_t const lowBits = BIT_readBits(bitD, nbBits); - DStatePtr->state = DInfo.newState + lowBits; + FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; + U32 const nbBits = DInfo.nbBits; + size_t const lowBits = BIT_readBits(bitD, nbBits); + DStatePtr->state = DInfo.newState + lowBits; } MEM_STATIC BYTE FSE_decodeSymbol(FSE_DState_t* DStatePtr, BIT_DStream_t* bitD) { - FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; - U32 const nbBits = DInfo.nbBits; - BYTE const symbol = DInfo.symbol; - size_t const lowBits = BIT_readBits(bitD, nbBits); + FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; + U32 const nbBits = DInfo.nbBits; + BYTE const symbol = DInfo.symbol; + size_t const lowBits = BIT_readBits(bitD, nbBits); - DStatePtr->state = DInfo.newState + lowBits; - return symbol; + DStatePtr->state = DInfo.newState + lowBits; + return symbol; } /*! FSE_decodeSymbolFast() : - unsafe, only works if no symbol has a probability > 50% */ + unsafe, only works if no symbol has a probability > 50% */ MEM_STATIC BYTE FSE_decodeSymbolFast(FSE_DState_t* DStatePtr, BIT_DStream_t* bitD) { - FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; - U32 const nbBits = DInfo.nbBits; - BYTE const symbol = DInfo.symbol; - size_t const lowBits = BIT_readBitsFast(bitD, nbBits); + FSE_decode_t const DInfo = ((const FSE_decode_t*)(DStatePtr->table))[DStatePtr->state]; + U32 const nbBits = DInfo.nbBits; + BYTE const symbol = DInfo.symbol; + size_t const lowBits = BIT_readBitsFast(bitD, nbBits); - DStatePtr->state = DInfo.newState + lowBits; - return symbol; + DStatePtr->state = DInfo.newState + lowBits; + return symbol; } MEM_STATIC unsigned FSE_endOfDState(const FSE_DState_t* DStatePtr) { - return DStatePtr->state == 0; + return DStatePtr->state == 0; } diff --git a/contrib/linux-kernel/lib/fse_compress.c b/contrib/linux-kernel/lib/fse_compress.c index 13654d6e6..7340d413a 100644 --- a/contrib/linux-kernel/lib/fse_compress.c +++ b/contrib/linux-kernel/lib/fse_compress.c @@ -8,9 +8,9 @@ modification, are permitted provided that the following conditions are met: - * Redistributions of source code must retain the above copyright + * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - * Redistributions in binary form must reproduce the above + * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. @@ -27,9 +27,9 @@ (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - You can contact the author at : - - FSE source repository : https://github.com/Cyan4973/FiniteStateEntropy - - Public forum : https://groups.google.com/forum/#!forum/lz4c + You can contact the author at : + - FSE source repository : https://github.com/Cyan4973/FiniteStateEntropy + - Public forum : https://groups.google.com/forum/#!forum/lz4c ****************************************************************** */ /* ************************************************************** @@ -102,90 +102,90 @@ */ size_t FSE_buildCTable_wksp(FSE_CTable* ct, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize) { - U32 const tableSize = 1 << tableLog; - U32 const tableMask = tableSize - 1; - void* const ptr = ct; - U16* const tableU16 = ( (U16*) ptr) + 2; - void* const FSCT = ((U32*)ptr) + 1 /* header */ + (tableLog ? tableSize>>1 : 1) ; - FSE_symbolCompressionTransform* const symbolTT = (FSE_symbolCompressionTransform*) (FSCT); - U32 const step = FSE_TABLESTEP(tableSize); - U32 cumul[FSE_MAX_SYMBOL_VALUE+2]; + U32 const tableSize = 1 << tableLog; + U32 const tableMask = tableSize - 1; + void* const ptr = ct; + U16* const tableU16 = ( (U16*) ptr) + 2; + void* const FSCT = ((U32*)ptr) + 1 /* header */ + (tableLog ? tableSize>>1 : 1) ; + FSE_symbolCompressionTransform* const symbolTT = (FSE_symbolCompressionTransform*) (FSCT); + U32 const step = FSE_TABLESTEP(tableSize); + U32 cumul[FSE_MAX_SYMBOL_VALUE+2]; - FSE_FUNCTION_TYPE* const tableSymbol = (FSE_FUNCTION_TYPE*)workSpace; - U32 highThreshold = tableSize-1; + FSE_FUNCTION_TYPE* const tableSymbol = (FSE_FUNCTION_TYPE*)workSpace; + U32 highThreshold = tableSize-1; - /* CTable header */ - if (((size_t)1 << tableLog) * sizeof(FSE_FUNCTION_TYPE) > wkspSize) return ERROR(tableLog_tooLarge); - tableU16[-2] = (U16) tableLog; - tableU16[-1] = (U16) maxSymbolValue; + /* CTable header */ + if (((size_t)1 << tableLog) * sizeof(FSE_FUNCTION_TYPE) > wkspSize) return ERROR(tableLog_tooLarge); + tableU16[-2] = (U16) tableLog; + tableU16[-1] = (U16) maxSymbolValue; - /* For explanations on how to distribute symbol values over the table : - * http://fastcompression.blogspot.fr/2014/02/fse-distributing-symbol-values.html */ + /* For explanations on how to distribute symbol values over the table : + * http://fastcompression.blogspot.fr/2014/02/fse-distributing-symbol-values.html */ - /* symbol start positions */ - { U32 u; - cumul[0] = 0; - for (u=1; u<=maxSymbolValue+1; u++) { - if (normalizedCounter[u-1]==-1) { /* Low proba symbol */ - cumul[u] = cumul[u-1] + 1; - tableSymbol[highThreshold--] = (FSE_FUNCTION_TYPE)(u-1); - } else { - cumul[u] = cumul[u-1] + normalizedCounter[u-1]; - } } - cumul[maxSymbolValue+1] = tableSize+1; - } + /* symbol start positions */ + { U32 u; + cumul[0] = 0; + for (u=1; u<=maxSymbolValue+1; u++) { + if (normalizedCounter[u-1]==-1) { /* Low proba symbol */ + cumul[u] = cumul[u-1] + 1; + tableSymbol[highThreshold--] = (FSE_FUNCTION_TYPE)(u-1); + } else { + cumul[u] = cumul[u-1] + normalizedCounter[u-1]; + } } + cumul[maxSymbolValue+1] = tableSize+1; + } - /* Spread symbols */ - { U32 position = 0; - U32 symbol; - for (symbol=0; symbol<=maxSymbolValue; symbol++) { - int nbOccurences; - for (nbOccurences=0; nbOccurences highThreshold) position = (position + step) & tableMask; /* Low proba area */ - } } + /* Spread symbols */ + { U32 position = 0; + U32 symbol; + for (symbol=0; symbol<=maxSymbolValue; symbol++) { + int nbOccurences; + for (nbOccurences=0; nbOccurences highThreshold) position = (position + step) & tableMask; /* Low proba area */ + } } - if (position!=0) return ERROR(GENERIC); /* Must have gone through all positions */ - } + if (position!=0) return ERROR(GENERIC); /* Must have gone through all positions */ + } - /* Build table */ - { U32 u; for (u=0; u> 3) + 3; - return maxSymbolValue ? maxHeaderSize : FSE_NCOUNTBOUND; /* maxSymbolValue==0 ? use default */ + size_t const maxHeaderSize = (((maxSymbolValue+1) * tableLog) >> 3) + 3; + return maxSymbolValue ? maxHeaderSize : FSE_NCOUNTBOUND; /* maxSymbolValue==0 ? use default */ } static size_t FSE_writeNCount_generic (void* header, size_t headerBufferSize, - const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog, - unsigned writeIsSafe) + const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog, + unsigned writeIsSafe) { - BYTE* const ostart = (BYTE*) header; - BYTE* out = ostart; - BYTE* const oend = ostart + headerBufferSize; - int nbBits; - const int tableSize = 1 << tableLog; - int remaining; - int threshold; - U32 bitStream; - int bitCount; - unsigned charnum = 0; - int previous0 = 0; + BYTE* const ostart = (BYTE*) header; + BYTE* out = ostart; + BYTE* const oend = ostart + headerBufferSize; + int nbBits; + const int tableSize = 1 << tableLog; + int remaining; + int threshold; + U32 bitStream; + int bitCount; + unsigned charnum = 0; + int previous0 = 0; - bitStream = 0; - bitCount = 0; - /* Table Size */ - bitStream += (tableLog-FSE_MIN_TABLELOG) << bitCount; - bitCount += 4; + bitStream = 0; + bitCount = 0; + /* Table Size */ + bitStream += (tableLog-FSE_MIN_TABLELOG) << bitCount; + bitCount += 4; - /* Init */ - remaining = tableSize+1; /* +1 for extra accuracy */ - threshold = tableSize; - nbBits = tableLog+1; + /* Init */ + remaining = tableSize+1; /* +1 for extra accuracy */ + threshold = tableSize; + nbBits = tableLog+1; - while (remaining>1) { /* stops at 1 */ - if (previous0) { - unsigned start = charnum; - while (!normalizedCounter[charnum]) charnum++; - while (charnum >= start+24) { - start+=24; - bitStream += 0xFFFFU << bitCount; - if ((!writeIsSafe) && (out > oend-2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ - out[0] = (BYTE) bitStream; - out[1] = (BYTE)(bitStream>>8); - out+=2; - bitStream>>=16; - } - while (charnum >= start+3) { - start+=3; - bitStream += 3 << bitCount; - bitCount += 2; - } - bitStream += (charnum-start) << bitCount; - bitCount += 2; - if (bitCount>16) { - if ((!writeIsSafe) && (out > oend - 2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ - out[0] = (BYTE)bitStream; - out[1] = (BYTE)(bitStream>>8); - out += 2; - bitStream >>= 16; - bitCount -= 16; - } } - { int count = normalizedCounter[charnum++]; - int const max = (2*threshold-1)-remaining; - remaining -= count < 0 ? -count : count; - count++; /* +1 for extra accuracy */ - if (count>=threshold) count += max; /* [0..max[ [max..threshold[ (...) [threshold+max 2*threshold[ */ - bitStream += count << bitCount; - bitCount += nbBits; - bitCount -= (count>=1; - } - if (bitCount>16) { - if ((!writeIsSafe) && (out > oend - 2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ - out[0] = (BYTE)bitStream; - out[1] = (BYTE)(bitStream>>8); - out += 2; - bitStream >>= 16; - bitCount -= 16; - } } + while (remaining>1) { /* stops at 1 */ + if (previous0) { + unsigned start = charnum; + while (!normalizedCounter[charnum]) charnum++; + while (charnum >= start+24) { + start+=24; + bitStream += 0xFFFFU << bitCount; + if ((!writeIsSafe) && (out > oend-2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ + out[0] = (BYTE) bitStream; + out[1] = (BYTE)(bitStream>>8); + out+=2; + bitStream>>=16; + } + while (charnum >= start+3) { + start+=3; + bitStream += 3 << bitCount; + bitCount += 2; + } + bitStream += (charnum-start) << bitCount; + bitCount += 2; + if (bitCount>16) { + if ((!writeIsSafe) && (out > oend - 2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ + out[0] = (BYTE)bitStream; + out[1] = (BYTE)(bitStream>>8); + out += 2; + bitStream >>= 16; + bitCount -= 16; + } } + { int count = normalizedCounter[charnum++]; + int const max = (2*threshold-1)-remaining; + remaining -= count < 0 ? -count : count; + count++; /* +1 for extra accuracy */ + if (count>=threshold) count += max; /* [0..max[ [max..threshold[ (...) [threshold+max 2*threshold[ */ + bitStream += count << bitCount; + bitCount += nbBits; + bitCount -= (count>=1; + } + if (bitCount>16) { + if ((!writeIsSafe) && (out > oend - 2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ + out[0] = (BYTE)bitStream; + out[1] = (BYTE)(bitStream>>8); + out += 2; + bitStream >>= 16; + bitCount -= 16; + } } - /* flush remaining bitStream */ - if ((!writeIsSafe) && (out > oend - 2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ - out[0] = (BYTE)bitStream; - out[1] = (BYTE)(bitStream>>8); - out+= (bitCount+7) /8; + /* flush remaining bitStream */ + if ((!writeIsSafe) && (out > oend - 2)) return ERROR(dstSize_tooSmall); /* Buffer overflow */ + out[0] = (BYTE)bitStream; + out[1] = (BYTE)(bitStream>>8); + out+= (bitCount+7) /8; - if (charnum > maxSymbolValue + 1) return ERROR(GENERIC); + if (charnum > maxSymbolValue + 1) return ERROR(GENERIC); - return (out-ostart); + return (out-ostart); } size_t FSE_writeNCount (void* buffer, size_t bufferSize, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog) { - if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); /* Unsupported */ - if (tableLog < FSE_MIN_TABLELOG) return ERROR(GENERIC); /* Unsupported */ + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); /* Unsupported */ + if (tableLog < FSE_MIN_TABLELOG) return ERROR(GENERIC); /* Unsupported */ - if (bufferSize < FSE_NCountWriteBound(maxSymbolValue, tableLog)) - return FSE_writeNCount_generic(buffer, bufferSize, normalizedCounter, maxSymbolValue, tableLog, 0); + if (bufferSize < FSE_NCountWriteBound(maxSymbolValue, tableLog)) + return FSE_writeNCount_generic(buffer, bufferSize, normalizedCounter, maxSymbolValue, tableLog, 0); - return FSE_writeNCount_generic(buffer, bufferSize, normalizedCounter, maxSymbolValue, tableLog, 1); + return FSE_writeNCount_generic(buffer, bufferSize, normalizedCounter, maxSymbolValue, tableLog, 1); } @@ -306,31 +306,31 @@ size_t FSE_writeNCount (void* buffer, size_t bufferSize, const short* normalized * Counting histogram ****************************************************************/ /*! FSE_count_simple - This function counts byte values within `src`, and store the histogram into table `count`. - It doesn't use any additional memory. - But this function is unsafe : it doesn't check that all values within `src` can fit into `count`. - For this reason, prefer using a table `count` with 256 elements. - @return : count of most numerous element + This function counts byte values within `src`, and store the histogram into table `count`. + It doesn't use any additional memory. + But this function is unsafe : it doesn't check that all values within `src` can fit into `count`. + For this reason, prefer using a table `count` with 256 elements. + @return : count of most numerous element */ size_t FSE_count_simple(unsigned* count, unsigned* maxSymbolValuePtr, - const void* src, size_t srcSize) + const void* src, size_t srcSize) { - const BYTE* ip = (const BYTE*)src; - const BYTE* const end = ip + srcSize; - unsigned maxSymbolValue = *maxSymbolValuePtr; - unsigned max=0; + const BYTE* ip = (const BYTE*)src; + const BYTE* const end = ip + srcSize; + unsigned maxSymbolValue = *maxSymbolValuePtr; + unsigned max=0; - memset(count, 0, (maxSymbolValue+1)*sizeof(*count)); - if (srcSize==0) { *maxSymbolValuePtr = 0; return 0; } + memset(count, 0, (maxSymbolValue+1)*sizeof(*count)); + if (srcSize==0) { *maxSymbolValuePtr = 0; return 0; } - while (ip max) max = count[s]; } + { U32 s; for (s=0; s<=maxSymbolValue; s++) if (count[s] > max) max = count[s]; } - return (size_t)max; + return (size_t)max; } @@ -338,110 +338,110 @@ size_t FSE_count_simple(unsigned* count, unsigned* maxSymbolValuePtr, * Same as FSE_count_parallel(), but using an externally provided scratch buffer. * `workSpace` size must be a minimum of `1024 * sizeof(unsigned)`` */ static size_t FSE_count_parallel_wksp( - unsigned* count, unsigned* maxSymbolValuePtr, - const void* source, size_t sourceSize, - unsigned checkMax, unsigned* const workSpace) + unsigned* count, unsigned* maxSymbolValuePtr, + const void* source, size_t sourceSize, + unsigned checkMax, unsigned* const workSpace) { - const BYTE* ip = (const BYTE*)source; - const BYTE* const iend = ip+sourceSize; - unsigned maxSymbolValue = *maxSymbolValuePtr; - unsigned max=0; - U32* const Counting1 = workSpace; - U32* const Counting2 = Counting1 + 256; - U32* const Counting3 = Counting2 + 256; - U32* const Counting4 = Counting3 + 256; + const BYTE* ip = (const BYTE*)source; + const BYTE* const iend = ip+sourceSize; + unsigned maxSymbolValue = *maxSymbolValuePtr; + unsigned max=0; + U32* const Counting1 = workSpace; + U32* const Counting2 = Counting1 + 256; + U32* const Counting3 = Counting2 + 256; + U32* const Counting4 = Counting3 + 256; - memset(Counting1, 0, 4*256*sizeof(unsigned)); + memset(Counting1, 0, 4*256*sizeof(unsigned)); - /* safety checks */ - if (!sourceSize) { - memset(count, 0, maxSymbolValue + 1); - *maxSymbolValuePtr = 0; - return 0; - } - if (!maxSymbolValue) maxSymbolValue = 255; /* 0 == default */ + /* safety checks */ + if (!sourceSize) { + memset(count, 0, maxSymbolValue + 1); + *maxSymbolValuePtr = 0; + return 0; + } + if (!maxSymbolValue) maxSymbolValue = 255; /* 0 == default */ - /* by stripes of 16 bytes */ - { U32 cached = MEM_read32(ip); ip += 4; - while (ip < iend-15) { - U32 c = cached; cached = MEM_read32(ip); ip += 4; - Counting1[(BYTE) c ]++; - Counting2[(BYTE)(c>>8) ]++; - Counting3[(BYTE)(c>>16)]++; - Counting4[ c>>24 ]++; - c = cached; cached = MEM_read32(ip); ip += 4; - Counting1[(BYTE) c ]++; - Counting2[(BYTE)(c>>8) ]++; - Counting3[(BYTE)(c>>16)]++; - Counting4[ c>>24 ]++; - c = cached; cached = MEM_read32(ip); ip += 4; - Counting1[(BYTE) c ]++; - Counting2[(BYTE)(c>>8) ]++; - Counting3[(BYTE)(c>>16)]++; - Counting4[ c>>24 ]++; - c = cached; cached = MEM_read32(ip); ip += 4; - Counting1[(BYTE) c ]++; - Counting2[(BYTE)(c>>8) ]++; - Counting3[(BYTE)(c>>16)]++; - Counting4[ c>>24 ]++; - } - ip-=4; - } + /* by stripes of 16 bytes */ + { U32 cached = MEM_read32(ip); ip += 4; + while (ip < iend-15) { + U32 c = cached; cached = MEM_read32(ip); ip += 4; + Counting1[(BYTE) c ]++; + Counting2[(BYTE)(c>>8) ]++; + Counting3[(BYTE)(c>>16)]++; + Counting4[ c>>24 ]++; + c = cached; cached = MEM_read32(ip); ip += 4; + Counting1[(BYTE) c ]++; + Counting2[(BYTE)(c>>8) ]++; + Counting3[(BYTE)(c>>16)]++; + Counting4[ c>>24 ]++; + c = cached; cached = MEM_read32(ip); ip += 4; + Counting1[(BYTE) c ]++; + Counting2[(BYTE)(c>>8) ]++; + Counting3[(BYTE)(c>>16)]++; + Counting4[ c>>24 ]++; + c = cached; cached = MEM_read32(ip); ip += 4; + Counting1[(BYTE) c ]++; + Counting2[(BYTE)(c>>8) ]++; + Counting3[(BYTE)(c>>16)]++; + Counting4[ c>>24 ]++; + } + ip-=4; + } - /* finish last symbols */ - while (ipmaxSymbolValue; s--) { - Counting1[s] += Counting2[s] + Counting3[s] + Counting4[s]; - if (Counting1[s]) return ERROR(maxSymbolValue_tooSmall); - } } + if (checkMax) { /* verify stats will fit into destination table */ + U32 s; for (s=255; s>maxSymbolValue; s--) { + Counting1[s] += Counting2[s] + Counting3[s] + Counting4[s]; + if (Counting1[s]) return ERROR(maxSymbolValue_tooSmall); + } } - { U32 s; for (s=0; s<=maxSymbolValue; s++) { - count[s] = Counting1[s] + Counting2[s] + Counting3[s] + Counting4[s]; - if (count[s] > max) max = count[s]; - } } + { U32 s; for (s=0; s<=maxSymbolValue; s++) { + count[s] = Counting1[s] + Counting2[s] + Counting3[s] + Counting4[s]; + if (count[s] > max) max = count[s]; + } } - while (!count[maxSymbolValue]) maxSymbolValue--; - *maxSymbolValuePtr = maxSymbolValue; - return (size_t)max; + while (!count[maxSymbolValue]) maxSymbolValue--; + *maxSymbolValuePtr = maxSymbolValue; + return (size_t)max; } /* FSE_countFast_wksp() : * Same as FSE_countFast(), but using an externally provided scratch buffer. * `workSpace` size must be table of >= `1024` unsigned */ size_t FSE_countFast_wksp(unsigned* count, unsigned* maxSymbolValuePtr, - const void* source, size_t sourceSize, unsigned* workSpace) + const void* source, size_t sourceSize, unsigned* workSpace) { - if (sourceSize < 1500) return FSE_count_simple(count, maxSymbolValuePtr, source, sourceSize); - return FSE_count_parallel_wksp(count, maxSymbolValuePtr, source, sourceSize, 0, workSpace); + if (sourceSize < 1500) return FSE_count_simple(count, maxSymbolValuePtr, source, sourceSize); + return FSE_count_parallel_wksp(count, maxSymbolValuePtr, source, sourceSize, 0, workSpace); } /* fast variant (unsafe : won't check if src contains values beyond count[] limit) */ size_t FSE_countFast(unsigned* count, unsigned* maxSymbolValuePtr, - const void* source, size_t sourceSize) + const void* source, size_t sourceSize) { - unsigned tmpCounters[1024]; - return FSE_countFast_wksp(count, maxSymbolValuePtr, source, sourceSize, tmpCounters); + unsigned tmpCounters[1024]; + return FSE_countFast_wksp(count, maxSymbolValuePtr, source, sourceSize, tmpCounters); } /* FSE_count_wksp() : * Same as FSE_count(), but using an externally provided scratch buffer. * `workSpace` size must be table of >= `1024` unsigned */ size_t FSE_count_wksp(unsigned* count, unsigned* maxSymbolValuePtr, - const void* source, size_t sourceSize, unsigned* workSpace) + const void* source, size_t sourceSize, unsigned* workSpace) { - if (*maxSymbolValuePtr < 255) - return FSE_count_parallel_wksp(count, maxSymbolValuePtr, source, sourceSize, 1, workSpace); - *maxSymbolValuePtr = 255; - return FSE_countFast_wksp(count, maxSymbolValuePtr, source, sourceSize, workSpace); + if (*maxSymbolValuePtr < 255) + return FSE_count_parallel_wksp(count, maxSymbolValuePtr, source, sourceSize, 1, workSpace); + *maxSymbolValuePtr = 255; + return FSE_countFast_wksp(count, maxSymbolValuePtr, source, sourceSize, workSpace); } size_t FSE_count(unsigned* count, unsigned* maxSymbolValuePtr, - const void* src, size_t srcSize) + const void* src, size_t srcSize) { - unsigned tmpCounters[1024]; - return FSE_count_wksp(count, maxSymbolValuePtr, src, srcSize, tmpCounters); + unsigned tmpCounters[1024]; + return FSE_count_wksp(count, maxSymbolValuePtr, src, srcSize, tmpCounters); } @@ -450,25 +450,25 @@ size_t FSE_count(unsigned* count, unsigned* maxSymbolValuePtr, * FSE Compression Code ****************************************************************/ /*! FSE_sizeof_CTable() : - FSE_CTable is a variable size structure which contains : - `U16 tableLog;` - `U16 maxSymbolValue;` - `U16 nextStateNumber[1 << tableLog];` // This size is variable - `FSE_symbolCompressionTransform symbolTT[maxSymbolValue+1];` // This size is variable + FSE_CTable is a variable size structure which contains : + `U16 tableLog;` + `U16 maxSymbolValue;` + `U16 nextStateNumber[1 << tableLog];` // This size is variable + `FSE_symbolCompressionTransform symbolTT[maxSymbolValue+1];` // This size is variable Allocation is manual (C standard does not support variable-size structures). */ size_t FSE_sizeof_CTable (unsigned maxSymbolValue, unsigned tableLog) { - if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); - return FSE_CTABLE_SIZE_U32 (tableLog, maxSymbolValue) * sizeof(U32); + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); + return FSE_CTABLE_SIZE_U32 (tableLog, maxSymbolValue) * sizeof(U32); } FSE_CTable* FSE_createCTable (unsigned maxSymbolValue, unsigned tableLog) { - size_t size; - if (tableLog > FSE_TABLELOG_ABSOLUTE_MAX) tableLog = FSE_TABLELOG_ABSOLUTE_MAX; - size = FSE_CTABLE_SIZE_U32 (tableLog, maxSymbolValue) * sizeof(U32); - return (FSE_CTable*)malloc(size); + size_t size; + if (tableLog > FSE_TABLELOG_ABSOLUTE_MAX) tableLog = FSE_TABLELOG_ABSOLUTE_MAX; + size = FSE_CTABLE_SIZE_U32 (tableLog, maxSymbolValue) * sizeof(U32); + return (FSE_CTable*)malloc(size); } void FSE_freeCTable (FSE_CTable* ct) { free(ct); } @@ -476,28 +476,28 @@ void FSE_freeCTable (FSE_CTable* ct) { free(ct); } /* provides the minimum logSize to safely represent a distribution */ static unsigned FSE_minTableLog(size_t srcSize, unsigned maxSymbolValue) { - U32 minBitsSrc = BIT_highbit32((U32)(srcSize - 1)) + 1; - U32 minBitsSymbols = BIT_highbit32(maxSymbolValue) + 2; - U32 minBits = minBitsSrc < minBitsSymbols ? minBitsSrc : minBitsSymbols; - return minBits; + U32 minBitsSrc = BIT_highbit32((U32)(srcSize - 1)) + 1; + U32 minBitsSymbols = BIT_highbit32(maxSymbolValue) + 2; + U32 minBits = minBitsSrc < minBitsSymbols ? minBitsSrc : minBitsSymbols; + return minBits; } unsigned FSE_optimalTableLog_internal(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue, unsigned minus) { - U32 maxBitsSrc = BIT_highbit32((U32)(srcSize - 1)) - minus; - U32 tableLog = maxTableLog; - U32 minBits = FSE_minTableLog(srcSize, maxSymbolValue); - if (tableLog==0) tableLog = FSE_DEFAULT_TABLELOG; - if (maxBitsSrc < tableLog) tableLog = maxBitsSrc; /* Accuracy can be reduced */ - if (minBits > tableLog) tableLog = minBits; /* Need a minimum to safely represent all symbol values */ - if (tableLog < FSE_MIN_TABLELOG) tableLog = FSE_MIN_TABLELOG; - if (tableLog > FSE_MAX_TABLELOG) tableLog = FSE_MAX_TABLELOG; - return tableLog; + U32 maxBitsSrc = BIT_highbit32((U32)(srcSize - 1)) - minus; + U32 tableLog = maxTableLog; + U32 minBits = FSE_minTableLog(srcSize, maxSymbolValue); + if (tableLog==0) tableLog = FSE_DEFAULT_TABLELOG; + if (maxBitsSrc < tableLog) tableLog = maxBitsSrc; /* Accuracy can be reduced */ + if (minBits > tableLog) tableLog = minBits; /* Need a minimum to safely represent all symbol values */ + if (tableLog < FSE_MIN_TABLELOG) tableLog = FSE_MIN_TABLELOG; + if (tableLog > FSE_MAX_TABLELOG) tableLog = FSE_MAX_TABLELOG; + return tableLog; } unsigned FSE_optimalTableLog(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue) { - return FSE_optimalTableLog_internal(maxTableLog, srcSize, maxSymbolValue, 2); + return FSE_optimalTableLog_internal(maxTableLog, srcSize, maxSymbolValue, 2); } @@ -506,276 +506,276 @@ unsigned FSE_optimalTableLog(unsigned maxTableLog, size_t srcSize, unsigned maxS static size_t FSE_normalizeM2(short* norm, U32 tableLog, const unsigned* count, size_t total, U32 maxSymbolValue) { - short const NOT_YET_ASSIGNED = -2; - U32 s; - U32 distributed = 0; - U32 ToDistribute; + short const NOT_YET_ASSIGNED = -2; + U32 s; + U32 distributed = 0; + U32 ToDistribute; - /* Init */ - U32 const lowThreshold = (U32)(total >> tableLog); - U32 lowOne = (U32)((total * 3) >> (tableLog + 1)); + /* Init */ + U32 const lowThreshold = (U32)(total >> tableLog); + U32 lowOne = (U32)((total * 3) >> (tableLog + 1)); - for (s=0; s<=maxSymbolValue; s++) { - if (count[s] == 0) { - norm[s]=0; - continue; - } - if (count[s] <= lowThreshold) { - norm[s] = -1; - distributed++; - total -= count[s]; - continue; - } - if (count[s] <= lowOne) { - norm[s] = 1; - distributed++; - total -= count[s]; - continue; - } + for (s=0; s<=maxSymbolValue; s++) { + if (count[s] == 0) { + norm[s]=0; + continue; + } + if (count[s] <= lowThreshold) { + norm[s] = -1; + distributed++; + total -= count[s]; + continue; + } + if (count[s] <= lowOne) { + norm[s] = 1; + distributed++; + total -= count[s]; + continue; + } - norm[s]=NOT_YET_ASSIGNED; - } - ToDistribute = (1 << tableLog) - distributed; + norm[s]=NOT_YET_ASSIGNED; + } + ToDistribute = (1 << tableLog) - distributed; - if ((total / ToDistribute) > lowOne) { - /* risk of rounding to zero */ - lowOne = (U32)((total * 3) / (ToDistribute * 2)); - for (s=0; s<=maxSymbolValue; s++) { - if ((norm[s] == NOT_YET_ASSIGNED) && (count[s] <= lowOne)) { - norm[s] = 1; - distributed++; - total -= count[s]; - continue; - } } - ToDistribute = (1 << tableLog) - distributed; - } + if ((total / ToDistribute) > lowOne) { + /* risk of rounding to zero */ + lowOne = (U32)((total * 3) / (ToDistribute * 2)); + for (s=0; s<=maxSymbolValue; s++) { + if ((norm[s] == NOT_YET_ASSIGNED) && (count[s] <= lowOne)) { + norm[s] = 1; + distributed++; + total -= count[s]; + continue; + } } + ToDistribute = (1 << tableLog) - distributed; + } - if (distributed == maxSymbolValue+1) { - /* all values are pretty poor; - probably incompressible data (should have already been detected); - find max, then give all remaining points to max */ - U32 maxV = 0, maxC = 0; - for (s=0; s<=maxSymbolValue; s++) - if (count[s] > maxC) maxV=s, maxC=count[s]; - norm[maxV] += (short)ToDistribute; - return 0; - } + if (distributed == maxSymbolValue+1) { + /* all values are pretty poor; + probably incompressible data (should have already been detected); + find max, then give all remaining points to max */ + U32 maxV = 0, maxC = 0; + for (s=0; s<=maxSymbolValue; s++) + if (count[s] > maxC) maxV=s, maxC=count[s]; + norm[maxV] += (short)ToDistribute; + return 0; + } - if (total == 0) { - /* all of the symbols were low enough for the lowOne or lowThreshold */ - for (s=0; ToDistribute > 0; s = (s+1)%(maxSymbolValue+1)) - if (norm[s] > 0) ToDistribute--, norm[s]++; - return 0; - } + if (total == 0) { + /* all of the symbols were low enough for the lowOne or lowThreshold */ + for (s=0; ToDistribute > 0; s = (s+1)%(maxSymbolValue+1)) + if (norm[s] > 0) ToDistribute--, norm[s]++; + return 0; + } - { U64 const vStepLog = 62 - tableLog; - U64 const mid = (1ULL << (vStepLog-1)) - 1; - U64 const rStep = ((((U64)1<> vStepLog); - U32 const sEnd = (U32)(end >> vStepLog); - U32 const weight = sEnd - sStart; - if (weight < 1) - return ERROR(GENERIC); - norm[s] = (short)weight; - tmpTotal = end; - } } } + { U64 const vStepLog = 62 - tableLog; + U64 const mid = (1ULL << (vStepLog-1)) - 1; + U64 const rStep = ((((U64)1<> vStepLog); + U32 const sEnd = (U32)(end >> vStepLog); + U32 const weight = sEnd - sStart; + if (weight < 1) + return ERROR(GENERIC); + norm[s] = (short)weight; + tmpTotal = end; + } } } - return 0; + return 0; } size_t FSE_normalizeCount (short* normalizedCounter, unsigned tableLog, - const unsigned* count, size_t total, - unsigned maxSymbolValue) + const unsigned* count, size_t total, + unsigned maxSymbolValue) { - /* Sanity checks */ - if (tableLog==0) tableLog = FSE_DEFAULT_TABLELOG; - if (tableLog < FSE_MIN_TABLELOG) return ERROR(GENERIC); /* Unsupported size */ - if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); /* Unsupported size */ - if (tableLog < FSE_minTableLog(total, maxSymbolValue)) return ERROR(GENERIC); /* Too small tableLog, compression potentially impossible */ + /* Sanity checks */ + if (tableLog==0) tableLog = FSE_DEFAULT_TABLELOG; + if (tableLog < FSE_MIN_TABLELOG) return ERROR(GENERIC); /* Unsupported size */ + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); /* Unsupported size */ + if (tableLog < FSE_minTableLog(total, maxSymbolValue)) return ERROR(GENERIC); /* Too small tableLog, compression potentially impossible */ - { U32 const rtbTable[] = { 0, 473195, 504333, 520860, 550000, 700000, 750000, 830000 }; - U64 const scale = 62 - tableLog; - U64 const step = ((U64)1<<62) / total; /* <== here, one division ! */ - U64 const vStep = 1ULL<<(scale-20); - int stillToDistribute = 1<> tableLog); + { U32 const rtbTable[] = { 0, 473195, 504333, 520860, 550000, 700000, 750000, 830000 }; + U64 const scale = 62 - tableLog; + U64 const step = ((U64)1<<62) / total; /* <== here, one division ! */ + U64 const vStep = 1ULL<<(scale-20); + int stillToDistribute = 1<> tableLog); - for (s=0; s<=maxSymbolValue; s++) { - if (count[s] == total) return 0; /* rle special case */ - if (count[s] == 0) { normalizedCounter[s]=0; continue; } - if (count[s] <= lowThreshold) { - normalizedCounter[s] = -1; - stillToDistribute--; - } else { - short proba = (short)((count[s]*step) >> scale); - if (proba<8) { - U64 restToBeat = vStep * rtbTable[proba]; - proba += (count[s]*step) - ((U64)proba< restToBeat; - } - if (proba > largestP) largestP=proba, largest=s; - normalizedCounter[s] = proba; - stillToDistribute -= proba; - } } - if (-stillToDistribute >= (normalizedCounter[largest] >> 1)) { - /* corner case, need another normalization method */ - size_t const errorCode = FSE_normalizeM2(normalizedCounter, tableLog, count, total, maxSymbolValue); - if (FSE_isError(errorCode)) return errorCode; - } - else normalizedCounter[largest] += (short)stillToDistribute; - } + for (s=0; s<=maxSymbolValue; s++) { + if (count[s] == total) return 0; /* rle special case */ + if (count[s] == 0) { normalizedCounter[s]=0; continue; } + if (count[s] <= lowThreshold) { + normalizedCounter[s] = -1; + stillToDistribute--; + } else { + short proba = (short)((count[s]*step) >> scale); + if (proba<8) { + U64 restToBeat = vStep * rtbTable[proba]; + proba += (count[s]*step) - ((U64)proba< restToBeat; + } + if (proba > largestP) largestP=proba, largest=s; + normalizedCounter[s] = proba; + stillToDistribute -= proba; + } } + if (-stillToDistribute >= (normalizedCounter[largest] >> 1)) { + /* corner case, need another normalization method */ + size_t const errorCode = FSE_normalizeM2(normalizedCounter, tableLog, count, total, maxSymbolValue); + if (FSE_isError(errorCode)) return errorCode; + } + else normalizedCounter[largest] += (short)stillToDistribute; + } #if 0 - { /* Print Table (debug) */ - U32 s; - U32 nTotal = 0; - for (s=0; s<=maxSymbolValue; s++) - printf("%3i: %4i \n", s, normalizedCounter[s]); - for (s=0; s<=maxSymbolValue; s++) - nTotal += abs(normalizedCounter[s]); - if (nTotal != (1U<>1); /* assumption : tableLog >= 1 */ - FSE_symbolCompressionTransform* const symbolTT = (FSE_symbolCompressionTransform*) (FSCT); - unsigned s; + const unsigned tableSize = 1 << nbBits; + const unsigned tableMask = tableSize - 1; + const unsigned maxSymbolValue = tableMask; + void* const ptr = ct; + U16* const tableU16 = ( (U16*) ptr) + 2; + void* const FSCT = ((U32*)ptr) + 1 /* header */ + (tableSize>>1); /* assumption : tableLog >= 1 */ + FSE_symbolCompressionTransform* const symbolTT = (FSE_symbolCompressionTransform*) (FSCT); + unsigned s; - /* Sanity checks */ - if (nbBits < 1) return ERROR(GENERIC); /* min size */ + /* Sanity checks */ + if (nbBits < 1) return ERROR(GENERIC); /* min size */ - /* header */ - tableU16[-2] = (U16) nbBits; - tableU16[-1] = (U16) maxSymbolValue; + /* header */ + tableU16[-2] = (U16) nbBits; + tableU16[-1] = (U16) maxSymbolValue; - /* Build table */ - for (s=0; s FSE_MAX_TABLELOG*4+7 ) && (srcSize & 2)) { /* test bit 2 */ - FSE_encodeSymbol(&bitC, &CState2, *--ip); - FSE_encodeSymbol(&bitC, &CState1, *--ip); - FSE_FLUSHBITS(&bitC); - } + /* join to mod 4 */ + srcSize -= 2; + if ((sizeof(bitC.bitContainer)*8 > FSE_MAX_TABLELOG*4+7 ) && (srcSize & 2)) { /* test bit 2 */ + FSE_encodeSymbol(&bitC, &CState2, *--ip); + FSE_encodeSymbol(&bitC, &CState1, *--ip); + FSE_FLUSHBITS(&bitC); + } - /* 2 or 4 encoding per loop */ - while ( ip>istart ) { + /* 2 or 4 encoding per loop */ + while ( ip>istart ) { - FSE_encodeSymbol(&bitC, &CState2, *--ip); + FSE_encodeSymbol(&bitC, &CState2, *--ip); - if (sizeof(bitC.bitContainer)*8 < FSE_MAX_TABLELOG*2+7 ) /* this test must be static */ - FSE_FLUSHBITS(&bitC); + if (sizeof(bitC.bitContainer)*8 < FSE_MAX_TABLELOG*2+7 ) /* this test must be static */ + FSE_FLUSHBITS(&bitC); - FSE_encodeSymbol(&bitC, &CState1, *--ip); + FSE_encodeSymbol(&bitC, &CState1, *--ip); - if (sizeof(bitC.bitContainer)*8 > FSE_MAX_TABLELOG*4+7 ) { /* this test must be static */ - FSE_encodeSymbol(&bitC, &CState2, *--ip); - FSE_encodeSymbol(&bitC, &CState1, *--ip); - } + if (sizeof(bitC.bitContainer)*8 > FSE_MAX_TABLELOG*4+7 ) { /* this test must be static */ + FSE_encodeSymbol(&bitC, &CState2, *--ip); + FSE_encodeSymbol(&bitC, &CState1, *--ip); + } - FSE_FLUSHBITS(&bitC); - } + FSE_FLUSHBITS(&bitC); + } - FSE_flushCState(&bitC, &CState2); - FSE_flushCState(&bitC, &CState1); - return BIT_closeCStream(&bitC); + FSE_flushCState(&bitC, &CState2); + FSE_flushCState(&bitC, &CState1); + return BIT_closeCStream(&bitC); } size_t FSE_compress_usingCTable (void* dst, size_t dstSize, - const void* src, size_t srcSize, - const FSE_CTable* ct) + const void* src, size_t srcSize, + const FSE_CTable* ct) { - unsigned const fast = (dstSize >= FSE_BLOCKBOUND(srcSize)); + unsigned const fast = (dstSize >= FSE_BLOCKBOUND(srcSize)); - if (fast) - return FSE_compress_usingCTable_generic(dst, dstSize, src, srcSize, ct, 1); - else - return FSE_compress_usingCTable_generic(dst, dstSize, src, srcSize, ct, 0); + if (fast) + return FSE_compress_usingCTable_generic(dst, dstSize, src, srcSize, ct, 1); + else + return FSE_compress_usingCTable_generic(dst, dstSize, src, srcSize, ct, 0); } @@ -790,67 +790,67 @@ size_t FSE_compressBound(size_t size) { return FSE_COMPRESSBOUND(size); } */ size_t FSE_compress_wksp (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize) { - BYTE* const ostart = (BYTE*) dst; - BYTE* op = ostart; - BYTE* const oend = ostart + dstSize; + BYTE* const ostart = (BYTE*) dst; + BYTE* op = ostart; + BYTE* const oend = ostart + dstSize; - U32 count[FSE_MAX_SYMBOL_VALUE+1]; - S16 norm[FSE_MAX_SYMBOL_VALUE+1]; - FSE_CTable* CTable = (FSE_CTable*)workSpace; - size_t const CTableSize = FSE_CTABLE_SIZE_U32(tableLog, maxSymbolValue); - void* scratchBuffer = (void*)(CTable + CTableSize); - size_t const scratchBufferSize = wkspSize - (CTableSize * sizeof(FSE_CTable)); + U32 count[FSE_MAX_SYMBOL_VALUE+1]; + S16 norm[FSE_MAX_SYMBOL_VALUE+1]; + FSE_CTable* CTable = (FSE_CTable*)workSpace; + size_t const CTableSize = FSE_CTABLE_SIZE_U32(tableLog, maxSymbolValue); + void* scratchBuffer = (void*)(CTable + CTableSize); + size_t const scratchBufferSize = wkspSize - (CTableSize * sizeof(FSE_CTable)); - /* init conditions */ - if (wkspSize < FSE_WKSP_SIZE_U32(tableLog, maxSymbolValue)) return ERROR(tableLog_tooLarge); - if (srcSize <= 1) return 0; /* Not compressible */ - if (!maxSymbolValue) maxSymbolValue = FSE_MAX_SYMBOL_VALUE; - if (!tableLog) tableLog = FSE_DEFAULT_TABLELOG; + /* init conditions */ + if (wkspSize < FSE_WKSP_SIZE_U32(tableLog, maxSymbolValue)) return ERROR(tableLog_tooLarge); + if (srcSize <= 1) return 0; /* Not compressible */ + if (!maxSymbolValue) maxSymbolValue = FSE_MAX_SYMBOL_VALUE; + if (!tableLog) tableLog = FSE_DEFAULT_TABLELOG; - /* Scan input and build symbol stats */ - { CHECK_V_F(maxCount, FSE_count(count, &maxSymbolValue, src, srcSize) ); - if (maxCount == srcSize) return 1; /* only a single symbol in src : rle */ - if (maxCount == 1) return 0; /* each symbol present maximum once => not compressible */ - if (maxCount < (srcSize >> 7)) return 0; /* Heuristic : not compressible enough */ - } + /* Scan input and build symbol stats */ + { CHECK_V_F(maxCount, FSE_count(count, &maxSymbolValue, src, srcSize) ); + if (maxCount == srcSize) return 1; /* only a single symbol in src : rle */ + if (maxCount == 1) return 0; /* each symbol present maximum once => not compressible */ + if (maxCount < (srcSize >> 7)) return 0; /* Heuristic : not compressible enough */ + } - tableLog = FSE_optimalTableLog(tableLog, srcSize, maxSymbolValue); - CHECK_F( FSE_normalizeCount(norm, tableLog, count, srcSize, maxSymbolValue) ); + tableLog = FSE_optimalTableLog(tableLog, srcSize, maxSymbolValue); + CHECK_F( FSE_normalizeCount(norm, tableLog, count, srcSize, maxSymbolValue) ); - /* Write table description header */ - { CHECK_V_F(nc_err, FSE_writeNCount(op, oend-op, norm, maxSymbolValue, tableLog) ); - op += nc_err; - } + /* Write table description header */ + { CHECK_V_F(nc_err, FSE_writeNCount(op, oend-op, norm, maxSymbolValue, tableLog) ); + op += nc_err; + } - /* Compress */ - CHECK_F( FSE_buildCTable_wksp(CTable, norm, maxSymbolValue, tableLog, scratchBuffer, scratchBufferSize) ); - { CHECK_V_F(cSize, FSE_compress_usingCTable(op, oend - op, src, srcSize, CTable) ); - if (cSize == 0) return 0; /* not enough space for compressed data */ - op += cSize; - } + /* Compress */ + CHECK_F( FSE_buildCTable_wksp(CTable, norm, maxSymbolValue, tableLog, scratchBuffer, scratchBufferSize) ); + { CHECK_V_F(cSize, FSE_compress_usingCTable(op, oend - op, src, srcSize, CTable) ); + if (cSize == 0) return 0; /* not enough space for compressed data */ + op += cSize; + } - /* check compressibility */ - if ( (size_t)(op-ostart) >= srcSize-1 ) return 0; + /* check compressibility */ + if ( (size_t)(op-ostart) >= srcSize-1 ) return 0; - return op-ostart; + return op-ostart; } typedef struct { - FSE_CTable CTable_max[FSE_CTABLE_SIZE_U32(FSE_MAX_TABLELOG, FSE_MAX_SYMBOL_VALUE)]; - BYTE scratchBuffer[1 << FSE_MAX_TABLELOG]; + FSE_CTable CTable_max[FSE_CTABLE_SIZE_U32(FSE_MAX_TABLELOG, FSE_MAX_SYMBOL_VALUE)]; + BYTE scratchBuffer[1 << FSE_MAX_TABLELOG]; } fseWkspMax_t; size_t FSE_compress2 (void* dst, size_t dstCapacity, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog) { - fseWkspMax_t scratchBuffer; - FSE_STATIC_ASSERT(sizeof(scratchBuffer) >= FSE_WKSP_SIZE_U32(FSE_MAX_TABLELOG, FSE_MAX_SYMBOL_VALUE)); /* compilation failures here means scratchBuffer is not large enough */ - if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); - return FSE_compress_wksp(dst, dstCapacity, src, srcSize, maxSymbolValue, tableLog, &scratchBuffer, sizeof(scratchBuffer)); + fseWkspMax_t scratchBuffer; + FSE_STATIC_ASSERT(sizeof(scratchBuffer) >= FSE_WKSP_SIZE_U32(FSE_MAX_TABLELOG, FSE_MAX_SYMBOL_VALUE)); /* compilation failures here means scratchBuffer is not large enough */ + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); + return FSE_compress_wksp(dst, dstCapacity, src, srcSize, maxSymbolValue, tableLog, &scratchBuffer, sizeof(scratchBuffer)); } size_t FSE_compress (void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - return FSE_compress2(dst, dstCapacity, src, srcSize, FSE_MAX_SYMBOL_VALUE, FSE_DEFAULT_TABLELOG); + return FSE_compress2(dst, dstCapacity, src, srcSize, FSE_MAX_SYMBOL_VALUE, FSE_DEFAULT_TABLELOG); } diff --git a/contrib/linux-kernel/lib/fse_decompress.c b/contrib/linux-kernel/lib/fse_decompress.c index 8474a4c07..dd2dadc20 100644 --- a/contrib/linux-kernel/lib/fse_decompress.c +++ b/contrib/linux-kernel/lib/fse_decompress.c @@ -8,9 +8,9 @@ modification, are permitted provided that the following conditions are met: - * Redistributions of source code must retain the above copyright + * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - * Redistributions in binary form must reproduce the above + * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. @@ -27,9 +27,9 @@ (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - You can contact the author at : - - FSE source repository : https://github.com/Cyan4973/FiniteStateEntropy - - Public forum : https://groups.google.com/forum/#!forum/lz4c + You can contact the author at : + - FSE source repository : https://github.com/Cyan4973/FiniteStateEntropy + - Public forum : https://groups.google.com/forum/#!forum/lz4c ****************************************************************** */ @@ -100,70 +100,70 @@ /* Function templates */ FSE_DTable* FSE_createDTable (unsigned tableLog) { - if (tableLog > FSE_TABLELOG_ABSOLUTE_MAX) tableLog = FSE_TABLELOG_ABSOLUTE_MAX; - return (FSE_DTable*)malloc( FSE_DTABLE_SIZE_U32(tableLog) * sizeof (U32) ); + if (tableLog > FSE_TABLELOG_ABSOLUTE_MAX) tableLog = FSE_TABLELOG_ABSOLUTE_MAX; + return (FSE_DTable*)malloc( FSE_DTABLE_SIZE_U32(tableLog) * sizeof (U32) ); } void FSE_freeDTable (FSE_DTable* dt) { - free(dt); + free(dt); } size_t FSE_buildDTable(FSE_DTable* dt, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog) { - void* const tdPtr = dt+1; /* because *dt is unsigned, 32-bits aligned on 32-bits */ - FSE_DECODE_TYPE* const tableDecode = (FSE_DECODE_TYPE*) (tdPtr); - U16 symbolNext[FSE_MAX_SYMBOL_VALUE+1]; + void* const tdPtr = dt+1; /* because *dt is unsigned, 32-bits aligned on 32-bits */ + FSE_DECODE_TYPE* const tableDecode = (FSE_DECODE_TYPE*) (tdPtr); + U16 symbolNext[FSE_MAX_SYMBOL_VALUE+1]; - U32 const maxSV1 = maxSymbolValue + 1; - U32 const tableSize = 1 << tableLog; - U32 highThreshold = tableSize-1; + U32 const maxSV1 = maxSymbolValue + 1; + U32 const tableSize = 1 << tableLog; + U32 highThreshold = tableSize-1; - /* Sanity Checks */ - if (maxSymbolValue > FSE_MAX_SYMBOL_VALUE) return ERROR(maxSymbolValue_tooLarge); - if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); + /* Sanity Checks */ + if (maxSymbolValue > FSE_MAX_SYMBOL_VALUE) return ERROR(maxSymbolValue_tooLarge); + if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); - /* Init, lay down lowprob symbols */ - { FSE_DTableHeader DTableH; - DTableH.tableLog = (U16)tableLog; - DTableH.fastMode = 1; - { S16 const largeLimit= (S16)(1 << (tableLog-1)); - U32 s; - for (s=0; s= largeLimit) DTableH.fastMode=0; - symbolNext[s] = normalizedCounter[s]; - } } } - memcpy(dt, &DTableH, sizeof(DTableH)); - } + /* Init, lay down lowprob symbols */ + { FSE_DTableHeader DTableH; + DTableH.tableLog = (U16)tableLog; + DTableH.fastMode = 1; + { S16 const largeLimit= (S16)(1 << (tableLog-1)); + U32 s; + for (s=0; s= largeLimit) DTableH.fastMode=0; + symbolNext[s] = normalizedCounter[s]; + } } } + memcpy(dt, &DTableH, sizeof(DTableH)); + } - /* Spread symbols */ - { U32 const tableMask = tableSize-1; - U32 const step = FSE_TABLESTEP(tableSize); - U32 s, position = 0; - for (s=0; s highThreshold) position = (position + step) & tableMask; /* lowprob area */ - } } - if (position!=0) return ERROR(GENERIC); /* position must reach all cells once, otherwise normalizedCounter is incorrect */ - } + /* Spread symbols */ + { U32 const tableMask = tableSize-1; + U32 const step = FSE_TABLESTEP(tableSize); + U32 s, position = 0; + for (s=0; s highThreshold) position = (position + step) & tableMask; /* lowprob area */ + } } + if (position!=0) return ERROR(GENERIC); /* position must reach all cells once, otherwise normalizedCounter is incorrect */ + } - /* Build Decoding table */ - { U32 u; - for (u=0; utableLog = 0; - DTableH->fastMode = 0; + DTableH->tableLog = 0; + DTableH->fastMode = 0; - cell->newState = 0; - cell->symbol = symbolValue; - cell->nbBits = 0; + cell->newState = 0; + cell->symbol = symbolValue; + cell->nbBits = 0; - return 0; + return 0; } size_t FSE_buildDTable_raw (FSE_DTable* dt, unsigned nbBits) { - void* ptr = dt; - FSE_DTableHeader* const DTableH = (FSE_DTableHeader*)ptr; - void* dPtr = dt + 1; - FSE_decode_t* const dinfo = (FSE_decode_t*)dPtr; - const unsigned tableSize = 1 << nbBits; - const unsigned tableMask = tableSize - 1; - const unsigned maxSV1 = tableMask+1; - unsigned s; + void* ptr = dt; + FSE_DTableHeader* const DTableH = (FSE_DTableHeader*)ptr; + void* dPtr = dt + 1; + FSE_decode_t* const dinfo = (FSE_decode_t*)dPtr; + const unsigned tableSize = 1 << nbBits; + const unsigned tableMask = tableSize - 1; + const unsigned maxSV1 = tableMask+1; + unsigned s; - /* Sanity checks */ - if (nbBits < 1) return ERROR(GENERIC); /* min size */ + /* Sanity checks */ + if (nbBits < 1) return ERROR(GENERIC); /* min size */ - /* Build Decoding Table */ - DTableH->tableLog = (U16)nbBits; - DTableH->fastMode = 1; - for (s=0; stableLog = (U16)nbBits; + DTableH->fastMode = 1; + for (s=0; s sizeof(bitD.bitContainer)*8) /* This test must be static */ - BIT_reloadDStream(&bitD); + if (FSE_MAX_TABLELOG*2+7 > sizeof(bitD.bitContainer)*8) /* This test must be static */ + BIT_reloadDStream(&bitD); - op[1] = FSE_GETSYMBOL(&state2); + op[1] = FSE_GETSYMBOL(&state2); - if (FSE_MAX_TABLELOG*4+7 > sizeof(bitD.bitContainer)*8) /* This test must be static */ - { if (BIT_reloadDStream(&bitD) > BIT_DStream_unfinished) { op+=2; break; } } + if (FSE_MAX_TABLELOG*4+7 > sizeof(bitD.bitContainer)*8) /* This test must be static */ + { if (BIT_reloadDStream(&bitD) > BIT_DStream_unfinished) { op+=2; break; } } - op[2] = FSE_GETSYMBOL(&state1); + op[2] = FSE_GETSYMBOL(&state1); - if (FSE_MAX_TABLELOG*2+7 > sizeof(bitD.bitContainer)*8) /* This test must be static */ - BIT_reloadDStream(&bitD); + if (FSE_MAX_TABLELOG*2+7 > sizeof(bitD.bitContainer)*8) /* This test must be static */ + BIT_reloadDStream(&bitD); - op[3] = FSE_GETSYMBOL(&state2); - } + op[3] = FSE_GETSYMBOL(&state2); + } - /* tail */ - /* note : BIT_reloadDStream(&bitD) >= FSE_DStream_partiallyFilled; Ends at exactly BIT_DStream_completed */ - while (1) { - if (op>(omax-2)) return ERROR(dstSize_tooSmall); - *op++ = FSE_GETSYMBOL(&state1); - if (BIT_reloadDStream(&bitD)==BIT_DStream_overflow) { - *op++ = FSE_GETSYMBOL(&state2); - break; - } + /* tail */ + /* note : BIT_reloadDStream(&bitD) >= FSE_DStream_partiallyFilled; Ends at exactly BIT_DStream_completed */ + while (1) { + if (op>(omax-2)) return ERROR(dstSize_tooSmall); + *op++ = FSE_GETSYMBOL(&state1); + if (BIT_reloadDStream(&bitD)==BIT_DStream_overflow) { + *op++ = FSE_GETSYMBOL(&state2); + break; + } - if (op>(omax-2)) return ERROR(dstSize_tooSmall); - *op++ = FSE_GETSYMBOL(&state2); - if (BIT_reloadDStream(&bitD)==BIT_DStream_overflow) { - *op++ = FSE_GETSYMBOL(&state1); - break; - } } + if (op>(omax-2)) return ERROR(dstSize_tooSmall); + *op++ = FSE_GETSYMBOL(&state2); + if (BIT_reloadDStream(&bitD)==BIT_DStream_overflow) { + *op++ = FSE_GETSYMBOL(&state1); + break; + } } - return op-ostart; + return op-ostart; } size_t FSE_decompress_usingDTable(void* dst, size_t originalSize, - const void* cSrc, size_t cSrcSize, - const FSE_DTable* dt) + const void* cSrc, size_t cSrcSize, + const FSE_DTable* dt) { - const void* ptr = dt; - const FSE_DTableHeader* DTableH = (const FSE_DTableHeader*)ptr; - const U32 fastMode = DTableH->fastMode; + const void* ptr = dt; + const FSE_DTableHeader* DTableH = (const FSE_DTableHeader*)ptr; + const U32 fastMode = DTableH->fastMode; - /* select fast mode (static) */ - if (fastMode) return FSE_decompress_usingDTable_generic(dst, originalSize, cSrc, cSrcSize, dt, 1); - return FSE_decompress_usingDTable_generic(dst, originalSize, cSrc, cSrcSize, dt, 0); + /* select fast mode (static) */ + if (fastMode) return FSE_decompress_usingDTable_generic(dst, originalSize, cSrc, cSrcSize, dt, 1); + return FSE_decompress_usingDTable_generic(dst, originalSize, cSrc, cSrcSize, dt, 0); } size_t FSE_decompress_wksp(void* dst, size_t dstCapacity, const void* cSrc, size_t cSrcSize, FSE_DTable* workSpace, unsigned maxLog) { - const BYTE* const istart = (const BYTE*)cSrc; - const BYTE* ip = istart; - short counting[FSE_MAX_SYMBOL_VALUE+1]; - unsigned tableLog; - unsigned maxSymbolValue = FSE_MAX_SYMBOL_VALUE; + const BYTE* const istart = (const BYTE*)cSrc; + const BYTE* ip = istart; + short counting[FSE_MAX_SYMBOL_VALUE+1]; + unsigned tableLog; + unsigned maxSymbolValue = FSE_MAX_SYMBOL_VALUE; - /* normal FSE decoding mode */ - size_t const NCountLength = FSE_readNCount (counting, &maxSymbolValue, &tableLog, istart, cSrcSize); - if (FSE_isError(NCountLength)) return NCountLength; - //if (NCountLength >= cSrcSize) return ERROR(srcSize_wrong); /* too small input size; supposed to be already checked in NCountLength, only remaining case : NCountLength==cSrcSize */ - if (tableLog > maxLog) return ERROR(tableLog_tooLarge); - ip += NCountLength; - cSrcSize -= NCountLength; + /* normal FSE decoding mode */ + size_t const NCountLength = FSE_readNCount (counting, &maxSymbolValue, &tableLog, istart, cSrcSize); + if (FSE_isError(NCountLength)) return NCountLength; + //if (NCountLength >= cSrcSize) return ERROR(srcSize_wrong); /* too small input size; supposed to be already checked in NCountLength, only remaining case : NCountLength==cSrcSize */ + if (tableLog > maxLog) return ERROR(tableLog_tooLarge); + ip += NCountLength; + cSrcSize -= NCountLength; - CHECK_F( FSE_buildDTable (workSpace, counting, maxSymbolValue, tableLog) ); + CHECK_F( FSE_buildDTable (workSpace, counting, maxSymbolValue, tableLog) ); - return FSE_decompress_usingDTable (dst, dstCapacity, ip, cSrcSize, workSpace); /* always return, even if it is an error code */ + return FSE_decompress_usingDTable (dst, dstCapacity, ip, cSrcSize, workSpace); /* always return, even if it is an error code */ } @@ -319,8 +319,8 @@ typedef FSE_DTable DTable_max_t[FSE_DTABLE_SIZE_U32(FSE_MAX_TABLELOG)]; size_t FSE_decompress(void* dst, size_t dstCapacity, const void* cSrc, size_t cSrcSize) { - DTable_max_t dt; /* Static analyzer seems unable to understand this table will be properly initialized later */ - return FSE_decompress_wksp(dst, dstCapacity, cSrc, cSrcSize, dt, FSE_MAX_TABLELOG); + DTable_max_t dt; /* Static analyzer seems unable to understand this table will be properly initialized later */ + return FSE_decompress_wksp(dst, dstCapacity, cSrc, cSrcSize, dt, FSE_MAX_TABLELOG); } diff --git a/contrib/linux-kernel/lib/huf.h b/contrib/linux-kernel/lib/huf.h index e5572760a..8ad3e2b42 100644 --- a/contrib/linux-kernel/lib/huf.h +++ b/contrib/linux-kernel/lib/huf.h @@ -9,9 +9,9 @@ modification, are permitted provided that the following conditions are met: - * Redistributions of source code must retain the above copyright + * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - * Redistributions in binary form must reproduce the above + * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. @@ -46,31 +46,31 @@ extern "C" { /* *** simple functions *** */ /** HUF_compress() : - Compress content from buffer 'src', of size 'srcSize', into buffer 'dst'. - 'dst' buffer must be already allocated. - Compression runs faster if `dstCapacity` >= HUF_compressBound(srcSize). - `srcSize` must be <= `HUF_BLOCKSIZE_MAX` == 128 KB. - @return : size of compressed data (<= `dstCapacity`). - Special values : if return == 0, srcData is not compressible => Nothing is stored within dst !!! - if return == 1, srcData is a single repeated byte symbol (RLE compression). - if HUF_isError(return), compression failed (more details using HUF_getErrorName()) + Compress content from buffer 'src', of size 'srcSize', into buffer 'dst'. + 'dst' buffer must be already allocated. + Compression runs faster if `dstCapacity` >= HUF_compressBound(srcSize). + `srcSize` must be <= `HUF_BLOCKSIZE_MAX` == 128 KB. + @return : size of compressed data (<= `dstCapacity`). + Special values : if return == 0, srcData is not compressible => Nothing is stored within dst !!! + if return == 1, srcData is a single repeated byte symbol (RLE compression). + if HUF_isError(return), compression failed (more details using HUF_getErrorName()) */ size_t HUF_compress(void* dst, size_t dstCapacity, - const void* src, size_t srcSize); + const void* src, size_t srcSize); /** HUF_decompress() : - Decompress HUF data from buffer 'cSrc', of size 'cSrcSize', - into already allocated buffer 'dst', of minimum size 'dstSize'. - `originalSize` : **must** be the ***exact*** size of original (uncompressed) data. - Note : in contrast with FSE, HUF_decompress can regenerate - RLE (cSrcSize==1) and uncompressed (cSrcSize==dstSize) data, - because it knows size to regenerate. - @return : size of regenerated data (== originalSize), - or an error code, which can be tested using HUF_isError() + Decompress HUF data from buffer 'cSrc', of size 'cSrcSize', + into already allocated buffer 'dst', of minimum size 'dstSize'. + `originalSize` : **must** be the ***exact*** size of original (uncompressed) data. + Note : in contrast with FSE, HUF_decompress can regenerate + RLE (cSrcSize==1) and uncompressed (cSrcSize==dstSize) data, + because it knows size to regenerate. + @return : size of regenerated data (== originalSize), + or an error code, which can be tested using HUF_isError() */ size_t HUF_decompress(void* dst, size_t originalSize, - const void* cSrc, size_t cSrcSize); + const void* cSrc, size_t cSrcSize); /* *** Tool functions *** */ @@ -122,17 +122,17 @@ size_t HUF_compress4X_wksp (void* dst, size_t dstSize, const void* src, size_t s /* static allocation of HUF's Compression Table */ #define HUF_CREATE_STATIC_CTABLE(name, maxSymbolValue) \ - U32 name##hb[maxSymbolValue+1]; \ - void* name##hv = &(name##hb); \ - HUF_CElt* name = (HUF_CElt*)(name##hv) /* no final ; */ + U32 name##hb[maxSymbolValue+1]; \ + void* name##hv = &(name##hb); \ + HUF_CElt* name = (HUF_CElt*)(name##hv) /* no final ; */ /* static allocation of HUF's DTable */ typedef U32 HUF_DTable; #define HUF_DTABLE_SIZE(maxTableLog) (1 + (1<<(maxTableLog))) #define HUF_CREATE_STATIC_DTABLEX2(DTable, maxTableLog) \ - HUF_DTable DTable[HUF_DTABLE_SIZE((maxTableLog)-1)] = { ((U32)((maxTableLog)-1) * 0x01000001) } + HUF_DTable DTable[HUF_DTABLE_SIZE((maxTableLog)-1)] = { ((U32)((maxTableLog)-1) * 0x01000001) } #define HUF_CREATE_STATIC_DTABLEX4(DTable, maxTableLog) \ - HUF_DTable DTable[HUF_DTABLE_SIZE(maxTableLog)] = { ((U32)(maxTableLog) * 0x01000001) } + HUF_DTable DTable[HUF_DTABLE_SIZE(maxTableLog)] = { ((U32)(maxTableLog) * 0x01000001) } /* The workspace must have alignment at least 4 and be at least this large */ #define HUF_WORKSPACE_SIZE (6 << 10) @@ -192,13 +192,13 @@ size_t HUF_compress4X_repeat(void* dst, size_t dstSize, const void* src, size_t size_t HUF_buildCTable_wksp (HUF_CElt* tree, const U32* count, U32 maxSymbolValue, U32 maxNbBits, void* workSpace, size_t wkspSize); /*! HUF_readStats() : - Read compact Huffman tree, saved by HUF_writeCTable(). - `huffWeight` is destination buffer. - @return : size read from `src` , or an error Code . - Note : Needed by HUF_readCTable() and HUF_readDTableXn() . */ + Read compact Huffman tree, saved by HUF_writeCTable(). + `huffWeight` is destination buffer. + @return : size read from `src` , or an error Code . + Note : Needed by HUF_readCTable() and HUF_readDTableXn() . */ size_t HUF_readStats(BYTE* huffWeight, size_t hwSize, U32* rankStats, - U32* nbSymbolsPtr, U32* tableLogPtr, - const void* src, size_t srcSize); + U32* nbSymbolsPtr, U32* tableLogPtr, + const void* src, size_t srcSize); /** HUF_readCTable() : * Loading a CTable saved with HUF_writeCTable() */ diff --git a/contrib/linux-kernel/lib/huf_compress.c b/contrib/linux-kernel/lib/huf_compress.c index fe11aafb8..4a6ea9963 100644 --- a/contrib/linux-kernel/lib/huf_compress.c +++ b/contrib/linux-kernel/lib/huf_compress.c @@ -8,9 +8,9 @@ modification, are permitted provided that the following conditions are met: - * Redistributions of source code must retain the above copyright + * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - * Redistributions in binary form must reproduce the above + * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. @@ -27,9 +27,9 @@ (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - You can contact the author at : - - FSE+HUF source repository : https://github.com/Cyan4973/FiniteStateEntropy - - Public forum : https://groups.google.com/forum/#!forum/lz4c + You can contact the author at : + - FSE+HUF source repository : https://github.com/Cyan4973/FiniteStateEntropy + - Public forum : https://groups.google.com/forum/#!forum/lz4c ****************************************************************** */ /* ************************************************************** @@ -65,7 +65,7 @@ ****************************************************************/ unsigned HUF_optimalTableLog(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue) { - return FSE_optimalTableLog_internal(maxTableLog, srcSize, maxSymbolValue, 1); + return FSE_optimalTableLog_internal(maxTableLog, srcSize, maxSymbolValue, 1); } @@ -80,44 +80,44 @@ unsigned HUF_optimalTableLog(unsigned maxTableLog, size_t srcSize, unsigned maxS #define MAX_FSE_TABLELOG_FOR_HUFF_HEADER 6 size_t HUF_compressWeights (void* dst, size_t dstSize, const void* weightTable, size_t wtSize) { - BYTE* const ostart = (BYTE*) dst; - BYTE* op = ostart; - BYTE* const oend = ostart + dstSize; + BYTE* const ostart = (BYTE*) dst; + BYTE* op = ostart; + BYTE* const oend = ostart + dstSize; - U32 maxSymbolValue = HUF_TABLELOG_MAX; - U32 tableLog = MAX_FSE_TABLELOG_FOR_HUFF_HEADER; + U32 maxSymbolValue = HUF_TABLELOG_MAX; + U32 tableLog = MAX_FSE_TABLELOG_FOR_HUFF_HEADER; - FSE_CTable CTable[FSE_CTABLE_SIZE_U32(MAX_FSE_TABLELOG_FOR_HUFF_HEADER, HUF_TABLELOG_MAX)]; - BYTE scratchBuffer[1< not compressible */ - } + /* Scan input and build symbol stats */ + { CHECK_V_F(maxCount, FSE_count_simple(count, &maxSymbolValue, weightTable, wtSize) ); + if (maxCount == wtSize) return 1; /* only a single symbol in src : rle */ + if (maxCount == 1) return 0; /* each symbol present maximum once => not compressible */ + } - tableLog = FSE_optimalTableLog(tableLog, wtSize, maxSymbolValue); - CHECK_F( FSE_normalizeCount(norm, tableLog, count, wtSize, maxSymbolValue) ); + tableLog = FSE_optimalTableLog(tableLog, wtSize, maxSymbolValue); + CHECK_F( FSE_normalizeCount(norm, tableLog, count, wtSize, maxSymbolValue) ); - /* Write table description header */ - { CHECK_V_F(hSize, FSE_writeNCount(op, oend-op, norm, maxSymbolValue, tableLog) ); - op += hSize; - } + /* Write table description header */ + { CHECK_V_F(hSize, FSE_writeNCount(op, oend-op, norm, maxSymbolValue, tableLog) ); + op += hSize; + } - /* Compress */ - CHECK_F( FSE_buildCTable_wksp(CTable, norm, maxSymbolValue, tableLog, scratchBuffer, sizeof(scratchBuffer)) ); - { CHECK_V_F(cSize, FSE_compress_usingCTable(op, oend - op, weightTable, wtSize, CTable) ); - if (cSize == 0) return 0; /* not enough space for compressed data */ - op += cSize; - } + /* Compress */ + CHECK_F( FSE_buildCTable_wksp(CTable, norm, maxSymbolValue, tableLog, scratchBuffer, sizeof(scratchBuffer)) ); + { CHECK_V_F(cSize, FSE_compress_usingCTable(op, oend - op, weightTable, wtSize, CTable) ); + if (cSize == 0) return 0; /* not enough space for compressed data */ + op += cSize; + } - return op-ostart; + return op-ostart; } @@ -127,201 +127,201 @@ struct HUF_CElt_s { }; /* typedef'd to HUF_CElt within "huf.h" */ /*! HUF_writeCTable() : - `CTable` : Huffman tree to save, using huf representation. - @return : size of saved CTable */ + `CTable` : Huffman tree to save, using huf representation. + @return : size of saved CTable */ size_t HUF_writeCTable (void* dst, size_t maxDstSize, - const HUF_CElt* CTable, U32 maxSymbolValue, U32 huffLog) + const HUF_CElt* CTable, U32 maxSymbolValue, U32 huffLog) { - BYTE bitsToWeight[HUF_TABLELOG_MAX + 1]; /* precomputed conversion table */ - BYTE huffWeight[HUF_SYMBOLVALUE_MAX]; - BYTE* op = (BYTE*)dst; - U32 n; + BYTE bitsToWeight[HUF_TABLELOG_MAX + 1]; /* precomputed conversion table */ + BYTE huffWeight[HUF_SYMBOLVALUE_MAX]; + BYTE* op = (BYTE*)dst; + U32 n; - /* check conditions */ - if (maxSymbolValue > HUF_SYMBOLVALUE_MAX) return ERROR(maxSymbolValue_tooLarge); + /* check conditions */ + if (maxSymbolValue > HUF_SYMBOLVALUE_MAX) return ERROR(maxSymbolValue_tooLarge); - /* convert to weight */ - bitsToWeight[0] = 0; - for (n=1; n1) & (hSize < maxSymbolValue/2)) { /* FSE compressed */ - op[0] = (BYTE)hSize; - return hSize+1; - } } + /* attempt weights compression by FSE */ + { CHECK_V_F(hSize, HUF_compressWeights(op+1, maxDstSize-1, huffWeight, maxSymbolValue) ); + if ((hSize>1) & (hSize < maxSymbolValue/2)) { /* FSE compressed */ + op[0] = (BYTE)hSize; + return hSize+1; + } } - /* write raw values as 4-bits (max : 15) */ - if (maxSymbolValue > (256-128)) return ERROR(GENERIC); /* should not happen : likely means source cannot be compressed */ - if (((maxSymbolValue+1)/2) + 1 > maxDstSize) return ERROR(dstSize_tooSmall); /* not enough space within dst buffer */ - op[0] = (BYTE)(128 /*special case*/ + (maxSymbolValue-1)); - huffWeight[maxSymbolValue] = 0; /* to be sure it doesn't cause msan issue in final combination */ - for (n=0; n (256-128)) return ERROR(GENERIC); /* should not happen : likely means source cannot be compressed */ + if (((maxSymbolValue+1)/2) + 1 > maxDstSize) return ERROR(dstSize_tooSmall); /* not enough space within dst buffer */ + op[0] = (BYTE)(128 /*special case*/ + (maxSymbolValue-1)); + huffWeight[maxSymbolValue] = 0; /* to be sure it doesn't cause msan issue in final combination */ + for (n=0; n HUF_TABLELOG_MAX) return ERROR(tableLog_tooLarge); - if (nbSymbols > maxSymbolValue+1) return ERROR(maxSymbolValue_tooSmall); + /* check result */ + if (tableLog > HUF_TABLELOG_MAX) return ERROR(tableLog_tooLarge); + if (nbSymbols > maxSymbolValue+1) return ERROR(maxSymbolValue_tooSmall); - /* Prepare base value per rank */ - { U32 n, nextRankStart = 0; - for (n=1; n<=tableLog; n++) { - U32 current = nextRankStart; - nextRankStart += (rankVal[n] << (n-1)); - rankVal[n] = current; - } } + /* Prepare base value per rank */ + { U32 n, nextRankStart = 0; + for (n=1; n<=tableLog; n++) { + U32 current = nextRankStart; + nextRankStart += (rankVal[n] << (n-1)); + rankVal[n] = current; + } } - /* fill nbBits */ - { U32 n; for (n=0; nn=tableLog+1 */ - U16 valPerRank[HUF_TABLELOG_MAX+2] = {0}; - { U32 n; for (n=0; n0; n--) { /* start at n=tablelog <-> w=1 */ - valPerRank[n] = min; /* get starting value within each rank */ - min += nbPerRank[n]; - min >>= 1; - } } - /* assign value within rank, symbol order */ - { U32 n; for (n=0; n<=maxSymbolValue; n++) CTable[n].val = valPerRank[CTable[n].nbBits]++; } - } + /* fill val */ + { U16 nbPerRank[HUF_TABLELOG_MAX+2] = {0}; /* support w=0=>n=tableLog+1 */ + U16 valPerRank[HUF_TABLELOG_MAX+2] = {0}; + { U32 n; for (n=0; n0; n--) { /* start at n=tablelog <-> w=1 */ + valPerRank[n] = min; /* get starting value within each rank */ + min += nbPerRank[n]; + min >>= 1; + } } + /* assign value within rank, symbol order */ + { U32 n; for (n=0; n<=maxSymbolValue; n++) CTable[n].val = valPerRank[CTable[n].nbBits]++; } + } - return readSize; + return readSize; } typedef struct nodeElt_s { - U32 count; - U16 parent; - BYTE byte; - BYTE nbBits; + U32 count; + U16 parent; + BYTE byte; + BYTE nbBits; } nodeElt; static U32 HUF_setMaxHeight(nodeElt* huffNode, U32 lastNonNull, U32 maxNbBits) { - const U32 largestBits = huffNode[lastNonNull].nbBits; - if (largestBits <= maxNbBits) return largestBits; /* early exit : no elt > maxNbBits */ + const U32 largestBits = huffNode[lastNonNull].nbBits; + if (largestBits <= maxNbBits) return largestBits; /* early exit : no elt > maxNbBits */ - /* there are several too large elements (at least >= 2) */ - { int totalCost = 0; - const U32 baseCost = 1 << (largestBits - maxNbBits); - U32 n = lastNonNull; + /* there are several too large elements (at least >= 2) */ + { int totalCost = 0; + const U32 baseCost = 1 << (largestBits - maxNbBits); + U32 n = lastNonNull; - while (huffNode[n].nbBits > maxNbBits) { - totalCost += baseCost - (1 << (largestBits - huffNode[n].nbBits)); - huffNode[n].nbBits = (BYTE)maxNbBits; - n --; - } /* n stops at huffNode[n].nbBits <= maxNbBits */ - while (huffNode[n].nbBits == maxNbBits) n--; /* n end at index of smallest symbol using < maxNbBits */ + while (huffNode[n].nbBits > maxNbBits) { + totalCost += baseCost - (1 << (largestBits - huffNode[n].nbBits)); + huffNode[n].nbBits = (BYTE)maxNbBits; + n --; + } /* n stops at huffNode[n].nbBits <= maxNbBits */ + while (huffNode[n].nbBits == maxNbBits) n--; /* n end at index of smallest symbol using < maxNbBits */ - /* renorm totalCost */ - totalCost >>= (largestBits - maxNbBits); /* note : totalCost is necessarily a multiple of baseCost */ + /* renorm totalCost */ + totalCost >>= (largestBits - maxNbBits); /* note : totalCost is necessarily a multiple of baseCost */ - /* repay normalized cost */ - { U32 const noSymbol = 0xF0F0F0F0; - U32 rankLast[HUF_TABLELOG_MAX+2]; - int pos; + /* repay normalized cost */ + { U32 const noSymbol = 0xF0F0F0F0; + U32 rankLast[HUF_TABLELOG_MAX+2]; + int pos; - /* Get pos of last (smallest) symbol per rank */ - memset(rankLast, 0xF0, sizeof(rankLast)); - { U32 currentNbBits = maxNbBits; - for (pos=n ; pos >= 0; pos--) { - if (huffNode[pos].nbBits >= currentNbBits) continue; - currentNbBits = huffNode[pos].nbBits; /* < maxNbBits */ - rankLast[maxNbBits-currentNbBits] = pos; - } } + /* Get pos of last (smallest) symbol per rank */ + memset(rankLast, 0xF0, sizeof(rankLast)); + { U32 currentNbBits = maxNbBits; + for (pos=n ; pos >= 0; pos--) { + if (huffNode[pos].nbBits >= currentNbBits) continue; + currentNbBits = huffNode[pos].nbBits; /* < maxNbBits */ + rankLast[maxNbBits-currentNbBits] = pos; + } } - while (totalCost > 0) { - U32 nBitsToDecrease = BIT_highbit32(totalCost) + 1; - for ( ; nBitsToDecrease > 1; nBitsToDecrease--) { - U32 highPos = rankLast[nBitsToDecrease]; - U32 lowPos = rankLast[nBitsToDecrease-1]; - if (highPos == noSymbol) continue; - if (lowPos == noSymbol) break; - { U32 const highTotal = huffNode[highPos].count; - U32 const lowTotal = 2 * huffNode[lowPos].count; - if (highTotal <= lowTotal) break; - } } - /* only triggered when no more rank 1 symbol left => find closest one (note : there is necessarily at least one !) */ - while ((nBitsToDecrease<=HUF_TABLELOG_MAX) && (rankLast[nBitsToDecrease] == noSymbol)) /* HUF_MAX_TABLELOG test just to please gcc 5+; but it should not be necessary */ - nBitsToDecrease ++; - totalCost -= 1 << (nBitsToDecrease-1); - if (rankLast[nBitsToDecrease-1] == noSymbol) - rankLast[nBitsToDecrease-1] = rankLast[nBitsToDecrease]; /* this rank is no longer empty */ - huffNode[rankLast[nBitsToDecrease]].nbBits ++; - if (rankLast[nBitsToDecrease] == 0) /* special case, reached largest symbol */ - rankLast[nBitsToDecrease] = noSymbol; - else { - rankLast[nBitsToDecrease]--; - if (huffNode[rankLast[nBitsToDecrease]].nbBits != maxNbBits-nBitsToDecrease) - rankLast[nBitsToDecrease] = noSymbol; /* this rank is now empty */ - } } /* while (totalCost > 0) */ + while (totalCost > 0) { + U32 nBitsToDecrease = BIT_highbit32(totalCost) + 1; + for ( ; nBitsToDecrease > 1; nBitsToDecrease--) { + U32 highPos = rankLast[nBitsToDecrease]; + U32 lowPos = rankLast[nBitsToDecrease-1]; + if (highPos == noSymbol) continue; + if (lowPos == noSymbol) break; + { U32 const highTotal = huffNode[highPos].count; + U32 const lowTotal = 2 * huffNode[lowPos].count; + if (highTotal <= lowTotal) break; + } } + /* only triggered when no more rank 1 symbol left => find closest one (note : there is necessarily at least one !) */ + while ((nBitsToDecrease<=HUF_TABLELOG_MAX) && (rankLast[nBitsToDecrease] == noSymbol)) /* HUF_MAX_TABLELOG test just to please gcc 5+; but it should not be necessary */ + nBitsToDecrease ++; + totalCost -= 1 << (nBitsToDecrease-1); + if (rankLast[nBitsToDecrease-1] == noSymbol) + rankLast[nBitsToDecrease-1] = rankLast[nBitsToDecrease]; /* this rank is no longer empty */ + huffNode[rankLast[nBitsToDecrease]].nbBits ++; + if (rankLast[nBitsToDecrease] == 0) /* special case, reached largest symbol */ + rankLast[nBitsToDecrease] = noSymbol; + else { + rankLast[nBitsToDecrease]--; + if (huffNode[rankLast[nBitsToDecrease]].nbBits != maxNbBits-nBitsToDecrease) + rankLast[nBitsToDecrease] = noSymbol; /* this rank is now empty */ + } } /* while (totalCost > 0) */ - while (totalCost < 0) { /* Sometimes, cost correction overshoot */ - if (rankLast[1] == noSymbol) { /* special case : no rank 1 symbol (using maxNbBits-1); let's create one from largest rank 0 (using maxNbBits) */ - while (huffNode[n].nbBits == maxNbBits) n--; - huffNode[n+1].nbBits--; - rankLast[1] = n+1; - totalCost++; - continue; - } - huffNode[ rankLast[1] + 1 ].nbBits--; - rankLast[1]++; - totalCost ++; - } } } /* there are several too large elements (at least >= 2) */ + while (totalCost < 0) { /* Sometimes, cost correction overshoot */ + if (rankLast[1] == noSymbol) { /* special case : no rank 1 symbol (using maxNbBits-1); let's create one from largest rank 0 (using maxNbBits) */ + while (huffNode[n].nbBits == maxNbBits) n--; + huffNode[n+1].nbBits--; + rankLast[1] = n+1; + totalCost++; + continue; + } + huffNode[ rankLast[1] + 1 ].nbBits--; + rankLast[1]++; + totalCost ++; + } } } /* there are several too large elements (at least >= 2) */ - return maxNbBits; + return maxNbBits; } typedef struct { - U32 base; - U32 current; + U32 base; + U32 current; } rankPos; static void HUF_sort(nodeElt* huffNode, const U32* count, U32 maxSymbolValue) { - rankPos rank[32]; - U32 n; + rankPos rank[32]; + U32 n; - memset(rank, 0, sizeof(rank)); - for (n=0; n<=maxSymbolValue; n++) { - U32 r = BIT_highbit32(count[n] + 1); - rank[r].base ++; - } - for (n=30; n>0; n--) rank[n-1].base += rank[n].base; - for (n=0; n<32; n++) rank[n].current = rank[n].base; - for (n=0; n<=maxSymbolValue; n++) { - U32 const c = count[n]; - U32 const r = BIT_highbit32(c+1) + 1; - U32 pos = rank[r].current++; - while ((pos > rank[r].base) && (c > huffNode[pos-1].count)) huffNode[pos]=huffNode[pos-1], pos--; - huffNode[pos].count = c; - huffNode[pos].byte = (BYTE)n; - } + memset(rank, 0, sizeof(rank)); + for (n=0; n<=maxSymbolValue; n++) { + U32 r = BIT_highbit32(count[n] + 1); + rank[r].base ++; + } + for (n=30; n>0; n--) rank[n-1].base += rank[n].base; + for (n=0; n<32; n++) rank[n].current = rank[n].base; + for (n=0; n<=maxSymbolValue; n++) { + U32 const c = count[n]; + U32 const r = BIT_highbit32(c+1) + 1; + U32 pos = rank[r].current++; + while ((pos > rank[r].base) && (c > huffNode[pos-1].count)) huffNode[pos]=huffNode[pos-1], pos--; + huffNode[pos].count = c; + huffNode[pos].byte = (BYTE)n; + } } @@ -333,71 +333,71 @@ static void HUF_sort(nodeElt* huffNode, const U32* count, U32 maxSymbolValue) typedef nodeElt huffNodeTable[2*HUF_SYMBOLVALUE_MAX+1 +1]; size_t HUF_buildCTable_wksp (HUF_CElt* tree, const U32* count, U32 maxSymbolValue, U32 maxNbBits, void* workSpace, size_t wkspSize) { - nodeElt* const huffNode0 = (nodeElt*)workSpace; - nodeElt* const huffNode = huffNode0+1; - U32 n, nonNullRank; - int lowS, lowN; - U16 nodeNb = STARTNODE; - U32 nodeRoot; + nodeElt* const huffNode0 = (nodeElt*)workSpace; + nodeElt* const huffNode = huffNode0+1; + U32 n, nonNullRank; + int lowS, lowN; + U16 nodeNb = STARTNODE; + U32 nodeRoot; - /* safety checks */ - if (wkspSize < sizeof(huffNodeTable)) return ERROR(GENERIC); /* workSpace is not large enough */ - if (maxNbBits == 0) maxNbBits = HUF_TABLELOG_DEFAULT; - if (maxSymbolValue > HUF_SYMBOLVALUE_MAX) return ERROR(GENERIC); - memset(huffNode0, 0, sizeof(huffNodeTable)); + /* safety checks */ + if (wkspSize < sizeof(huffNodeTable)) return ERROR(GENERIC); /* workSpace is not large enough */ + if (maxNbBits == 0) maxNbBits = HUF_TABLELOG_DEFAULT; + if (maxSymbolValue > HUF_SYMBOLVALUE_MAX) return ERROR(GENERIC); + memset(huffNode0, 0, sizeof(huffNodeTable)); - /* sort, decreasing order */ - HUF_sort(huffNode, count, maxSymbolValue); + /* sort, decreasing order */ + HUF_sort(huffNode, count, maxSymbolValue); - /* init for parents */ - nonNullRank = maxSymbolValue; - while(huffNode[nonNullRank].count == 0) nonNullRank--; - lowS = nonNullRank; nodeRoot = nodeNb + lowS - 1; lowN = nodeNb; - huffNode[nodeNb].count = huffNode[lowS].count + huffNode[lowS-1].count; - huffNode[lowS].parent = huffNode[lowS-1].parent = nodeNb; - nodeNb++; lowS-=2; - for (n=nodeNb; n<=nodeRoot; n++) huffNode[n].count = (U32)(1U<<30); - huffNode0[0].count = (U32)(1U<<31); /* fake entry, strong barrier */ + /* init for parents */ + nonNullRank = maxSymbolValue; + while(huffNode[nonNullRank].count == 0) nonNullRank--; + lowS = nonNullRank; nodeRoot = nodeNb + lowS - 1; lowN = nodeNb; + huffNode[nodeNb].count = huffNode[lowS].count + huffNode[lowS-1].count; + huffNode[lowS].parent = huffNode[lowS-1].parent = nodeNb; + nodeNb++; lowS-=2; + for (n=nodeNb; n<=nodeRoot; n++) huffNode[n].count = (U32)(1U<<30); + huffNode0[0].count = (U32)(1U<<31); /* fake entry, strong barrier */ - /* create parents */ - while (nodeNb <= nodeRoot) { - U32 n1 = (huffNode[lowS].count < huffNode[lowN].count) ? lowS-- : lowN++; - U32 n2 = (huffNode[lowS].count < huffNode[lowN].count) ? lowS-- : lowN++; - huffNode[nodeNb].count = huffNode[n1].count + huffNode[n2].count; - huffNode[n1].parent = huffNode[n2].parent = nodeNb; - nodeNb++; - } + /* create parents */ + while (nodeNb <= nodeRoot) { + U32 n1 = (huffNode[lowS].count < huffNode[lowN].count) ? lowS-- : lowN++; + U32 n2 = (huffNode[lowS].count < huffNode[lowN].count) ? lowS-- : lowN++; + huffNode[nodeNb].count = huffNode[n1].count + huffNode[n2].count; + huffNode[n1].parent = huffNode[n2].parent = nodeNb; + nodeNb++; + } - /* distribute weights (unlimited tree height) */ - huffNode[nodeRoot].nbBits = 0; - for (n=nodeRoot-1; n>=STARTNODE; n--) - huffNode[n].nbBits = huffNode[ huffNode[n].parent ].nbBits + 1; - for (n=0; n<=nonNullRank; n++) - huffNode[n].nbBits = huffNode[ huffNode[n].parent ].nbBits + 1; + /* distribute weights (unlimited tree height) */ + huffNode[nodeRoot].nbBits = 0; + for (n=nodeRoot-1; n>=STARTNODE; n--) + huffNode[n].nbBits = huffNode[ huffNode[n].parent ].nbBits + 1; + for (n=0; n<=nonNullRank; n++) + huffNode[n].nbBits = huffNode[ huffNode[n].parent ].nbBits + 1; - /* enforce maxTableLog */ - maxNbBits = HUF_setMaxHeight(huffNode, nonNullRank, maxNbBits); + /* enforce maxTableLog */ + maxNbBits = HUF_setMaxHeight(huffNode, nonNullRank, maxNbBits); - /* fill result into tree (val, nbBits) */ - { U16 nbPerRank[HUF_TABLELOG_MAX+1] = {0}; - U16 valPerRank[HUF_TABLELOG_MAX+1] = {0}; - if (maxNbBits > HUF_TABLELOG_MAX) return ERROR(GENERIC); /* check fit into table */ - for (n=0; n<=nonNullRank; n++) - nbPerRank[huffNode[n].nbBits]++; - /* determine stating value per rank */ - { U16 min = 0; - for (n=maxNbBits; n>0; n--) { - valPerRank[n] = min; /* get starting value within each rank */ - min += nbPerRank[n]; - min >>= 1; - } } - for (n=0; n<=maxSymbolValue; n++) - tree[huffNode[n].byte].nbBits = huffNode[n].nbBits; /* push nbBits per symbol, symbol order */ - for (n=0; n<=maxSymbolValue; n++) - tree[n].val = valPerRank[tree[n].nbBits]++; /* assign value within rank, symbol order */ - } + /* fill result into tree (val, nbBits) */ + { U16 nbPerRank[HUF_TABLELOG_MAX+1] = {0}; + U16 valPerRank[HUF_TABLELOG_MAX+1] = {0}; + if (maxNbBits > HUF_TABLELOG_MAX) return ERROR(GENERIC); /* check fit into table */ + for (n=0; n<=nonNullRank; n++) + nbPerRank[huffNode[n].nbBits]++; + /* determine stating value per rank */ + { U16 min = 0; + for (n=maxNbBits; n>0; n--) { + valPerRank[n] = min; /* get starting value within each rank */ + min += nbPerRank[n]; + min >>= 1; + } } + for (n=0; n<=maxSymbolValue; n++) + tree[huffNode[n].byte].nbBits = huffNode[n].nbBits; /* push nbBits per symbol, symbol order */ + for (n=0; n<=maxSymbolValue; n++) + tree[n].val = valPerRank[tree[n].nbBits]++; /* assign value within rank, symbol order */ + } - return maxNbBits; + return maxNbBits; } /** HUF_buildCTable() : @@ -405,32 +405,32 @@ size_t HUF_buildCTable_wksp (HUF_CElt* tree, const U32* count, U32 maxSymbolValu */ size_t HUF_buildCTable (HUF_CElt* tree, const U32* count, U32 maxSymbolValue, U32 maxNbBits) { - huffNodeTable nodeTable; - return HUF_buildCTable_wksp(tree, count, maxSymbolValue, maxNbBits, nodeTable, sizeof(nodeTable)); + huffNodeTable nodeTable; + return HUF_buildCTable_wksp(tree, count, maxSymbolValue, maxNbBits, nodeTable, sizeof(nodeTable)); } static size_t HUF_estimateCompressedSize(HUF_CElt* CTable, const unsigned* count, unsigned maxSymbolValue) { - size_t nbBits = 0; - int s; - for (s = 0; s <= (int)maxSymbolValue; ++s) { - nbBits += CTable[s].nbBits * count[s]; - } - return nbBits >> 3; + size_t nbBits = 0; + int s; + for (s = 0; s <= (int)maxSymbolValue; ++s) { + nbBits += CTable[s].nbBits * count[s]; + } + return nbBits >> 3; } static int HUF_validateCTable(const HUF_CElt* CTable, const unsigned* count, unsigned maxSymbolValue) { int bad = 0; int s; for (s = 0; s <= (int)maxSymbolValue; ++s) { - bad |= (count[s] != 0) & (CTable[s].nbBits == 0); + bad |= (count[s] != 0) & (CTable[s].nbBits == 0); } return !bad; } static void HUF_encodeSymbol(BIT_CStream_t* bitCPtr, U32 symbol, const HUF_CElt* CTable) { - BIT_addBitsFast(bitCPtr, CTable[symbol].val, CTable[symbol].nbBits); + BIT_addBitsFast(bitCPtr, CTable[symbol].val, CTable[symbol].nbBits); } size_t HUF_compressBound(size_t size) { return HUF_COMPRESSBOUND(size); } @@ -438,247 +438,247 @@ size_t HUF_compressBound(size_t size) { return HUF_COMPRESSBOUND(size); } #define HUF_FLUSHBITS(s) (fast ? BIT_flushBitsFast(s) : BIT_flushBits(s)) #define HUF_FLUSHBITS_1(stream) \ - if (sizeof((stream)->bitContainer)*8 < HUF_TABLELOG_MAX*2+7) HUF_FLUSHBITS(stream) + if (sizeof((stream)->bitContainer)*8 < HUF_TABLELOG_MAX*2+7) HUF_FLUSHBITS(stream) #define HUF_FLUSHBITS_2(stream) \ - if (sizeof((stream)->bitContainer)*8 < HUF_TABLELOG_MAX*4+7) HUF_FLUSHBITS(stream) + if (sizeof((stream)->bitContainer)*8 < HUF_TABLELOG_MAX*4+7) HUF_FLUSHBITS(stream) size_t HUF_compress1X_usingCTable(void* dst, size_t dstSize, const void* src, size_t srcSize, const HUF_CElt* CTable) { - const BYTE* ip = (const BYTE*) src; - BYTE* const ostart = (BYTE*)dst; - BYTE* const oend = ostart + dstSize; - BYTE* op = ostart; - size_t n; - const unsigned fast = (dstSize >= HUF_BLOCKBOUND(srcSize)); - BIT_CStream_t bitC; + const BYTE* ip = (const BYTE*) src; + BYTE* const ostart = (BYTE*)dst; + BYTE* const oend = ostart + dstSize; + BYTE* op = ostart; + size_t n; + const unsigned fast = (dstSize >= HUF_BLOCKBOUND(srcSize)); + BIT_CStream_t bitC; - /* init */ - if (dstSize < 8) return 0; /* not enough space to compress */ - { size_t const initErr = BIT_initCStream(&bitC, op, oend-op); - if (HUF_isError(initErr)) return 0; } + /* init */ + if (dstSize < 8) return 0; /* not enough space to compress */ + { size_t const initErr = BIT_initCStream(&bitC, op, oend-op); + if (HUF_isError(initErr)) return 0; } - n = srcSize & ~3; /* join to mod 4 */ - switch (srcSize & 3) - { - case 3 : HUF_encodeSymbol(&bitC, ip[n+ 2], CTable); - HUF_FLUSHBITS_2(&bitC); - case 2 : HUF_encodeSymbol(&bitC, ip[n+ 1], CTable); - HUF_FLUSHBITS_1(&bitC); - case 1 : HUF_encodeSymbol(&bitC, ip[n+ 0], CTable); - HUF_FLUSHBITS(&bitC); - case 0 : - default: ; - } + n = srcSize & ~3; /* join to mod 4 */ + switch (srcSize & 3) + { + case 3 : HUF_encodeSymbol(&bitC, ip[n+ 2], CTable); + HUF_FLUSHBITS_2(&bitC); + case 2 : HUF_encodeSymbol(&bitC, ip[n+ 1], CTable); + HUF_FLUSHBITS_1(&bitC); + case 1 : HUF_encodeSymbol(&bitC, ip[n+ 0], CTable); + HUF_FLUSHBITS(&bitC); + case 0 : + default: ; + } - for (; n>0; n-=4) { /* note : n&3==0 at this stage */ - HUF_encodeSymbol(&bitC, ip[n- 1], CTable); - HUF_FLUSHBITS_1(&bitC); - HUF_encodeSymbol(&bitC, ip[n- 2], CTable); - HUF_FLUSHBITS_2(&bitC); - HUF_encodeSymbol(&bitC, ip[n- 3], CTable); - HUF_FLUSHBITS_1(&bitC); - HUF_encodeSymbol(&bitC, ip[n- 4], CTable); - HUF_FLUSHBITS(&bitC); - } + for (; n>0; n-=4) { /* note : n&3==0 at this stage */ + HUF_encodeSymbol(&bitC, ip[n- 1], CTable); + HUF_FLUSHBITS_1(&bitC); + HUF_encodeSymbol(&bitC, ip[n- 2], CTable); + HUF_FLUSHBITS_2(&bitC); + HUF_encodeSymbol(&bitC, ip[n- 3], CTable); + HUF_FLUSHBITS_1(&bitC); + HUF_encodeSymbol(&bitC, ip[n- 4], CTable); + HUF_FLUSHBITS(&bitC); + } - return BIT_closeCStream(&bitC); + return BIT_closeCStream(&bitC); } size_t HUF_compress4X_usingCTable(void* dst, size_t dstSize, const void* src, size_t srcSize, const HUF_CElt* CTable) { - size_t const segmentSize = (srcSize+3)/4; /* first 3 segments */ - const BYTE* ip = (const BYTE*) src; - const BYTE* const iend = ip + srcSize; - BYTE* const ostart = (BYTE*) dst; - BYTE* const oend = ostart + dstSize; - BYTE* op = ostart; + size_t const segmentSize = (srcSize+3)/4; /* first 3 segments */ + const BYTE* ip = (const BYTE*) src; + const BYTE* const iend = ip + srcSize; + BYTE* const ostart = (BYTE*) dst; + BYTE* const oend = ostart + dstSize; + BYTE* op = ostart; - if (dstSize < 6 + 1 + 1 + 1 + 8) return 0; /* minimum space to compress successfully */ - if (srcSize < 12) return 0; /* no saving possible : too small input */ - op += 6; /* jumpTable */ + if (dstSize < 6 + 1 + 1 + 1 + 8) return 0; /* minimum space to compress successfully */ + if (srcSize < 12) return 0; /* no saving possible : too small input */ + op += 6; /* jumpTable */ - { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, segmentSize, CTable) ); - if (cSize==0) return 0; - MEM_writeLE16(ostart, (U16)cSize); - op += cSize; - } + { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, segmentSize, CTable) ); + if (cSize==0) return 0; + MEM_writeLE16(ostart, (U16)cSize); + op += cSize; + } - ip += segmentSize; - { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, segmentSize, CTable) ); - if (cSize==0) return 0; - MEM_writeLE16(ostart+2, (U16)cSize); - op += cSize; - } + ip += segmentSize; + { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, segmentSize, CTable) ); + if (cSize==0) return 0; + MEM_writeLE16(ostart+2, (U16)cSize); + op += cSize; + } - ip += segmentSize; - { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, segmentSize, CTable) ); - if (cSize==0) return 0; - MEM_writeLE16(ostart+4, (U16)cSize); - op += cSize; - } + ip += segmentSize; + { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, segmentSize, CTable) ); + if (cSize==0) return 0; + MEM_writeLE16(ostart+4, (U16)cSize); + op += cSize; + } - ip += segmentSize; - { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, iend-ip, CTable) ); - if (cSize==0) return 0; - op += cSize; - } + ip += segmentSize; + { CHECK_V_F(cSize, HUF_compress1X_usingCTable(op, oend-op, ip, iend-ip, CTable) ); + if (cSize==0) return 0; + op += cSize; + } - return op-ostart; + return op-ostart; } static size_t HUF_compressCTable_internal( - BYTE* const ostart, BYTE* op, BYTE* const oend, - const void* src, size_t srcSize, - unsigned singleStream, const HUF_CElt* CTable) + BYTE* const ostart, BYTE* op, BYTE* const oend, + const void* src, size_t srcSize, + unsigned singleStream, const HUF_CElt* CTable) { - size_t const cSize = singleStream ? - HUF_compress1X_usingCTable(op, oend - op, src, srcSize, CTable) : - HUF_compress4X_usingCTable(op, oend - op, src, srcSize, CTable); - if (HUF_isError(cSize)) { return cSize; } - if (cSize==0) { return 0; } /* uncompressible */ - op += cSize; - /* check compressibility */ - if ((size_t)(op-ostart) >= srcSize-1) { return 0; } - return op-ostart; + size_t const cSize = singleStream ? + HUF_compress1X_usingCTable(op, oend - op, src, srcSize, CTable) : + HUF_compress4X_usingCTable(op, oend - op, src, srcSize, CTable); + if (HUF_isError(cSize)) { return cSize; } + if (cSize==0) { return 0; } /* uncompressible */ + op += cSize; + /* check compressibility */ + if ((size_t)(op-ostart) >= srcSize-1) { return 0; } + return op-ostart; } /* `workSpace` must a table of at least 1024 unsigned */ static size_t HUF_compress_internal ( - void* dst, size_t dstSize, - const void* src, size_t srcSize, - unsigned maxSymbolValue, unsigned huffLog, - unsigned singleStream, - void* workSpace, size_t wkspSize, - HUF_CElt* oldHufTable, HUF_repeat* repeat, int preferRepeat) + void* dst, size_t dstSize, + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog, + unsigned singleStream, + void* workSpace, size_t wkspSize, + HUF_CElt* oldHufTable, HUF_repeat* repeat, int preferRepeat) { - BYTE* const ostart = (BYTE*)dst; - BYTE* const oend = ostart + dstSize; - BYTE* op = ostart; + BYTE* const ostart = (BYTE*)dst; + BYTE* const oend = ostart + dstSize; + BYTE* op = ostart; - U32* count; - size_t const countSize = sizeof(U32) * (HUF_SYMBOLVALUE_MAX + 1); - HUF_CElt* CTable; - size_t const CTableSize = sizeof(HUF_CElt) * (HUF_SYMBOLVALUE_MAX + 1); + U32* count; + size_t const countSize = sizeof(U32) * (HUF_SYMBOLVALUE_MAX + 1); + HUF_CElt* CTable; + size_t const CTableSize = sizeof(HUF_CElt) * (HUF_SYMBOLVALUE_MAX + 1); - /* checks & inits */ - if (wkspSize < sizeof(huffNodeTable) + countSize + CTableSize) return ERROR(GENERIC); - if (!srcSize) return 0; /* Uncompressed (note : 1 means rle, so first byte must be correct) */ - if (!dstSize) return 0; /* cannot fit within dst budget */ - if (srcSize > HUF_BLOCKSIZE_MAX) return ERROR(srcSize_wrong); /* current block size limit */ - if (huffLog > HUF_TABLELOG_MAX) return ERROR(tableLog_tooLarge); - if (!maxSymbolValue) maxSymbolValue = HUF_SYMBOLVALUE_MAX; - if (!huffLog) huffLog = HUF_TABLELOG_DEFAULT; + /* checks & inits */ + if (wkspSize < sizeof(huffNodeTable) + countSize + CTableSize) return ERROR(GENERIC); + if (!srcSize) return 0; /* Uncompressed (note : 1 means rle, so first byte must be correct) */ + if (!dstSize) return 0; /* cannot fit within dst budget */ + if (srcSize > HUF_BLOCKSIZE_MAX) return ERROR(srcSize_wrong); /* current block size limit */ + if (huffLog > HUF_TABLELOG_MAX) return ERROR(tableLog_tooLarge); + if (!maxSymbolValue) maxSymbolValue = HUF_SYMBOLVALUE_MAX; + if (!huffLog) huffLog = HUF_TABLELOG_DEFAULT; - count = (U32*)workSpace; - workSpace = (BYTE*)workSpace + countSize; - wkspSize -= countSize; - CTable = (HUF_CElt*)workSpace; - workSpace = (BYTE*)workSpace + CTableSize; - wkspSize -= CTableSize; + count = (U32*)workSpace; + workSpace = (BYTE*)workSpace + countSize; + wkspSize -= countSize; + CTable = (HUF_CElt*)workSpace; + workSpace = (BYTE*)workSpace + CTableSize; + wkspSize -= CTableSize; - /* Heuristic : If we don't need to check the validity of the old table use the old table for small inputs */ - if (preferRepeat && repeat && *repeat == HUF_repeat_valid) { - return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, oldHufTable); - } + /* Heuristic : If we don't need to check the validity of the old table use the old table for small inputs */ + if (preferRepeat && repeat && *repeat == HUF_repeat_valid) { + return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, oldHufTable); + } - /* Scan input and build symbol stats */ - { CHECK_V_F(largest, FSE_count_wksp (count, &maxSymbolValue, (const BYTE*)src, srcSize, (U32*)workSpace) ); - if (largest == srcSize) { *ostart = ((const BYTE*)src)[0]; return 1; } /* single symbol, rle */ - if (largest <= (srcSize >> 7)+1) return 0; /* Fast heuristic : not compressible enough */ - } + /* Scan input and build symbol stats */ + { CHECK_V_F(largest, FSE_count_wksp (count, &maxSymbolValue, (const BYTE*)src, srcSize, (U32*)workSpace) ); + if (largest == srcSize) { *ostart = ((const BYTE*)src)[0]; return 1; } /* single symbol, rle */ + if (largest <= (srcSize >> 7)+1) return 0; /* Fast heuristic : not compressible enough */ + } - /* Check validity of previous table */ - if (repeat && *repeat == HUF_repeat_check && !HUF_validateCTable(oldHufTable, count, maxSymbolValue)) { - *repeat = HUF_repeat_none; - } - /* Heuristic : use existing table for small inputs */ - if (preferRepeat && repeat && *repeat != HUF_repeat_none) { - return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, oldHufTable); - } + /* Check validity of previous table */ + if (repeat && *repeat == HUF_repeat_check && !HUF_validateCTable(oldHufTable, count, maxSymbolValue)) { + *repeat = HUF_repeat_none; + } + /* Heuristic : use existing table for small inputs */ + if (preferRepeat && repeat && *repeat != HUF_repeat_none) { + return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, oldHufTable); + } - /* Build Huffman Tree */ - huffLog = HUF_optimalTableLog(huffLog, srcSize, maxSymbolValue); - { CHECK_V_F(maxBits, HUF_buildCTable_wksp (CTable, count, maxSymbolValue, huffLog, workSpace, wkspSize) ); - huffLog = (U32)maxBits; - /* Zero the unused symbols so we can check it for validity */ - memset(CTable + maxSymbolValue + 1, 0, CTableSize - (maxSymbolValue + 1) * sizeof(HUF_CElt)); - } + /* Build Huffman Tree */ + huffLog = HUF_optimalTableLog(huffLog, srcSize, maxSymbolValue); + { CHECK_V_F(maxBits, HUF_buildCTable_wksp (CTable, count, maxSymbolValue, huffLog, workSpace, wkspSize) ); + huffLog = (U32)maxBits; + /* Zero the unused symbols so we can check it for validity */ + memset(CTable + maxSymbolValue + 1, 0, CTableSize - (maxSymbolValue + 1) * sizeof(HUF_CElt)); + } - /* Write table description header */ - { CHECK_V_F(hSize, HUF_writeCTable (op, dstSize, CTable, maxSymbolValue, huffLog) ); - /* Check if using the previous table will be beneficial */ - if (repeat && *repeat != HUF_repeat_none) { - size_t const oldSize = HUF_estimateCompressedSize(oldHufTable, count, maxSymbolValue); - size_t const newSize = HUF_estimateCompressedSize(CTable, count, maxSymbolValue); - if (oldSize <= hSize + newSize || hSize + 12 >= srcSize) { - return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, oldHufTable); - } - } - /* Use the new table */ - if (hSize + 12ul >= srcSize) { return 0; } - op += hSize; - if (repeat) { *repeat = HUF_repeat_none; } - if (oldHufTable) { memcpy(oldHufTable, CTable, CTableSize); } /* Save the new table */ - } - return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, CTable); + /* Write table description header */ + { CHECK_V_F(hSize, HUF_writeCTable (op, dstSize, CTable, maxSymbolValue, huffLog) ); + /* Check if using the previous table will be beneficial */ + if (repeat && *repeat != HUF_repeat_none) { + size_t const oldSize = HUF_estimateCompressedSize(oldHufTable, count, maxSymbolValue); + size_t const newSize = HUF_estimateCompressedSize(CTable, count, maxSymbolValue); + if (oldSize <= hSize + newSize || hSize + 12 >= srcSize) { + return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, oldHufTable); + } + } + /* Use the new table */ + if (hSize + 12ul >= srcSize) { return 0; } + op += hSize; + if (repeat) { *repeat = HUF_repeat_none; } + if (oldHufTable) { memcpy(oldHufTable, CTable, CTableSize); } /* Save the new table */ + } + return HUF_compressCTable_internal(ostart, op, oend, src, srcSize, singleStream, CTable); } size_t HUF_compress1X_wksp (void* dst, size_t dstSize, - const void* src, size_t srcSize, - unsigned maxSymbolValue, unsigned huffLog, - void* workSpace, size_t wkspSize) + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog, + void* workSpace, size_t wkspSize) { - return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 1 /* single stream */, workSpace, wkspSize, NULL, NULL, 0); + return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 1 /* single stream */, workSpace, wkspSize, NULL, NULL, 0); } size_t HUF_compress1X_repeat (void* dst, size_t dstSize, - const void* src, size_t srcSize, - unsigned maxSymbolValue, unsigned huffLog, - void* workSpace, size_t wkspSize, - HUF_CElt* hufTable, HUF_repeat* repeat, int preferRepeat) + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog, + void* workSpace, size_t wkspSize, + HUF_CElt* hufTable, HUF_repeat* repeat, int preferRepeat) { - return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 1 /* single stream */, workSpace, wkspSize, hufTable, repeat, preferRepeat); + return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 1 /* single stream */, workSpace, wkspSize, hufTable, repeat, preferRepeat); } size_t HUF_compress1X (void* dst, size_t dstSize, - const void* src, size_t srcSize, - unsigned maxSymbolValue, unsigned huffLog) + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog) { - unsigned workSpace[1024]; - return HUF_compress1X_wksp(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, workSpace, sizeof(workSpace)); + unsigned workSpace[1024]; + return HUF_compress1X_wksp(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, workSpace, sizeof(workSpace)); } size_t HUF_compress4X_wksp (void* dst, size_t dstSize, - const void* src, size_t srcSize, - unsigned maxSymbolValue, unsigned huffLog, - void* workSpace, size_t wkspSize) + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog, + void* workSpace, size_t wkspSize) { - return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 0 /* 4 streams */, workSpace, wkspSize, NULL, NULL, 0); + return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 0 /* 4 streams */, workSpace, wkspSize, NULL, NULL, 0); } size_t HUF_compress4X_repeat (void* dst, size_t dstSize, - const void* src, size_t srcSize, - unsigned maxSymbolValue, unsigned huffLog, - void* workSpace, size_t wkspSize, - HUF_CElt* hufTable, HUF_repeat* repeat, int preferRepeat) + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog, + void* workSpace, size_t wkspSize, + HUF_CElt* hufTable, HUF_repeat* repeat, int preferRepeat) { - return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 0 /* 4 streams */, workSpace, wkspSize, hufTable, repeat, preferRepeat); + return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 0 /* 4 streams */, workSpace, wkspSize, hufTable, repeat, preferRepeat); } size_t HUF_compress2 (void* dst, size_t dstSize, - const void* src, size_t srcSize, - unsigned maxSymbolValue, unsigned huffLog) + const void* src, size_t srcSize, + unsigned maxSymbolValue, unsigned huffLog) { - unsigned workSpace[1024]; - return HUF_compress4X_wksp(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, workSpace, sizeof(workSpace)); + unsigned workSpace[1024]; + return HUF_compress4X_wksp(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, workSpace, sizeof(workSpace)); } size_t HUF_compress (void* dst, size_t maxDstSize, const void* src, size_t srcSize) { - return HUF_compress2(dst, maxDstSize, src, (U32)srcSize, 255, HUF_TABLELOG_DEFAULT); + return HUF_compress2(dst, maxDstSize, src, (U32)srcSize, 255, HUF_TABLELOG_DEFAULT); } diff --git a/contrib/linux-kernel/lib/huf_decompress.c b/contrib/linux-kernel/lib/huf_decompress.c index ea35c3620..a1e9ffc32 100644 --- a/contrib/linux-kernel/lib/huf_decompress.c +++ b/contrib/linux-kernel/lib/huf_decompress.c @@ -8,9 +8,9 @@ modification, are permitted provided that the following conditions are met: - * Redistributions of source code must retain the above copyright + * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - * Redistributions in binary form must reproduce the above + * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. @@ -27,9 +27,9 @@ (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - You can contact the author at : - - FSE+HUF source repository : https://github.com/Cyan4973/FiniteStateEntropy - - Public forum : https://groups.google.com/forum/#!forum/lz4c + You can contact the author at : + - FSE+HUF source repository : https://github.com/Cyan4973/FiniteStateEntropy + - Public forum : https://groups.google.com/forum/#!forum/lz4c ****************************************************************** */ /* ************************************************************** @@ -75,9 +75,9 @@ typedef struct { BYTE maxTableLog; BYTE tableType; BYTE tableLog; BYTE reserved; static DTableDesc HUF_getDTableDesc(const HUF_DTable* table) { - DTableDesc dtd; - memcpy(&dtd, table, sizeof(dtd)); - return dtd; + DTableDesc dtd; + memcpy(&dtd, table, sizeof(dtd)); + return dtd; } @@ -89,268 +89,268 @@ typedef struct { BYTE byte; BYTE nbBits; } HUF_DEltX2; /* single-symbol decodi size_t HUF_readDTableX2 (HUF_DTable* DTable, const void* src, size_t srcSize) { - BYTE huffWeight[HUF_SYMBOLVALUE_MAX + 1]; - U32 rankVal[HUF_TABLELOG_ABSOLUTEMAX + 1]; /* large enough for values from 0 to 16 */ - U32 tableLog = 0; - U32 nbSymbols = 0; - size_t iSize; - void* const dtPtr = DTable + 1; - HUF_DEltX2* const dt = (HUF_DEltX2*)dtPtr; + BYTE huffWeight[HUF_SYMBOLVALUE_MAX + 1]; + U32 rankVal[HUF_TABLELOG_ABSOLUTEMAX + 1]; /* large enough for values from 0 to 16 */ + U32 tableLog = 0; + U32 nbSymbols = 0; + size_t iSize; + void* const dtPtr = DTable + 1; + HUF_DEltX2* const dt = (HUF_DEltX2*)dtPtr; - HUF_STATIC_ASSERT(sizeof(DTableDesc) == sizeof(HUF_DTable)); - /* memset(huffWeight, 0, sizeof(huffWeight)); */ /* is not necessary, even though some analyzer complain ... */ + HUF_STATIC_ASSERT(sizeof(DTableDesc) == sizeof(HUF_DTable)); + /* memset(huffWeight, 0, sizeof(huffWeight)); */ /* is not necessary, even though some analyzer complain ... */ - iSize = HUF_readStats(huffWeight, HUF_SYMBOLVALUE_MAX + 1, rankVal, &nbSymbols, &tableLog, src, srcSize); - if (HUF_isError(iSize)) return iSize; + iSize = HUF_readStats(huffWeight, HUF_SYMBOLVALUE_MAX + 1, rankVal, &nbSymbols, &tableLog, src, srcSize); + if (HUF_isError(iSize)) return iSize; - /* Table header */ - { DTableDesc dtd = HUF_getDTableDesc(DTable); - if (tableLog > (U32)(dtd.maxTableLog+1)) return ERROR(tableLog_tooLarge); /* DTable too small, Huffman tree cannot fit in */ - dtd.tableType = 0; - dtd.tableLog = (BYTE)tableLog; - memcpy(DTable, &dtd, sizeof(dtd)); - } + /* Table header */ + { DTableDesc dtd = HUF_getDTableDesc(DTable); + if (tableLog > (U32)(dtd.maxTableLog+1)) return ERROR(tableLog_tooLarge); /* DTable too small, Huffman tree cannot fit in */ + dtd.tableType = 0; + dtd.tableLog = (BYTE)tableLog; + memcpy(DTable, &dtd, sizeof(dtd)); + } - /* Calculate starting value for each rank */ - { U32 n, nextRankStart = 0; - for (n=1; n> 1; - U32 u; - HUF_DEltX2 D; - D.byte = (BYTE)n; D.nbBits = (BYTE)(tableLog + 1 - w); - for (u = rankVal[w]; u < rankVal[w] + length; u++) - dt[u] = D; - rankVal[w] += length; - } } + /* fill DTable */ + { U32 n; + for (n=0; n> 1; + U32 u; + HUF_DEltX2 D; + D.byte = (BYTE)n; D.nbBits = (BYTE)(tableLog + 1 - w); + for (u = rankVal[w]; u < rankVal[w] + length; u++) + dt[u] = D; + rankVal[w] += length; + } } - return iSize; + return iSize; } static BYTE HUF_decodeSymbolX2(BIT_DStream_t* Dstream, const HUF_DEltX2* dt, const U32 dtLog) { - size_t const val = BIT_lookBitsFast(Dstream, dtLog); /* note : dtLog >= 1 */ - BYTE const c = dt[val].byte; - BIT_skipBits(Dstream, dt[val].nbBits); - return c; + size_t const val = BIT_lookBitsFast(Dstream, dtLog); /* note : dtLog >= 1 */ + BYTE const c = dt[val].byte; + BIT_skipBits(Dstream, dt[val].nbBits); + return c; } #define HUF_DECODE_SYMBOLX2_0(ptr, DStreamPtr) \ - *ptr++ = HUF_decodeSymbolX2(DStreamPtr, dt, dtLog) + *ptr++ = HUF_decodeSymbolX2(DStreamPtr, dt, dtLog) #define HUF_DECODE_SYMBOLX2_1(ptr, DStreamPtr) \ - if (MEM_64bits() || (HUF_TABLELOG_MAX<=12)) \ - HUF_DECODE_SYMBOLX2_0(ptr, DStreamPtr) + if (MEM_64bits() || (HUF_TABLELOG_MAX<=12)) \ + HUF_DECODE_SYMBOLX2_0(ptr, DStreamPtr) #define HUF_DECODE_SYMBOLX2_2(ptr, DStreamPtr) \ - if (MEM_64bits()) \ - HUF_DECODE_SYMBOLX2_0(ptr, DStreamPtr) + if (MEM_64bits()) \ + HUF_DECODE_SYMBOLX2_0(ptr, DStreamPtr) FORCE_INLINE size_t HUF_decodeStreamX2(BYTE* p, BIT_DStream_t* const bitDPtr, BYTE* const pEnd, const HUF_DEltX2* const dt, const U32 dtLog) { - BYTE* const pStart = p; + BYTE* const pStart = p; - /* up to 4 symbols at a time */ - while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) && (p <= pEnd-4)) { - HUF_DECODE_SYMBOLX2_2(p, bitDPtr); - HUF_DECODE_SYMBOLX2_1(p, bitDPtr); - HUF_DECODE_SYMBOLX2_2(p, bitDPtr); - HUF_DECODE_SYMBOLX2_0(p, bitDPtr); - } + /* up to 4 symbols at a time */ + while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) && (p <= pEnd-4)) { + HUF_DECODE_SYMBOLX2_2(p, bitDPtr); + HUF_DECODE_SYMBOLX2_1(p, bitDPtr); + HUF_DECODE_SYMBOLX2_2(p, bitDPtr); + HUF_DECODE_SYMBOLX2_0(p, bitDPtr); + } - /* closer to the end */ - while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) && (p < pEnd)) - HUF_DECODE_SYMBOLX2_0(p, bitDPtr); + /* closer to the end */ + while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) && (p < pEnd)) + HUF_DECODE_SYMBOLX2_0(p, bitDPtr); - /* no more data to retrieve from bitstream, hence no need to reload */ - while (p < pEnd) - HUF_DECODE_SYMBOLX2_0(p, bitDPtr); + /* no more data to retrieve from bitstream, hence no need to reload */ + while (p < pEnd) + HUF_DECODE_SYMBOLX2_0(p, bitDPtr); - return pEnd-pStart; + return pEnd-pStart; } static size_t HUF_decompress1X2_usingDTable_internal( - void* dst, size_t dstSize, - const void* cSrc, size_t cSrcSize, - const HUF_DTable* DTable) + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) { - BYTE* op = (BYTE*)dst; - BYTE* const oend = op + dstSize; - const void* dtPtr = DTable + 1; - const HUF_DEltX2* const dt = (const HUF_DEltX2*)dtPtr; - BIT_DStream_t bitD; - DTableDesc const dtd = HUF_getDTableDesc(DTable); - U32 const dtLog = dtd.tableLog; + BYTE* op = (BYTE*)dst; + BYTE* const oend = op + dstSize; + const void* dtPtr = DTable + 1; + const HUF_DEltX2* const dt = (const HUF_DEltX2*)dtPtr; + BIT_DStream_t bitD; + DTableDesc const dtd = HUF_getDTableDesc(DTable); + U32 const dtLog = dtd.tableLog; - { size_t const errorCode = BIT_initDStream(&bitD, cSrc, cSrcSize); - if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD, cSrc, cSrcSize); + if (HUF_isError(errorCode)) return errorCode; } - HUF_decodeStreamX2(op, &bitD, oend, dt, dtLog); + HUF_decodeStreamX2(op, &bitD, oend, dt, dtLog); - /* check */ - if (!BIT_endOfDStream(&bitD)) return ERROR(corruption_detected); + /* check */ + if (!BIT_endOfDStream(&bitD)) return ERROR(corruption_detected); - return dstSize; + return dstSize; } size_t HUF_decompress1X2_usingDTable( - void* dst, size_t dstSize, - const void* cSrc, size_t cSrcSize, - const HUF_DTable* DTable) + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) { - DTableDesc dtd = HUF_getDTableDesc(DTable); - if (dtd.tableType != 0) return ERROR(GENERIC); - return HUF_decompress1X2_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); + DTableDesc dtd = HUF_getDTableDesc(DTable); + if (dtd.tableType != 0) return ERROR(GENERIC); + return HUF_decompress1X2_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); } size_t HUF_decompress1X2_DCtx (HUF_DTable* DCtx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - const BYTE* ip = (const BYTE*) cSrc; + const BYTE* ip = (const BYTE*) cSrc; - size_t const hSize = HUF_readDTableX2 (DCtx, cSrc, cSrcSize); - if (HUF_isError(hSize)) return hSize; - if (hSize >= cSrcSize) return ERROR(srcSize_wrong); - ip += hSize; cSrcSize -= hSize; + size_t const hSize = HUF_readDTableX2 (DCtx, cSrc, cSrcSize); + if (HUF_isError(hSize)) return hSize; + if (hSize >= cSrcSize) return ERROR(srcSize_wrong); + ip += hSize; cSrcSize -= hSize; - return HUF_decompress1X2_usingDTable_internal (dst, dstSize, ip, cSrcSize, DCtx); + return HUF_decompress1X2_usingDTable_internal (dst, dstSize, ip, cSrcSize, DCtx); } size_t HUF_decompress1X2 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - HUF_CREATE_STATIC_DTABLEX2(DTable, HUF_TABLELOG_MAX); - return HUF_decompress1X2_DCtx (DTable, dst, dstSize, cSrc, cSrcSize); + HUF_CREATE_STATIC_DTABLEX2(DTable, HUF_TABLELOG_MAX); + return HUF_decompress1X2_DCtx (DTable, dst, dstSize, cSrc, cSrcSize); } static size_t HUF_decompress4X2_usingDTable_internal( - void* dst, size_t dstSize, - const void* cSrc, size_t cSrcSize, - const HUF_DTable* DTable) + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) { - /* Check */ - if (cSrcSize < 10) return ERROR(corruption_detected); /* strict minimum : jump table + 1 byte per stream */ + /* Check */ + if (cSrcSize < 10) return ERROR(corruption_detected); /* strict minimum : jump table + 1 byte per stream */ - { const BYTE* const istart = (const BYTE*) cSrc; - BYTE* const ostart = (BYTE*) dst; - BYTE* const oend = ostart + dstSize; - const void* const dtPtr = DTable + 1; - const HUF_DEltX2* const dt = (const HUF_DEltX2*)dtPtr; + { const BYTE* const istart = (const BYTE*) cSrc; + BYTE* const ostart = (BYTE*) dst; + BYTE* const oend = ostart + dstSize; + const void* const dtPtr = DTable + 1; + const HUF_DEltX2* const dt = (const HUF_DEltX2*)dtPtr; - /* Init */ - BIT_DStream_t bitD1; - BIT_DStream_t bitD2; - BIT_DStream_t bitD3; - BIT_DStream_t bitD4; - size_t const length1 = MEM_readLE16(istart); - size_t const length2 = MEM_readLE16(istart+2); - size_t const length3 = MEM_readLE16(istart+4); - size_t const length4 = cSrcSize - (length1 + length2 + length3 + 6); - const BYTE* const istart1 = istart + 6; /* jumpTable */ - const BYTE* const istart2 = istart1 + length1; - const BYTE* const istart3 = istart2 + length2; - const BYTE* const istart4 = istart3 + length3; - const size_t segmentSize = (dstSize+3) / 4; - BYTE* const opStart2 = ostart + segmentSize; - BYTE* const opStart3 = opStart2 + segmentSize; - BYTE* const opStart4 = opStart3 + segmentSize; - BYTE* op1 = ostart; - BYTE* op2 = opStart2; - BYTE* op3 = opStart3; - BYTE* op4 = opStart4; - U32 endSignal; - DTableDesc const dtd = HUF_getDTableDesc(DTable); - U32 const dtLog = dtd.tableLog; + /* Init */ + BIT_DStream_t bitD1; + BIT_DStream_t bitD2; + BIT_DStream_t bitD3; + BIT_DStream_t bitD4; + size_t const length1 = MEM_readLE16(istart); + size_t const length2 = MEM_readLE16(istart+2); + size_t const length3 = MEM_readLE16(istart+4); + size_t const length4 = cSrcSize - (length1 + length2 + length3 + 6); + const BYTE* const istart1 = istart + 6; /* jumpTable */ + const BYTE* const istart2 = istart1 + length1; + const BYTE* const istart3 = istart2 + length2; + const BYTE* const istart4 = istart3 + length3; + const size_t segmentSize = (dstSize+3) / 4; + BYTE* const opStart2 = ostart + segmentSize; + BYTE* const opStart3 = opStart2 + segmentSize; + BYTE* const opStart4 = opStart3 + segmentSize; + BYTE* op1 = ostart; + BYTE* op2 = opStart2; + BYTE* op3 = opStart3; + BYTE* op4 = opStart4; + U32 endSignal; + DTableDesc const dtd = HUF_getDTableDesc(DTable); + U32 const dtLog = dtd.tableLog; - if (length4 > cSrcSize) return ERROR(corruption_detected); /* overflow */ - { size_t const errorCode = BIT_initDStream(&bitD1, istart1, length1); - if (HUF_isError(errorCode)) return errorCode; } - { size_t const errorCode = BIT_initDStream(&bitD2, istart2, length2); - if (HUF_isError(errorCode)) return errorCode; } - { size_t const errorCode = BIT_initDStream(&bitD3, istart3, length3); - if (HUF_isError(errorCode)) return errorCode; } - { size_t const errorCode = BIT_initDStream(&bitD4, istart4, length4); - if (HUF_isError(errorCode)) return errorCode; } + if (length4 > cSrcSize) return ERROR(corruption_detected); /* overflow */ + { size_t const errorCode = BIT_initDStream(&bitD1, istart1, length1); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD2, istart2, length2); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD3, istart3, length3); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD4, istart4, length4); + if (HUF_isError(errorCode)) return errorCode; } - /* 16-32 symbols per loop (4-8 symbols per stream) */ - endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); - for ( ; (endSignal==BIT_DStream_unfinished) && (op4<(oend-7)) ; ) { - HUF_DECODE_SYMBOLX2_2(op1, &bitD1); - HUF_DECODE_SYMBOLX2_2(op2, &bitD2); - HUF_DECODE_SYMBOLX2_2(op3, &bitD3); - HUF_DECODE_SYMBOLX2_2(op4, &bitD4); - HUF_DECODE_SYMBOLX2_1(op1, &bitD1); - HUF_DECODE_SYMBOLX2_1(op2, &bitD2); - HUF_DECODE_SYMBOLX2_1(op3, &bitD3); - HUF_DECODE_SYMBOLX2_1(op4, &bitD4); - HUF_DECODE_SYMBOLX2_2(op1, &bitD1); - HUF_DECODE_SYMBOLX2_2(op2, &bitD2); - HUF_DECODE_SYMBOLX2_2(op3, &bitD3); - HUF_DECODE_SYMBOLX2_2(op4, &bitD4); - HUF_DECODE_SYMBOLX2_0(op1, &bitD1); - HUF_DECODE_SYMBOLX2_0(op2, &bitD2); - HUF_DECODE_SYMBOLX2_0(op3, &bitD3); - HUF_DECODE_SYMBOLX2_0(op4, &bitD4); - endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); - } + /* 16-32 symbols per loop (4-8 symbols per stream) */ + endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); + for ( ; (endSignal==BIT_DStream_unfinished) && (op4<(oend-7)) ; ) { + HUF_DECODE_SYMBOLX2_2(op1, &bitD1); + HUF_DECODE_SYMBOLX2_2(op2, &bitD2); + HUF_DECODE_SYMBOLX2_2(op3, &bitD3); + HUF_DECODE_SYMBOLX2_2(op4, &bitD4); + HUF_DECODE_SYMBOLX2_1(op1, &bitD1); + HUF_DECODE_SYMBOLX2_1(op2, &bitD2); + HUF_DECODE_SYMBOLX2_1(op3, &bitD3); + HUF_DECODE_SYMBOLX2_1(op4, &bitD4); + HUF_DECODE_SYMBOLX2_2(op1, &bitD1); + HUF_DECODE_SYMBOLX2_2(op2, &bitD2); + HUF_DECODE_SYMBOLX2_2(op3, &bitD3); + HUF_DECODE_SYMBOLX2_2(op4, &bitD4); + HUF_DECODE_SYMBOLX2_0(op1, &bitD1); + HUF_DECODE_SYMBOLX2_0(op2, &bitD2); + HUF_DECODE_SYMBOLX2_0(op3, &bitD3); + HUF_DECODE_SYMBOLX2_0(op4, &bitD4); + endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); + } - /* check corruption */ - if (op1 > opStart2) return ERROR(corruption_detected); - if (op2 > opStart3) return ERROR(corruption_detected); - if (op3 > opStart4) return ERROR(corruption_detected); - /* note : op4 supposed already verified within main loop */ + /* check corruption */ + if (op1 > opStart2) return ERROR(corruption_detected); + if (op2 > opStart3) return ERROR(corruption_detected); + if (op3 > opStart4) return ERROR(corruption_detected); + /* note : op4 supposed already verified within main loop */ - /* finish bitStreams one by one */ - HUF_decodeStreamX2(op1, &bitD1, opStart2, dt, dtLog); - HUF_decodeStreamX2(op2, &bitD2, opStart3, dt, dtLog); - HUF_decodeStreamX2(op3, &bitD3, opStart4, dt, dtLog); - HUF_decodeStreamX2(op4, &bitD4, oend, dt, dtLog); + /* finish bitStreams one by one */ + HUF_decodeStreamX2(op1, &bitD1, opStart2, dt, dtLog); + HUF_decodeStreamX2(op2, &bitD2, opStart3, dt, dtLog); + HUF_decodeStreamX2(op3, &bitD3, opStart4, dt, dtLog); + HUF_decodeStreamX2(op4, &bitD4, oend, dt, dtLog); - /* check */ - endSignal = BIT_endOfDStream(&bitD1) & BIT_endOfDStream(&bitD2) & BIT_endOfDStream(&bitD3) & BIT_endOfDStream(&bitD4); - if (!endSignal) return ERROR(corruption_detected); + /* check */ + endSignal = BIT_endOfDStream(&bitD1) & BIT_endOfDStream(&bitD2) & BIT_endOfDStream(&bitD3) & BIT_endOfDStream(&bitD4); + if (!endSignal) return ERROR(corruption_detected); - /* decoded size */ - return dstSize; - } + /* decoded size */ + return dstSize; + } } size_t HUF_decompress4X2_usingDTable( - void* dst, size_t dstSize, - const void* cSrc, size_t cSrcSize, - const HUF_DTable* DTable) + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) { - DTableDesc dtd = HUF_getDTableDesc(DTable); - if (dtd.tableType != 0) return ERROR(GENERIC); - return HUF_decompress4X2_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); + DTableDesc dtd = HUF_getDTableDesc(DTable); + if (dtd.tableType != 0) return ERROR(GENERIC); + return HUF_decompress4X2_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); } size_t HUF_decompress4X2_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - const BYTE* ip = (const BYTE*) cSrc; + const BYTE* ip = (const BYTE*) cSrc; - size_t const hSize = HUF_readDTableX2 (dctx, cSrc, cSrcSize); - if (HUF_isError(hSize)) return hSize; - if (hSize >= cSrcSize) return ERROR(srcSize_wrong); - ip += hSize; cSrcSize -= hSize; + size_t const hSize = HUF_readDTableX2 (dctx, cSrc, cSrcSize); + if (HUF_isError(hSize)) return hSize; + if (hSize >= cSrcSize) return ERROR(srcSize_wrong); + ip += hSize; cSrcSize -= hSize; - return HUF_decompress4X2_usingDTable_internal (dst, dstSize, ip, cSrcSize, dctx); + return HUF_decompress4X2_usingDTable_internal (dst, dstSize, ip, cSrcSize, dctx); } size_t HUF_decompress4X2 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - HUF_CREATE_STATIC_DTABLEX2(DTable, HUF_TABLELOG_MAX); - return HUF_decompress4X2_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); + HUF_CREATE_STATIC_DTABLEX2(DTable, HUF_TABLELOG_MAX); + return HUF_decompress4X2_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); } @@ -364,406 +364,406 @@ typedef struct { BYTE symbol; BYTE weight; } sortedSymbol_t; /* HUF_fillDTableX4Level2() : * `rankValOrigin` must be a table of at least (HUF_TABLELOG_MAX + 1) U32 */ static void HUF_fillDTableX4Level2(HUF_DEltX4* DTable, U32 sizeLog, const U32 consumed, - const U32* rankValOrigin, const int minWeight, - const sortedSymbol_t* sortedSymbols, const U32 sortedListSize, - U32 nbBitsBaseline, U16 baseSeq) + const U32* rankValOrigin, const int minWeight, + const sortedSymbol_t* sortedSymbols, const U32 sortedListSize, + U32 nbBitsBaseline, U16 baseSeq) { - HUF_DEltX4 DElt; - U32 rankVal[HUF_TABLELOG_MAX + 1]; + HUF_DEltX4 DElt; + U32 rankVal[HUF_TABLELOG_MAX + 1]; - /* get pre-calculated rankVal */ - memcpy(rankVal, rankValOrigin, sizeof(rankVal)); + /* get pre-calculated rankVal */ + memcpy(rankVal, rankValOrigin, sizeof(rankVal)); - /* fill skipped values */ - if (minWeight>1) { - U32 i, skipSize = rankVal[minWeight]; - MEM_writeLE16(&(DElt.sequence), baseSeq); - DElt.nbBits = (BYTE)(consumed); - DElt.length = 1; - for (i = 0; i < skipSize; i++) - DTable[i] = DElt; - } + /* fill skipped values */ + if (minWeight>1) { + U32 i, skipSize = rankVal[minWeight]; + MEM_writeLE16(&(DElt.sequence), baseSeq); + DElt.nbBits = (BYTE)(consumed); + DElt.length = 1; + for (i = 0; i < skipSize; i++) + DTable[i] = DElt; + } - /* fill DTable */ - { U32 s; for (s=0; s= 1 */ + MEM_writeLE16(&(DElt.sequence), (U16)(baseSeq + (symbol << 8))); + DElt.nbBits = (BYTE)(nbBits + consumed); + DElt.length = 2; + do { DTable[i++] = DElt; } while (i= 1 */ - rankVal[weight] += length; - } } + rankVal[weight] += length; + } } } typedef U32 rankVal_t[HUF_TABLELOG_MAX][HUF_TABLELOG_MAX + 1]; static void HUF_fillDTableX4(HUF_DEltX4* DTable, const U32 targetLog, - const sortedSymbol_t* sortedList, const U32 sortedListSize, - const U32* rankStart, rankVal_t rankValOrigin, const U32 maxWeight, - const U32 nbBitsBaseline) + const sortedSymbol_t* sortedList, const U32 sortedListSize, + const U32* rankStart, rankVal_t rankValOrigin, const U32 maxWeight, + const U32 nbBitsBaseline) { - U32 rankVal[HUF_TABLELOG_MAX + 1]; - const int scaleLog = nbBitsBaseline - targetLog; /* note : targetLog >= srcLog, hence scaleLog <= 1 */ - const U32 minBits = nbBitsBaseline - maxWeight; - U32 s; + U32 rankVal[HUF_TABLELOG_MAX + 1]; + const int scaleLog = nbBitsBaseline - targetLog; /* note : targetLog >= srcLog, hence scaleLog <= 1 */ + const U32 minBits = nbBitsBaseline - maxWeight; + U32 s; - memcpy(rankVal, rankValOrigin, sizeof(rankVal)); + memcpy(rankVal, rankValOrigin, sizeof(rankVal)); - /* fill DTable */ - for (s=0; s= minBits) { /* enough room for a second symbol */ - U32 sortedRank; - int minWeight = nbBits + scaleLog; - if (minWeight < 1) minWeight = 1; - sortedRank = rankStart[minWeight]; - HUF_fillDTableX4Level2(DTable+start, targetLog-nbBits, nbBits, - rankValOrigin[nbBits], minWeight, - sortedList+sortedRank, sortedListSize-sortedRank, - nbBitsBaseline, symbol); - } else { - HUF_DEltX4 DElt; - MEM_writeLE16(&(DElt.sequence), symbol); - DElt.nbBits = (BYTE)(nbBits); - DElt.length = 1; - { U32 const end = start + length; - U32 u; - for (u = start; u < end; u++) DTable[u] = DElt; - } } - rankVal[weight] += length; - } + if (targetLog-nbBits >= minBits) { /* enough room for a second symbol */ + U32 sortedRank; + int minWeight = nbBits + scaleLog; + if (minWeight < 1) minWeight = 1; + sortedRank = rankStart[minWeight]; + HUF_fillDTableX4Level2(DTable+start, targetLog-nbBits, nbBits, + rankValOrigin[nbBits], minWeight, + sortedList+sortedRank, sortedListSize-sortedRank, + nbBitsBaseline, symbol); + } else { + HUF_DEltX4 DElt; + MEM_writeLE16(&(DElt.sequence), symbol); + DElt.nbBits = (BYTE)(nbBits); + DElt.length = 1; + { U32 const end = start + length; + U32 u; + for (u = start; u < end; u++) DTable[u] = DElt; + } } + rankVal[weight] += length; + } } size_t HUF_readDTableX4 (HUF_DTable* DTable, const void* src, size_t srcSize) { - BYTE weightList[HUF_SYMBOLVALUE_MAX + 1]; - sortedSymbol_t sortedSymbol[HUF_SYMBOLVALUE_MAX + 1]; - U32 rankStats[HUF_TABLELOG_MAX + 1] = { 0 }; - U32 rankStart0[HUF_TABLELOG_MAX + 2] = { 0 }; - U32* const rankStart = rankStart0+1; - rankVal_t rankVal; - U32 tableLog, maxW, sizeOfSort, nbSymbols; - DTableDesc dtd = HUF_getDTableDesc(DTable); - U32 const maxTableLog = dtd.maxTableLog; - size_t iSize; - void* dtPtr = DTable+1; /* force compiler to avoid strict-aliasing */ - HUF_DEltX4* const dt = (HUF_DEltX4*)dtPtr; + BYTE weightList[HUF_SYMBOLVALUE_MAX + 1]; + sortedSymbol_t sortedSymbol[HUF_SYMBOLVALUE_MAX + 1]; + U32 rankStats[HUF_TABLELOG_MAX + 1] = { 0 }; + U32 rankStart0[HUF_TABLELOG_MAX + 2] = { 0 }; + U32* const rankStart = rankStart0+1; + rankVal_t rankVal; + U32 tableLog, maxW, sizeOfSort, nbSymbols; + DTableDesc dtd = HUF_getDTableDesc(DTable); + U32 const maxTableLog = dtd.maxTableLog; + size_t iSize; + void* dtPtr = DTable+1; /* force compiler to avoid strict-aliasing */ + HUF_DEltX4* const dt = (HUF_DEltX4*)dtPtr; - HUF_STATIC_ASSERT(sizeof(HUF_DEltX4) == sizeof(HUF_DTable)); /* if compiler fails here, assertion is wrong */ - if (maxTableLog > HUF_TABLELOG_MAX) return ERROR(tableLog_tooLarge); - /* memset(weightList, 0, sizeof(weightList)); */ /* is not necessary, even though some analyzer complain ... */ + HUF_STATIC_ASSERT(sizeof(HUF_DEltX4) == sizeof(HUF_DTable)); /* if compiler fails here, assertion is wrong */ + if (maxTableLog > HUF_TABLELOG_MAX) return ERROR(tableLog_tooLarge); + /* memset(weightList, 0, sizeof(weightList)); */ /* is not necessary, even though some analyzer complain ... */ - iSize = HUF_readStats(weightList, HUF_SYMBOLVALUE_MAX + 1, rankStats, &nbSymbols, &tableLog, src, srcSize); - if (HUF_isError(iSize)) return iSize; + iSize = HUF_readStats(weightList, HUF_SYMBOLVALUE_MAX + 1, rankStats, &nbSymbols, &tableLog, src, srcSize); + if (HUF_isError(iSize)) return iSize; - /* check result */ - if (tableLog > maxTableLog) return ERROR(tableLog_tooLarge); /* DTable can't fit code depth */ + /* check result */ + if (tableLog > maxTableLog) return ERROR(tableLog_tooLarge); /* DTable can't fit code depth */ - /* find maxWeight */ - for (maxW = tableLog; rankStats[maxW]==0; maxW--) {} /* necessarily finds a solution before 0 */ + /* find maxWeight */ + for (maxW = tableLog; rankStats[maxW]==0; maxW--) {} /* necessarily finds a solution before 0 */ - /* Get start index of each weight */ - { U32 w, nextRankStart = 0; - for (w=1; w> consumed; - } } } } + /* Build rankVal */ + { U32* const rankVal0 = rankVal[0]; + { int const rescale = (maxTableLog-tableLog) - 1; /* tableLog <= maxTableLog */ + U32 nextRankVal = 0; + U32 w; + for (w=1; w> consumed; + } } } } - HUF_fillDTableX4(dt, maxTableLog, - sortedSymbol, sizeOfSort, - rankStart0, rankVal, maxW, - tableLog+1); + HUF_fillDTableX4(dt, maxTableLog, + sortedSymbol, sizeOfSort, + rankStart0, rankVal, maxW, + tableLog+1); - dtd.tableLog = (BYTE)maxTableLog; - dtd.tableType = 1; - memcpy(DTable, &dtd, sizeof(dtd)); - return iSize; + dtd.tableLog = (BYTE)maxTableLog; + dtd.tableType = 1; + memcpy(DTable, &dtd, sizeof(dtd)); + return iSize; } static U32 HUF_decodeSymbolX4(void* op, BIT_DStream_t* DStream, const HUF_DEltX4* dt, const U32 dtLog) { - size_t const val = BIT_lookBitsFast(DStream, dtLog); /* note : dtLog >= 1 */ - memcpy(op, dt+val, 2); - BIT_skipBits(DStream, dt[val].nbBits); - return dt[val].length; + size_t const val = BIT_lookBitsFast(DStream, dtLog); /* note : dtLog >= 1 */ + memcpy(op, dt+val, 2); + BIT_skipBits(DStream, dt[val].nbBits); + return dt[val].length; } static U32 HUF_decodeLastSymbolX4(void* op, BIT_DStream_t* DStream, const HUF_DEltX4* dt, const U32 dtLog) { - size_t const val = BIT_lookBitsFast(DStream, dtLog); /* note : dtLog >= 1 */ - memcpy(op, dt+val, 1); - if (dt[val].length==1) BIT_skipBits(DStream, dt[val].nbBits); - else { - if (DStream->bitsConsumed < (sizeof(DStream->bitContainer)*8)) { - BIT_skipBits(DStream, dt[val].nbBits); - if (DStream->bitsConsumed > (sizeof(DStream->bitContainer)*8)) - DStream->bitsConsumed = (sizeof(DStream->bitContainer)*8); /* ugly hack; works only because it's the last symbol. Note : can't easily extract nbBits from just this symbol */ - } } - return 1; + size_t const val = BIT_lookBitsFast(DStream, dtLog); /* note : dtLog >= 1 */ + memcpy(op, dt+val, 1); + if (dt[val].length==1) BIT_skipBits(DStream, dt[val].nbBits); + else { + if (DStream->bitsConsumed < (sizeof(DStream->bitContainer)*8)) { + BIT_skipBits(DStream, dt[val].nbBits); + if (DStream->bitsConsumed > (sizeof(DStream->bitContainer)*8)) + DStream->bitsConsumed = (sizeof(DStream->bitContainer)*8); /* ugly hack; works only because it's the last symbol. Note : can't easily extract nbBits from just this symbol */ + } } + return 1; } #define HUF_DECODE_SYMBOLX4_0(ptr, DStreamPtr) \ - ptr += HUF_decodeSymbolX4(ptr, DStreamPtr, dt, dtLog) + ptr += HUF_decodeSymbolX4(ptr, DStreamPtr, dt, dtLog) #define HUF_DECODE_SYMBOLX4_1(ptr, DStreamPtr) \ - if (MEM_64bits() || (HUF_TABLELOG_MAX<=12)) \ - ptr += HUF_decodeSymbolX4(ptr, DStreamPtr, dt, dtLog) + if (MEM_64bits() || (HUF_TABLELOG_MAX<=12)) \ + ptr += HUF_decodeSymbolX4(ptr, DStreamPtr, dt, dtLog) #define HUF_DECODE_SYMBOLX4_2(ptr, DStreamPtr) \ - if (MEM_64bits()) \ - ptr += HUF_decodeSymbolX4(ptr, DStreamPtr, dt, dtLog) + if (MEM_64bits()) \ + ptr += HUF_decodeSymbolX4(ptr, DStreamPtr, dt, dtLog) FORCE_INLINE size_t HUF_decodeStreamX4(BYTE* p, BIT_DStream_t* bitDPtr, BYTE* const pEnd, const HUF_DEltX4* const dt, const U32 dtLog) { - BYTE* const pStart = p; + BYTE* const pStart = p; - /* up to 8 symbols at a time */ - while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) & (p < pEnd-(sizeof(bitDPtr->bitContainer)-1))) { - HUF_DECODE_SYMBOLX4_2(p, bitDPtr); - HUF_DECODE_SYMBOLX4_1(p, bitDPtr); - HUF_DECODE_SYMBOLX4_2(p, bitDPtr); - HUF_DECODE_SYMBOLX4_0(p, bitDPtr); - } + /* up to 8 symbols at a time */ + while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) & (p < pEnd-(sizeof(bitDPtr->bitContainer)-1))) { + HUF_DECODE_SYMBOLX4_2(p, bitDPtr); + HUF_DECODE_SYMBOLX4_1(p, bitDPtr); + HUF_DECODE_SYMBOLX4_2(p, bitDPtr); + HUF_DECODE_SYMBOLX4_0(p, bitDPtr); + } - /* closer to end : up to 2 symbols at a time */ - while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) & (p <= pEnd-2)) - HUF_DECODE_SYMBOLX4_0(p, bitDPtr); + /* closer to end : up to 2 symbols at a time */ + while ((BIT_reloadDStream(bitDPtr) == BIT_DStream_unfinished) & (p <= pEnd-2)) + HUF_DECODE_SYMBOLX4_0(p, bitDPtr); - while (p <= pEnd-2) - HUF_DECODE_SYMBOLX4_0(p, bitDPtr); /* no need to reload : reached the end of DStream */ + while (p <= pEnd-2) + HUF_DECODE_SYMBOLX4_0(p, bitDPtr); /* no need to reload : reached the end of DStream */ - if (p < pEnd) - p += HUF_decodeLastSymbolX4(p, bitDPtr, dt, dtLog); + if (p < pEnd) + p += HUF_decodeLastSymbolX4(p, bitDPtr, dt, dtLog); - return p-pStart; + return p-pStart; } static size_t HUF_decompress1X4_usingDTable_internal( - void* dst, size_t dstSize, - const void* cSrc, size_t cSrcSize, - const HUF_DTable* DTable) + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) { - BIT_DStream_t bitD; + BIT_DStream_t bitD; - /* Init */ - { size_t const errorCode = BIT_initDStream(&bitD, cSrc, cSrcSize); - if (HUF_isError(errorCode)) return errorCode; - } + /* Init */ + { size_t const errorCode = BIT_initDStream(&bitD, cSrc, cSrcSize); + if (HUF_isError(errorCode)) return errorCode; + } - /* decode */ - { BYTE* const ostart = (BYTE*) dst; - BYTE* const oend = ostart + dstSize; - const void* const dtPtr = DTable+1; /* force compiler to not use strict-aliasing */ - const HUF_DEltX4* const dt = (const HUF_DEltX4*)dtPtr; - DTableDesc const dtd = HUF_getDTableDesc(DTable); - HUF_decodeStreamX4(ostart, &bitD, oend, dt, dtd.tableLog); - } + /* decode */ + { BYTE* const ostart = (BYTE*) dst; + BYTE* const oend = ostart + dstSize; + const void* const dtPtr = DTable+1; /* force compiler to not use strict-aliasing */ + const HUF_DEltX4* const dt = (const HUF_DEltX4*)dtPtr; + DTableDesc const dtd = HUF_getDTableDesc(DTable); + HUF_decodeStreamX4(ostart, &bitD, oend, dt, dtd.tableLog); + } - /* check */ - if (!BIT_endOfDStream(&bitD)) return ERROR(corruption_detected); + /* check */ + if (!BIT_endOfDStream(&bitD)) return ERROR(corruption_detected); - /* decoded size */ - return dstSize; + /* decoded size */ + return dstSize; } size_t HUF_decompress1X4_usingDTable( - void* dst, size_t dstSize, - const void* cSrc, size_t cSrcSize, - const HUF_DTable* DTable) + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) { - DTableDesc dtd = HUF_getDTableDesc(DTable); - if (dtd.tableType != 1) return ERROR(GENERIC); - return HUF_decompress1X4_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); + DTableDesc dtd = HUF_getDTableDesc(DTable); + if (dtd.tableType != 1) return ERROR(GENERIC); + return HUF_decompress1X4_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); } size_t HUF_decompress1X4_DCtx (HUF_DTable* DCtx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - const BYTE* ip = (const BYTE*) cSrc; + const BYTE* ip = (const BYTE*) cSrc; - size_t const hSize = HUF_readDTableX4 (DCtx, cSrc, cSrcSize); - if (HUF_isError(hSize)) return hSize; - if (hSize >= cSrcSize) return ERROR(srcSize_wrong); - ip += hSize; cSrcSize -= hSize; + size_t const hSize = HUF_readDTableX4 (DCtx, cSrc, cSrcSize); + if (HUF_isError(hSize)) return hSize; + if (hSize >= cSrcSize) return ERROR(srcSize_wrong); + ip += hSize; cSrcSize -= hSize; - return HUF_decompress1X4_usingDTable_internal (dst, dstSize, ip, cSrcSize, DCtx); + return HUF_decompress1X4_usingDTable_internal (dst, dstSize, ip, cSrcSize, DCtx); } size_t HUF_decompress1X4 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - HUF_CREATE_STATIC_DTABLEX4(DTable, HUF_TABLELOG_MAX); - return HUF_decompress1X4_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); + HUF_CREATE_STATIC_DTABLEX4(DTable, HUF_TABLELOG_MAX); + return HUF_decompress1X4_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); } static size_t HUF_decompress4X4_usingDTable_internal( - void* dst, size_t dstSize, - const void* cSrc, size_t cSrcSize, - const HUF_DTable* DTable) + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) { - if (cSrcSize < 10) return ERROR(corruption_detected); /* strict minimum : jump table + 1 byte per stream */ + if (cSrcSize < 10) return ERROR(corruption_detected); /* strict minimum : jump table + 1 byte per stream */ - { const BYTE* const istart = (const BYTE*) cSrc; - BYTE* const ostart = (BYTE*) dst; - BYTE* const oend = ostart + dstSize; - const void* const dtPtr = DTable+1; - const HUF_DEltX4* const dt = (const HUF_DEltX4*)dtPtr; + { const BYTE* const istart = (const BYTE*) cSrc; + BYTE* const ostart = (BYTE*) dst; + BYTE* const oend = ostart + dstSize; + const void* const dtPtr = DTable+1; + const HUF_DEltX4* const dt = (const HUF_DEltX4*)dtPtr; - /* Init */ - BIT_DStream_t bitD1; - BIT_DStream_t bitD2; - BIT_DStream_t bitD3; - BIT_DStream_t bitD4; - size_t const length1 = MEM_readLE16(istart); - size_t const length2 = MEM_readLE16(istart+2); - size_t const length3 = MEM_readLE16(istart+4); - size_t const length4 = cSrcSize - (length1 + length2 + length3 + 6); - const BYTE* const istart1 = istart + 6; /* jumpTable */ - const BYTE* const istart2 = istart1 + length1; - const BYTE* const istart3 = istart2 + length2; - const BYTE* const istart4 = istart3 + length3; - size_t const segmentSize = (dstSize+3) / 4; - BYTE* const opStart2 = ostart + segmentSize; - BYTE* const opStart3 = opStart2 + segmentSize; - BYTE* const opStart4 = opStart3 + segmentSize; - BYTE* op1 = ostart; - BYTE* op2 = opStart2; - BYTE* op3 = opStart3; - BYTE* op4 = opStart4; - U32 endSignal; - DTableDesc const dtd = HUF_getDTableDesc(DTable); - U32 const dtLog = dtd.tableLog; + /* Init */ + BIT_DStream_t bitD1; + BIT_DStream_t bitD2; + BIT_DStream_t bitD3; + BIT_DStream_t bitD4; + size_t const length1 = MEM_readLE16(istart); + size_t const length2 = MEM_readLE16(istart+2); + size_t const length3 = MEM_readLE16(istart+4); + size_t const length4 = cSrcSize - (length1 + length2 + length3 + 6); + const BYTE* const istart1 = istart + 6; /* jumpTable */ + const BYTE* const istart2 = istart1 + length1; + const BYTE* const istart3 = istart2 + length2; + const BYTE* const istart4 = istart3 + length3; + size_t const segmentSize = (dstSize+3) / 4; + BYTE* const opStart2 = ostart + segmentSize; + BYTE* const opStart3 = opStart2 + segmentSize; + BYTE* const opStart4 = opStart3 + segmentSize; + BYTE* op1 = ostart; + BYTE* op2 = opStart2; + BYTE* op3 = opStart3; + BYTE* op4 = opStart4; + U32 endSignal; + DTableDesc const dtd = HUF_getDTableDesc(DTable); + U32 const dtLog = dtd.tableLog; - if (length4 > cSrcSize) return ERROR(corruption_detected); /* overflow */ - { size_t const errorCode = BIT_initDStream(&bitD1, istart1, length1); - if (HUF_isError(errorCode)) return errorCode; } - { size_t const errorCode = BIT_initDStream(&bitD2, istart2, length2); - if (HUF_isError(errorCode)) return errorCode; } - { size_t const errorCode = BIT_initDStream(&bitD3, istart3, length3); - if (HUF_isError(errorCode)) return errorCode; } - { size_t const errorCode = BIT_initDStream(&bitD4, istart4, length4); - if (HUF_isError(errorCode)) return errorCode; } + if (length4 > cSrcSize) return ERROR(corruption_detected); /* overflow */ + { size_t const errorCode = BIT_initDStream(&bitD1, istart1, length1); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD2, istart2, length2); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD3, istart3, length3); + if (HUF_isError(errorCode)) return errorCode; } + { size_t const errorCode = BIT_initDStream(&bitD4, istart4, length4); + if (HUF_isError(errorCode)) return errorCode; } - /* 16-32 symbols per loop (4-8 symbols per stream) */ - endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); - for ( ; (endSignal==BIT_DStream_unfinished) & (op4<(oend-(sizeof(bitD4.bitContainer)-1))) ; ) { - HUF_DECODE_SYMBOLX4_2(op1, &bitD1); - HUF_DECODE_SYMBOLX4_2(op2, &bitD2); - HUF_DECODE_SYMBOLX4_2(op3, &bitD3); - HUF_DECODE_SYMBOLX4_2(op4, &bitD4); - HUF_DECODE_SYMBOLX4_1(op1, &bitD1); - HUF_DECODE_SYMBOLX4_1(op2, &bitD2); - HUF_DECODE_SYMBOLX4_1(op3, &bitD3); - HUF_DECODE_SYMBOLX4_1(op4, &bitD4); - HUF_DECODE_SYMBOLX4_2(op1, &bitD1); - HUF_DECODE_SYMBOLX4_2(op2, &bitD2); - HUF_DECODE_SYMBOLX4_2(op3, &bitD3); - HUF_DECODE_SYMBOLX4_2(op4, &bitD4); - HUF_DECODE_SYMBOLX4_0(op1, &bitD1); - HUF_DECODE_SYMBOLX4_0(op2, &bitD2); - HUF_DECODE_SYMBOLX4_0(op3, &bitD3); - HUF_DECODE_SYMBOLX4_0(op4, &bitD4); + /* 16-32 symbols per loop (4-8 symbols per stream) */ + endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); + for ( ; (endSignal==BIT_DStream_unfinished) & (op4<(oend-(sizeof(bitD4.bitContainer)-1))) ; ) { + HUF_DECODE_SYMBOLX4_2(op1, &bitD1); + HUF_DECODE_SYMBOLX4_2(op2, &bitD2); + HUF_DECODE_SYMBOLX4_2(op3, &bitD3); + HUF_DECODE_SYMBOLX4_2(op4, &bitD4); + HUF_DECODE_SYMBOLX4_1(op1, &bitD1); + HUF_DECODE_SYMBOLX4_1(op2, &bitD2); + HUF_DECODE_SYMBOLX4_1(op3, &bitD3); + HUF_DECODE_SYMBOLX4_1(op4, &bitD4); + HUF_DECODE_SYMBOLX4_2(op1, &bitD1); + HUF_DECODE_SYMBOLX4_2(op2, &bitD2); + HUF_DECODE_SYMBOLX4_2(op3, &bitD3); + HUF_DECODE_SYMBOLX4_2(op4, &bitD4); + HUF_DECODE_SYMBOLX4_0(op1, &bitD1); + HUF_DECODE_SYMBOLX4_0(op2, &bitD2); + HUF_DECODE_SYMBOLX4_0(op3, &bitD3); + HUF_DECODE_SYMBOLX4_0(op4, &bitD4); - endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); - } + endSignal = BIT_reloadDStream(&bitD1) | BIT_reloadDStream(&bitD2) | BIT_reloadDStream(&bitD3) | BIT_reloadDStream(&bitD4); + } - /* check corruption */ - if (op1 > opStart2) return ERROR(corruption_detected); - if (op2 > opStart3) return ERROR(corruption_detected); - if (op3 > opStart4) return ERROR(corruption_detected); - /* note : op4 already verified within main loop */ + /* check corruption */ + if (op1 > opStart2) return ERROR(corruption_detected); + if (op2 > opStart3) return ERROR(corruption_detected); + if (op3 > opStart4) return ERROR(corruption_detected); + /* note : op4 already verified within main loop */ - /* finish bitStreams one by one */ - HUF_decodeStreamX4(op1, &bitD1, opStart2, dt, dtLog); - HUF_decodeStreamX4(op2, &bitD2, opStart3, dt, dtLog); - HUF_decodeStreamX4(op3, &bitD3, opStart4, dt, dtLog); - HUF_decodeStreamX4(op4, &bitD4, oend, dt, dtLog); + /* finish bitStreams one by one */ + HUF_decodeStreamX4(op1, &bitD1, opStart2, dt, dtLog); + HUF_decodeStreamX4(op2, &bitD2, opStart3, dt, dtLog); + HUF_decodeStreamX4(op3, &bitD3, opStart4, dt, dtLog); + HUF_decodeStreamX4(op4, &bitD4, oend, dt, dtLog); - /* check */ - { U32 const endCheck = BIT_endOfDStream(&bitD1) & BIT_endOfDStream(&bitD2) & BIT_endOfDStream(&bitD3) & BIT_endOfDStream(&bitD4); - if (!endCheck) return ERROR(corruption_detected); } + /* check */ + { U32 const endCheck = BIT_endOfDStream(&bitD1) & BIT_endOfDStream(&bitD2) & BIT_endOfDStream(&bitD3) & BIT_endOfDStream(&bitD4); + if (!endCheck) return ERROR(corruption_detected); } - /* decoded size */ - return dstSize; - } + /* decoded size */ + return dstSize; + } } size_t HUF_decompress4X4_usingDTable( - void* dst, size_t dstSize, - const void* cSrc, size_t cSrcSize, - const HUF_DTable* DTable) + void* dst, size_t dstSize, + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) { - DTableDesc dtd = HUF_getDTableDesc(DTable); - if (dtd.tableType != 1) return ERROR(GENERIC); - return HUF_decompress4X4_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); + DTableDesc dtd = HUF_getDTableDesc(DTable); + if (dtd.tableType != 1) return ERROR(GENERIC); + return HUF_decompress4X4_usingDTable_internal(dst, dstSize, cSrc, cSrcSize, DTable); } size_t HUF_decompress4X4_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - const BYTE* ip = (const BYTE*) cSrc; + const BYTE* ip = (const BYTE*) cSrc; - size_t hSize = HUF_readDTableX4 (dctx, cSrc, cSrcSize); - if (HUF_isError(hSize)) return hSize; - if (hSize >= cSrcSize) return ERROR(srcSize_wrong); - ip += hSize; cSrcSize -= hSize; + size_t hSize = HUF_readDTableX4 (dctx, cSrc, cSrcSize); + if (HUF_isError(hSize)) return hSize; + if (hSize >= cSrcSize) return ERROR(srcSize_wrong); + ip += hSize; cSrcSize -= hSize; - return HUF_decompress4X4_usingDTable_internal(dst, dstSize, ip, cSrcSize, dctx); + return HUF_decompress4X4_usingDTable_internal(dst, dstSize, ip, cSrcSize, dctx); } size_t HUF_decompress4X4 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - HUF_CREATE_STATIC_DTABLEX4(DTable, HUF_TABLELOG_MAX); - return HUF_decompress4X4_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); + HUF_CREATE_STATIC_DTABLEX4(DTable, HUF_TABLELOG_MAX); + return HUF_decompress4X4_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); } @@ -772,44 +772,44 @@ size_t HUF_decompress4X4 (void* dst, size_t dstSize, const void* cSrc, size_t cS /* ********************************/ size_t HUF_decompress1X_usingDTable(void* dst, size_t maxDstSize, - const void* cSrc, size_t cSrcSize, - const HUF_DTable* DTable) + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) { - DTableDesc const dtd = HUF_getDTableDesc(DTable); - return dtd.tableType ? HUF_decompress1X4_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable) : - HUF_decompress1X2_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable); + DTableDesc const dtd = HUF_getDTableDesc(DTable); + return dtd.tableType ? HUF_decompress1X4_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable) : + HUF_decompress1X2_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable); } size_t HUF_decompress4X_usingDTable(void* dst, size_t maxDstSize, - const void* cSrc, size_t cSrcSize, - const HUF_DTable* DTable) + const void* cSrc, size_t cSrcSize, + const HUF_DTable* DTable) { - DTableDesc const dtd = HUF_getDTableDesc(DTable); - return dtd.tableType ? HUF_decompress4X4_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable) : - HUF_decompress4X2_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable); + DTableDesc const dtd = HUF_getDTableDesc(DTable); + return dtd.tableType ? HUF_decompress4X4_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable) : + HUF_decompress4X2_usingDTable_internal(dst, maxDstSize, cSrc, cSrcSize, DTable); } typedef struct { U32 tableTime; U32 decode256Time; } algo_time_t; static const algo_time_t algoTime[16 /* Quantization */][3 /* single, double, quad */] = { - /* single, double, quad */ - {{0,0}, {1,1}, {2,2}}, /* Q==0 : impossible */ - {{0,0}, {1,1}, {2,2}}, /* Q==1 : impossible */ - {{ 38,130}, {1313, 74}, {2151, 38}}, /* Q == 2 : 12-18% */ - {{ 448,128}, {1353, 74}, {2238, 41}}, /* Q == 3 : 18-25% */ - {{ 556,128}, {1353, 74}, {2238, 47}}, /* Q == 4 : 25-32% */ - {{ 714,128}, {1418, 74}, {2436, 53}}, /* Q == 5 : 32-38% */ - {{ 883,128}, {1437, 74}, {2464, 61}}, /* Q == 6 : 38-44% */ - {{ 897,128}, {1515, 75}, {2622, 68}}, /* Q == 7 : 44-50% */ - {{ 926,128}, {1613, 75}, {2730, 75}}, /* Q == 8 : 50-56% */ - {{ 947,128}, {1729, 77}, {3359, 77}}, /* Q == 9 : 56-62% */ - {{1107,128}, {2083, 81}, {4006, 84}}, /* Q ==10 : 62-69% */ - {{1177,128}, {2379, 87}, {4785, 88}}, /* Q ==11 : 69-75% */ - {{1242,128}, {2415, 93}, {5155, 84}}, /* Q ==12 : 75-81% */ - {{1349,128}, {2644,106}, {5260,106}}, /* Q ==13 : 81-87% */ - {{1455,128}, {2422,124}, {4174,124}}, /* Q ==14 : 87-93% */ - {{ 722,128}, {1891,145}, {1936,146}}, /* Q ==15 : 93-99% */ + /* single, double, quad */ + {{0,0}, {1,1}, {2,2}}, /* Q==0 : impossible */ + {{0,0}, {1,1}, {2,2}}, /* Q==1 : impossible */ + {{ 38,130}, {1313, 74}, {2151, 38}}, /* Q == 2 : 12-18% */ + {{ 448,128}, {1353, 74}, {2238, 41}}, /* Q == 3 : 18-25% */ + {{ 556,128}, {1353, 74}, {2238, 47}}, /* Q == 4 : 25-32% */ + {{ 714,128}, {1418, 74}, {2436, 53}}, /* Q == 5 : 32-38% */ + {{ 883,128}, {1437, 74}, {2464, 61}}, /* Q == 6 : 38-44% */ + {{ 897,128}, {1515, 75}, {2622, 68}}, /* Q == 7 : 44-50% */ + {{ 926,128}, {1613, 75}, {2730, 75}}, /* Q == 8 : 50-56% */ + {{ 947,128}, {1729, 77}, {3359, 77}}, /* Q == 9 : 56-62% */ + {{1107,128}, {2083, 81}, {4006, 84}}, /* Q ==10 : 62-69% */ + {{1177,128}, {2379, 87}, {4785, 88}}, /* Q ==11 : 69-75% */ + {{1242,128}, {2415, 93}, {5155, 84}}, /* Q ==12 : 75-81% */ + {{1349,128}, {2644,106}, {5260,106}}, /* Q ==13 : 81-87% */ + {{1455,128}, {2422,124}, {4174,124}}, /* Q ==14 : 87-93% */ + {{ 722,128}, {1891,145}, {1936,146}}, /* Q ==15 : 93-99% */ }; /** HUF_selectDecoder() : @@ -819,14 +819,14 @@ static const algo_time_t algoTime[16 /* Quantization */][3 /* single, double, qu * Assumption : 0 < cSrcSize < dstSize <= 128 KB */ U32 HUF_selectDecoder (size_t dstSize, size_t cSrcSize) { - /* decoder timing evaluation */ - U32 const Q = (U32)(cSrcSize * 16 / dstSize); /* Q < 16 since dstSize > cSrcSize */ - U32 const D256 = (U32)(dstSize >> 8); - U32 const DTime0 = algoTime[Q][0].tableTime + (algoTime[Q][0].decode256Time * D256); - U32 DTime1 = algoTime[Q][1].tableTime + (algoTime[Q][1].decode256Time * D256); - DTime1 += DTime1 >> 3; /* advantage to algorithm using less memory, for cache eviction */ + /* decoder timing evaluation */ + U32 const Q = (U32)(cSrcSize * 16 / dstSize); /* Q < 16 since dstSize > cSrcSize */ + U32 const D256 = (U32)(dstSize >> 8); + U32 const DTime0 = algoTime[Q][0].tableTime + (algoTime[Q][0].decode256Time * D256); + U32 DTime1 = algoTime[Q][1].tableTime + (algoTime[Q][1].decode256Time * D256); + DTime1 += DTime1 >> 3; /* advantage to algorithm using less memory, for cache eviction */ - return DTime1 < DTime0; + return DTime1 < DTime0; } @@ -834,55 +834,55 @@ typedef size_t (*decompressionAlgo)(void* dst, size_t dstSize, const void* cSrc, size_t HUF_decompress (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - static const decompressionAlgo decompress[2] = { HUF_decompress4X2, HUF_decompress4X4 }; + static const decompressionAlgo decompress[2] = { HUF_decompress4X2, HUF_decompress4X4 }; - /* validation checks */ - if (dstSize == 0) return ERROR(dstSize_tooSmall); - if (cSrcSize > dstSize) return ERROR(corruption_detected); /* invalid */ - if (cSrcSize == dstSize) { memcpy(dst, cSrc, dstSize); return dstSize; } /* not compressed */ - if (cSrcSize == 1) { memset(dst, *(const BYTE*)cSrc, dstSize); return dstSize; } /* RLE */ + /* validation checks */ + if (dstSize == 0) return ERROR(dstSize_tooSmall); + if (cSrcSize > dstSize) return ERROR(corruption_detected); /* invalid */ + if (cSrcSize == dstSize) { memcpy(dst, cSrc, dstSize); return dstSize; } /* not compressed */ + if (cSrcSize == 1) { memset(dst, *(const BYTE*)cSrc, dstSize); return dstSize; } /* RLE */ - { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); - return decompress[algoNb](dst, dstSize, cSrc, cSrcSize); - } + { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); + return decompress[algoNb](dst, dstSize, cSrc, cSrcSize); + } } size_t HUF_decompress4X_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - /* validation checks */ - if (dstSize == 0) return ERROR(dstSize_tooSmall); - if (cSrcSize > dstSize) return ERROR(corruption_detected); /* invalid */ - if (cSrcSize == dstSize) { memcpy(dst, cSrc, dstSize); return dstSize; } /* not compressed */ - if (cSrcSize == 1) { memset(dst, *(const BYTE*)cSrc, dstSize); return dstSize; } /* RLE */ + /* validation checks */ + if (dstSize == 0) return ERROR(dstSize_tooSmall); + if (cSrcSize > dstSize) return ERROR(corruption_detected); /* invalid */ + if (cSrcSize == dstSize) { memcpy(dst, cSrc, dstSize); return dstSize; } /* not compressed */ + if (cSrcSize == 1) { memset(dst, *(const BYTE*)cSrc, dstSize); return dstSize; } /* RLE */ - { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); - return algoNb ? HUF_decompress4X4_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) : - HUF_decompress4X2_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) ; - } + { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); + return algoNb ? HUF_decompress4X4_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) : + HUF_decompress4X2_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) ; + } } size_t HUF_decompress4X_hufOnly (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - /* validation checks */ - if (dstSize == 0) return ERROR(dstSize_tooSmall); - if ((cSrcSize >= dstSize) || (cSrcSize <= 1)) return ERROR(corruption_detected); /* invalid */ + /* validation checks */ + if (dstSize == 0) return ERROR(dstSize_tooSmall); + if ((cSrcSize >= dstSize) || (cSrcSize <= 1)) return ERROR(corruption_detected); /* invalid */ - { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); - return algoNb ? HUF_decompress4X4_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) : - HUF_decompress4X2_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) ; - } + { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); + return algoNb ? HUF_decompress4X4_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) : + HUF_decompress4X2_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) ; + } } size_t HUF_decompress1X_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { - /* validation checks */ - if (dstSize == 0) return ERROR(dstSize_tooSmall); - if (cSrcSize > dstSize) return ERROR(corruption_detected); /* invalid */ - if (cSrcSize == dstSize) { memcpy(dst, cSrc, dstSize); return dstSize; } /* not compressed */ - if (cSrcSize == 1) { memset(dst, *(const BYTE*)cSrc, dstSize); return dstSize; } /* RLE */ + /* validation checks */ + if (dstSize == 0) return ERROR(dstSize_tooSmall); + if (cSrcSize > dstSize) return ERROR(corruption_detected); /* invalid */ + if (cSrcSize == dstSize) { memcpy(dst, cSrc, dstSize); return dstSize; } /* not compressed */ + if (cSrcSize == 1) { memset(dst, *(const BYTE*)cSrc, dstSize); return dstSize; } /* RLE */ - { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); - return algoNb ? HUF_decompress1X4_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) : - HUF_decompress1X2_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) ; - } + { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); + return algoNb ? HUF_decompress1X4_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) : + HUF_decompress1X2_DCtx(dctx, dst, dstSize, cSrc, cSrcSize) ; + } } diff --git a/contrib/linux-kernel/lib/mem.h b/contrib/linux-kernel/lib/mem.h index f049d181b..75e75088b 100644 --- a/contrib/linux-kernel/lib/mem.h +++ b/contrib/linux-kernel/lib/mem.h @@ -100,8 +100,8 @@ MEM_STATIC unsigned MEM_64bits(void) { return sizeof(size_t)==8; } MEM_STATIC unsigned MEM_isLittleEndian(void) { - const union { U32 u; BYTE c[4]; } one = { 1 }; /* don't use static : performance detrimental */ - return one.c[0]; + const union { U32 u; BYTE c[4]; } one = { 1 }; /* don't use static : performance detrimental */ + return one.c[0]; } #if defined(MEM_FORCE_MEMORY_ACCESS) && (MEM_FORCE_MEMORY_ACCESS==2) @@ -122,11 +122,11 @@ MEM_STATIC void MEM_write64(void* memPtr, U64 value) { *(U64*)memPtr = value; } /* __pack instructions are safer, but compiler specific, hence potentially problematic for some compilers */ /* currently only defined for gcc and icc */ #if defined(_MSC_VER) || (defined(__INTEL_COMPILER) && defined(WIN32)) - __pragma( pack(push, 1) ) - typedef union { U16 u16; U32 u32; U64 u64; size_t st; } unalign; - __pragma( pack(pop) ) + __pragma( pack(push, 1) ) + typedef union { U16 u16; U32 u32; U64 u64; size_t st; } unalign; + __pragma( pack(pop) ) #else - typedef union { U16 u16; U32 u32; U64 u64; size_t st; } __attribute__((packed)) unalign; + typedef union { U16 u16; U32 u32; U64 u64; size_t st; } __attribute__((packed)) unalign; #endif MEM_STATIC U16 MEM_read16(const void* ptr) { return ((const unalign*)ptr)->u16; } @@ -145,37 +145,37 @@ MEM_STATIC void MEM_write64(void* memPtr, U64 value) { ((unalign*)memPtr)->u64 = MEM_STATIC U16 MEM_read16(const void* memPtr) { - U16 val; memcpy(&val, memPtr, sizeof(val)); return val; + U16 val; memcpy(&val, memPtr, sizeof(val)); return val; } MEM_STATIC U32 MEM_read32(const void* memPtr) { - U32 val; memcpy(&val, memPtr, sizeof(val)); return val; + U32 val; memcpy(&val, memPtr, sizeof(val)); return val; } MEM_STATIC U64 MEM_read64(const void* memPtr) { - U64 val; memcpy(&val, memPtr, sizeof(val)); return val; + U64 val; memcpy(&val, memPtr, sizeof(val)); return val; } MEM_STATIC size_t MEM_readST(const void* memPtr) { - size_t val; memcpy(&val, memPtr, sizeof(val)); return val; + size_t val; memcpy(&val, memPtr, sizeof(val)); return val; } MEM_STATIC void MEM_write16(void* memPtr, U16 value) { - memcpy(memPtr, &value, sizeof(value)); + memcpy(memPtr, &value, sizeof(value)); } MEM_STATIC void MEM_write32(void* memPtr, U32 value) { - memcpy(memPtr, &value, sizeof(value)); + memcpy(memPtr, &value, sizeof(value)); } MEM_STATIC void MEM_write64(void* memPtr, U64 value) { - memcpy(memPtr, &value, sizeof(value)); + memcpy(memPtr, &value, sizeof(value)); } #endif /* MEM_FORCE_MEMORY_ACCESS */ @@ -183,188 +183,188 @@ MEM_STATIC void MEM_write64(void* memPtr, U64 value) MEM_STATIC U32 MEM_swap32(U32 in) { #if defined(_MSC_VER) /* Visual Studio */ - return _byteswap_ulong(in); + return _byteswap_ulong(in); #elif defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403) - return __builtin_bswap32(in); + return __builtin_bswap32(in); #else - return ((in << 24) & 0xff000000 ) | - ((in << 8) & 0x00ff0000 ) | - ((in >> 8) & 0x0000ff00 ) | - ((in >> 24) & 0x000000ff ); + return ((in << 24) & 0xff000000 ) | + ((in << 8) & 0x00ff0000 ) | + ((in >> 8) & 0x0000ff00 ) | + ((in >> 24) & 0x000000ff ); #endif } MEM_STATIC U64 MEM_swap64(U64 in) { #if defined(_MSC_VER) /* Visual Studio */ - return _byteswap_uint64(in); + return _byteswap_uint64(in); #elif defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403) - return __builtin_bswap64(in); + return __builtin_bswap64(in); #else - return ((in << 56) & 0xff00000000000000ULL) | - ((in << 40) & 0x00ff000000000000ULL) | - ((in << 24) & 0x0000ff0000000000ULL) | - ((in << 8) & 0x000000ff00000000ULL) | - ((in >> 8) & 0x00000000ff000000ULL) | - ((in >> 24) & 0x0000000000ff0000ULL) | - ((in >> 40) & 0x000000000000ff00ULL) | - ((in >> 56) & 0x00000000000000ffULL); + return ((in << 56) & 0xff00000000000000ULL) | + ((in << 40) & 0x00ff000000000000ULL) | + ((in << 24) & 0x0000ff0000000000ULL) | + ((in << 8) & 0x000000ff00000000ULL) | + ((in >> 8) & 0x00000000ff000000ULL) | + ((in >> 24) & 0x0000000000ff0000ULL) | + ((in >> 40) & 0x000000000000ff00ULL) | + ((in >> 56) & 0x00000000000000ffULL); #endif } MEM_STATIC size_t MEM_swapST(size_t in) { - if (MEM_32bits()) - return (size_t)MEM_swap32((U32)in); - else - return (size_t)MEM_swap64((U64)in); + if (MEM_32bits()) + return (size_t)MEM_swap32((U32)in); + else + return (size_t)MEM_swap64((U64)in); } /*=== Little endian r/w ===*/ MEM_STATIC U16 MEM_readLE16(const void* memPtr) { - if (MEM_isLittleEndian()) - return MEM_read16(memPtr); - else { - const BYTE* p = (const BYTE*)memPtr; - return (U16)(p[0] + (p[1]<<8)); - } + if (MEM_isLittleEndian()) + return MEM_read16(memPtr); + else { + const BYTE* p = (const BYTE*)memPtr; + return (U16)(p[0] + (p[1]<<8)); + } } MEM_STATIC void MEM_writeLE16(void* memPtr, U16 val) { - if (MEM_isLittleEndian()) { - MEM_write16(memPtr, val); - } else { - BYTE* p = (BYTE*)memPtr; - p[0] = (BYTE)val; - p[1] = (BYTE)(val>>8); - } + if (MEM_isLittleEndian()) { + MEM_write16(memPtr, val); + } else { + BYTE* p = (BYTE*)memPtr; + p[0] = (BYTE)val; + p[1] = (BYTE)(val>>8); + } } MEM_STATIC U32 MEM_readLE24(const void* memPtr) { - return MEM_readLE16(memPtr) + (((const BYTE*)memPtr)[2] << 16); + return MEM_readLE16(memPtr) + (((const BYTE*)memPtr)[2] << 16); } MEM_STATIC void MEM_writeLE24(void* memPtr, U32 val) { - MEM_writeLE16(memPtr, (U16)val); - ((BYTE*)memPtr)[2] = (BYTE)(val>>16); + MEM_writeLE16(memPtr, (U16)val); + ((BYTE*)memPtr)[2] = (BYTE)(val>>16); } MEM_STATIC U32 MEM_readLE32(const void* memPtr) { - if (MEM_isLittleEndian()) - return MEM_read32(memPtr); - else - return MEM_swap32(MEM_read32(memPtr)); + if (MEM_isLittleEndian()) + return MEM_read32(memPtr); + else + return MEM_swap32(MEM_read32(memPtr)); } MEM_STATIC void MEM_writeLE32(void* memPtr, U32 val32) { - if (MEM_isLittleEndian()) - MEM_write32(memPtr, val32); - else - MEM_write32(memPtr, MEM_swap32(val32)); + if (MEM_isLittleEndian()) + MEM_write32(memPtr, val32); + else + MEM_write32(memPtr, MEM_swap32(val32)); } MEM_STATIC U64 MEM_readLE64(const void* memPtr) { - if (MEM_isLittleEndian()) - return MEM_read64(memPtr); - else - return MEM_swap64(MEM_read64(memPtr)); + if (MEM_isLittleEndian()) + return MEM_read64(memPtr); + else + return MEM_swap64(MEM_read64(memPtr)); } MEM_STATIC void MEM_writeLE64(void* memPtr, U64 val64) { - if (MEM_isLittleEndian()) - MEM_write64(memPtr, val64); - else - MEM_write64(memPtr, MEM_swap64(val64)); + if (MEM_isLittleEndian()) + MEM_write64(memPtr, val64); + else + MEM_write64(memPtr, MEM_swap64(val64)); } MEM_STATIC size_t MEM_readLEST(const void* memPtr) { - if (MEM_32bits()) - return (size_t)MEM_readLE32(memPtr); - else - return (size_t)MEM_readLE64(memPtr); + if (MEM_32bits()) + return (size_t)MEM_readLE32(memPtr); + else + return (size_t)MEM_readLE64(memPtr); } MEM_STATIC void MEM_writeLEST(void* memPtr, size_t val) { - if (MEM_32bits()) - MEM_writeLE32(memPtr, (U32)val); - else - MEM_writeLE64(memPtr, (U64)val); + if (MEM_32bits()) + MEM_writeLE32(memPtr, (U32)val); + else + MEM_writeLE64(memPtr, (U64)val); } /*=== Big endian r/w ===*/ MEM_STATIC U32 MEM_readBE32(const void* memPtr) { - if (MEM_isLittleEndian()) - return MEM_swap32(MEM_read32(memPtr)); - else - return MEM_read32(memPtr); + if (MEM_isLittleEndian()) + return MEM_swap32(MEM_read32(memPtr)); + else + return MEM_read32(memPtr); } MEM_STATIC void MEM_writeBE32(void* memPtr, U32 val32) { - if (MEM_isLittleEndian()) - MEM_write32(memPtr, MEM_swap32(val32)); - else - MEM_write32(memPtr, val32); + if (MEM_isLittleEndian()) + MEM_write32(memPtr, MEM_swap32(val32)); + else + MEM_write32(memPtr, val32); } MEM_STATIC U64 MEM_readBE64(const void* memPtr) { - if (MEM_isLittleEndian()) - return MEM_swap64(MEM_read64(memPtr)); - else - return MEM_read64(memPtr); + if (MEM_isLittleEndian()) + return MEM_swap64(MEM_read64(memPtr)); + else + return MEM_read64(memPtr); } MEM_STATIC void MEM_writeBE64(void* memPtr, U64 val64) { - if (MEM_isLittleEndian()) - MEM_write64(memPtr, MEM_swap64(val64)); - else - MEM_write64(memPtr, val64); + if (MEM_isLittleEndian()) + MEM_write64(memPtr, MEM_swap64(val64)); + else + MEM_write64(memPtr, val64); } MEM_STATIC size_t MEM_readBEST(const void* memPtr) { - if (MEM_32bits()) - return (size_t)MEM_readBE32(memPtr); - else - return (size_t)MEM_readBE64(memPtr); + if (MEM_32bits()) + return (size_t)MEM_readBE32(memPtr); + else + return (size_t)MEM_readBE64(memPtr); } MEM_STATIC void MEM_writeBEST(void* memPtr, size_t val) { - if (MEM_32bits()) - MEM_writeBE32(memPtr, (U32)val); - else - MEM_writeBE64(memPtr, (U64)val); + if (MEM_32bits()) + MEM_writeBE32(memPtr, (U32)val); + else + MEM_writeBE64(memPtr, (U64)val); } /* function safe only for comparisons */ MEM_STATIC U32 MEM_readMINMATCH(const void* memPtr, U32 length) { - switch (length) - { - default : - case 4 : return MEM_read32(memPtr); - case 3 : if (MEM_isLittleEndian()) - return MEM_read32(memPtr)<<8; - else - return MEM_read32(memPtr)>>8; - } + switch (length) + { + default : + case 4 : return MEM_read32(memPtr); + case 3 : if (MEM_isLittleEndian()) + return MEM_read32(memPtr)<<8; + else + return MEM_read32(memPtr)>>8; + } } #if defined (__cplusplus) diff --git a/contrib/linux-kernel/lib/xxhash.c b/contrib/linux-kernel/lib/xxhash.c index eb44222c5..19b8bb46b 100644 --- a/contrib/linux-kernel/lib/xxhash.c +++ b/contrib/linux-kernel/lib/xxhash.c @@ -136,17 +136,17 @@ static void* XXH_memcpy(void* dest, const void* src, size_t size) { return memcp # define MEM_MODULE # if !defined (__VMS) && (defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) ) # include - typedef uint8_t BYTE; - typedef uint16_t U16; - typedef uint32_t U32; - typedef int32_t S32; - typedef uint64_t U64; + typedef uint8_t BYTE; + typedef uint16_t U16; + typedef uint32_t U32; + typedef int32_t S32; + typedef uint64_t U64; # else - typedef unsigned char BYTE; - typedef unsigned short U16; - typedef unsigned int U32; - typedef signed int S32; - typedef unsigned long long U64; /* if your compiler doesn't support unsigned long long, replace by another 64-bit type here. Note that xxhash.h will also need to be updated. */ + typedef unsigned char BYTE; + typedef unsigned short U16; + typedef unsigned int U32; + typedef signed int S32; + typedef unsigned long long U64; /* if your compiler doesn't support unsigned long long, replace by another 64-bit type here. Note that xxhash.h will also need to be updated. */ # endif #endif @@ -174,16 +174,16 @@ static U64 XXH_read64(const void* ptr) { return ((const unalign*)ptr)->u64; } static U32 XXH_read32(const void* memPtr) { - U32 val; - memcpy(&val, memPtr, sizeof(val)); - return val; + U32 val; + memcpy(&val, memPtr, sizeof(val)); + return val; } static U64 XXH_read64(const void* memPtr) { - U64 val; - memcpy(&val, memPtr, sizeof(val)); - return val; + U64 val; + memcpy(&val, memPtr, sizeof(val)); + return val; } #endif /* XXH_FORCE_DIRECT_MEMORY_ACCESS */ @@ -212,21 +212,21 @@ static U64 XXH_read64(const void* memPtr) #else static U32 XXH_swap32 (U32 x) { - return ((x << 24) & 0xff000000 ) | - ((x << 8) & 0x00ff0000 ) | - ((x >> 8) & 0x0000ff00 ) | - ((x >> 24) & 0x000000ff ); + return ((x << 24) & 0xff000000 ) | + ((x << 8) & 0x00ff0000 ) | + ((x >> 8) & 0x0000ff00 ) | + ((x >> 24) & 0x000000ff ); } static U64 XXH_swap64 (U64 x) { - return ((x << 56) & 0xff00000000000000ULL) | - ((x << 40) & 0x00ff000000000000ULL) | - ((x << 24) & 0x0000ff0000000000ULL) | - ((x << 8) & 0x000000ff00000000ULL) | - ((x >> 8) & 0x00000000ff000000ULL) | - ((x >> 24) & 0x0000000000ff0000ULL) | - ((x >> 40) & 0x000000000000ff00ULL) | - ((x >> 56) & 0x00000000000000ffULL); + return ((x << 56) & 0xff00000000000000ULL) | + ((x << 40) & 0x00ff000000000000ULL) | + ((x << 24) & 0x0000ff0000000000ULL) | + ((x << 8) & 0x000000ff00000000ULL) | + ((x >> 8) & 0x00000000ff000000ULL) | + ((x >> 24) & 0x0000000000ff0000ULL) | + ((x >> 40) & 0x000000000000ff00ULL) | + ((x >> 56) & 0x00000000000000ffULL); } #endif @@ -238,7 +238,7 @@ typedef enum { XXH_bigEndian=0, XXH_littleEndian=1 } XXH_endianess; /* XXH_CPU_LITTLE_ENDIAN can be defined externally, for example on the compiler command line */ #ifndef XXH_CPU_LITTLE_ENDIAN - static const int g_one = 1; + static const int g_one = 1; # define XXH_CPU_LITTLE_ENDIAN (*(const char*)(&g_one)) #endif @@ -250,38 +250,38 @@ typedef enum { XXH_aligned, XXH_unaligned } XXH_alignment; FORCE_INLINE U32 XXH_readLE32_align(const void* ptr, XXH_endianess endian, XXH_alignment align) { - if (align==XXH_unaligned) - return endian==XXH_littleEndian ? XXH_read32(ptr) : XXH_swap32(XXH_read32(ptr)); - else - return endian==XXH_littleEndian ? *(const U32*)ptr : XXH_swap32(*(const U32*)ptr); + if (align==XXH_unaligned) + return endian==XXH_littleEndian ? XXH_read32(ptr) : XXH_swap32(XXH_read32(ptr)); + else + return endian==XXH_littleEndian ? *(const U32*)ptr : XXH_swap32(*(const U32*)ptr); } FORCE_INLINE U32 XXH_readLE32(const void* ptr, XXH_endianess endian) { - return XXH_readLE32_align(ptr, endian, XXH_unaligned); + return XXH_readLE32_align(ptr, endian, XXH_unaligned); } static U32 XXH_readBE32(const void* ptr) { - return XXH_CPU_LITTLE_ENDIAN ? XXH_swap32(XXH_read32(ptr)) : XXH_read32(ptr); + return XXH_CPU_LITTLE_ENDIAN ? XXH_swap32(XXH_read32(ptr)) : XXH_read32(ptr); } FORCE_INLINE U64 XXH_readLE64_align(const void* ptr, XXH_endianess endian, XXH_alignment align) { - if (align==XXH_unaligned) - return endian==XXH_littleEndian ? XXH_read64(ptr) : XXH_swap64(XXH_read64(ptr)); - else - return endian==XXH_littleEndian ? *(const U64*)ptr : XXH_swap64(*(const U64*)ptr); + if (align==XXH_unaligned) + return endian==XXH_littleEndian ? XXH_read64(ptr) : XXH_swap64(XXH_read64(ptr)); + else + return endian==XXH_littleEndian ? *(const U64*)ptr : XXH_swap64(*(const U64*)ptr); } FORCE_INLINE U64 XXH_readLE64(const void* ptr, XXH_endianess endian) { - return XXH_readLE64_align(ptr, endian, XXH_unaligned); + return XXH_readLE64_align(ptr, endian, XXH_unaligned); } static U64 XXH_readBE64(const void* ptr) { - return XXH_CPU_LITTLE_ENDIAN ? XXH_swap64(XXH_read64(ptr)) : XXH_read64(ptr); + return XXH_CPU_LITTLE_ENDIAN ? XXH_swap64(XXH_read64(ptr)) : XXH_read64(ptr); } @@ -314,12 +314,12 @@ XXH_PUBLIC_API unsigned XXH_versionNumber (void) { return XXH_VERSION_NUMBER; } ****************************/ XXH_PUBLIC_API void XXH32_copyState(XXH32_state_t* restrict dstState, const XXH32_state_t* restrict srcState) { - memcpy(dstState, srcState, sizeof(*dstState)); + memcpy(dstState, srcState, sizeof(*dstState)); } XXH_PUBLIC_API void XXH64_copyState(XXH64_state_t* restrict dstState, const XXH64_state_t* restrict srcState) { - memcpy(dstState, srcState, sizeof(*dstState)); + memcpy(dstState, srcState, sizeof(*dstState)); } @@ -329,204 +329,204 @@ XXH_PUBLIC_API void XXH64_copyState(XXH64_state_t* restrict dstState, const XXH6 static U32 XXH32_round(U32 seed, U32 input) { - seed += input * PRIME32_2; - seed = XXH_rotl32(seed, 13); - seed *= PRIME32_1; - return seed; + seed += input * PRIME32_2; + seed = XXH_rotl32(seed, 13); + seed *= PRIME32_1; + return seed; } FORCE_INLINE U32 XXH32_endian_align(const void* input, size_t len, U32 seed, XXH_endianess endian, XXH_alignment align) { - const BYTE* p = (const BYTE*)input; - const BYTE* bEnd = p + len; - U32 h32; + const BYTE* p = (const BYTE*)input; + const BYTE* bEnd = p + len; + U32 h32; #define XXH_get32bits(p) XXH_readLE32_align(p, endian, align) #ifdef XXH_ACCEPT_NULL_INPUT_POINTER - if (p==NULL) { - len=0; - bEnd=p=(const BYTE*)(size_t)16; - } + if (p==NULL) { + len=0; + bEnd=p=(const BYTE*)(size_t)16; + } #endif - if (len>=16) { - const BYTE* const limit = bEnd - 16; - U32 v1 = seed + PRIME32_1 + PRIME32_2; - U32 v2 = seed + PRIME32_2; - U32 v3 = seed + 0; - U32 v4 = seed - PRIME32_1; + if (len>=16) { + const BYTE* const limit = bEnd - 16; + U32 v1 = seed + PRIME32_1 + PRIME32_2; + U32 v2 = seed + PRIME32_2; + U32 v3 = seed + 0; + U32 v4 = seed - PRIME32_1; - do { - v1 = XXH32_round(v1, XXH_get32bits(p)); p+=4; - v2 = XXH32_round(v2, XXH_get32bits(p)); p+=4; - v3 = XXH32_round(v3, XXH_get32bits(p)); p+=4; - v4 = XXH32_round(v4, XXH_get32bits(p)); p+=4; - } while (p<=limit); + do { + v1 = XXH32_round(v1, XXH_get32bits(p)); p+=4; + v2 = XXH32_round(v2, XXH_get32bits(p)); p+=4; + v3 = XXH32_round(v3, XXH_get32bits(p)); p+=4; + v4 = XXH32_round(v4, XXH_get32bits(p)); p+=4; + } while (p<=limit); - h32 = XXH_rotl32(v1, 1) + XXH_rotl32(v2, 7) + XXH_rotl32(v3, 12) + XXH_rotl32(v4, 18); - } else { - h32 = seed + PRIME32_5; - } + h32 = XXH_rotl32(v1, 1) + XXH_rotl32(v2, 7) + XXH_rotl32(v3, 12) + XXH_rotl32(v4, 18); + } else { + h32 = seed + PRIME32_5; + } - h32 += (U32) len; + h32 += (U32) len; - while (p+4<=bEnd) { - h32 += XXH_get32bits(p) * PRIME32_3; - h32 = XXH_rotl32(h32, 17) * PRIME32_4 ; - p+=4; - } + while (p+4<=bEnd) { + h32 += XXH_get32bits(p) * PRIME32_3; + h32 = XXH_rotl32(h32, 17) * PRIME32_4 ; + p+=4; + } - while (p> 15; - h32 *= PRIME32_2; - h32 ^= h32 >> 13; - h32 *= PRIME32_3; - h32 ^= h32 >> 16; + h32 ^= h32 >> 15; + h32 *= PRIME32_2; + h32 ^= h32 >> 13; + h32 *= PRIME32_3; + h32 ^= h32 >> 16; - return h32; + return h32; } XXH_PUBLIC_API unsigned int XXH32 (const void* input, size_t len, unsigned int seed) { #if 0 - /* Simple version, good for code maintenance, but unfortunately slow for small inputs */ - XXH32_CREATESTATE_STATIC(state); - XXH32_reset(state, seed); - XXH32_update(state, input, len); - return XXH32_digest(state); + /* Simple version, good for code maintenance, but unfortunately slow for small inputs */ + XXH32_CREATESTATE_STATIC(state); + XXH32_reset(state, seed); + XXH32_update(state, input, len); + return XXH32_digest(state); #else - XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; - if (XXH_FORCE_ALIGN_CHECK) { - if ((((size_t)input) & 3) == 0) { /* Input is 4-bytes aligned, leverage the speed benefit */ - if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) - return XXH32_endian_align(input, len, seed, XXH_littleEndian, XXH_aligned); - else - return XXH32_endian_align(input, len, seed, XXH_bigEndian, XXH_aligned); - } } + if (XXH_FORCE_ALIGN_CHECK) { + if ((((size_t)input) & 3) == 0) { /* Input is 4-bytes aligned, leverage the speed benefit */ + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH32_endian_align(input, len, seed, XXH_littleEndian, XXH_aligned); + else + return XXH32_endian_align(input, len, seed, XXH_bigEndian, XXH_aligned); + } } - if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) - return XXH32_endian_align(input, len, seed, XXH_littleEndian, XXH_unaligned); - else - return XXH32_endian_align(input, len, seed, XXH_bigEndian, XXH_unaligned); + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH32_endian_align(input, len, seed, XXH_littleEndian, XXH_unaligned); + else + return XXH32_endian_align(input, len, seed, XXH_bigEndian, XXH_unaligned); #endif } static U64 XXH64_round(U64 acc, U64 input) { - acc += input * PRIME64_2; - acc = XXH_rotl64(acc, 31); - acc *= PRIME64_1; - return acc; + acc += input * PRIME64_2; + acc = XXH_rotl64(acc, 31); + acc *= PRIME64_1; + return acc; } static U64 XXH64_mergeRound(U64 acc, U64 val) { - val = XXH64_round(0, val); - acc ^= val; - acc = acc * PRIME64_1 + PRIME64_4; - return acc; + val = XXH64_round(0, val); + acc ^= val; + acc = acc * PRIME64_1 + PRIME64_4; + return acc; } FORCE_INLINE U64 XXH64_endian_align(const void* input, size_t len, U64 seed, XXH_endianess endian, XXH_alignment align) { - const BYTE* p = (const BYTE*)input; - const BYTE* const bEnd = p + len; - U64 h64; + const BYTE* p = (const BYTE*)input; + const BYTE* const bEnd = p + len; + U64 h64; #define XXH_get64bits(p) XXH_readLE64_align(p, endian, align) #ifdef XXH_ACCEPT_NULL_INPUT_POINTER - if (p==NULL) { - len=0; - bEnd=p=(const BYTE*)(size_t)32; - } + if (p==NULL) { + len=0; + bEnd=p=(const BYTE*)(size_t)32; + } #endif - if (len>=32) { - const BYTE* const limit = bEnd - 32; - U64 v1 = seed + PRIME64_1 + PRIME64_2; - U64 v2 = seed + PRIME64_2; - U64 v3 = seed + 0; - U64 v4 = seed - PRIME64_1; + if (len>=32) { + const BYTE* const limit = bEnd - 32; + U64 v1 = seed + PRIME64_1 + PRIME64_2; + U64 v2 = seed + PRIME64_2; + U64 v3 = seed + 0; + U64 v4 = seed - PRIME64_1; - do { - v1 = XXH64_round(v1, XXH_get64bits(p)); p+=8; - v2 = XXH64_round(v2, XXH_get64bits(p)); p+=8; - v3 = XXH64_round(v3, XXH_get64bits(p)); p+=8; - v4 = XXH64_round(v4, XXH_get64bits(p)); p+=8; - } while (p<=limit); + do { + v1 = XXH64_round(v1, XXH_get64bits(p)); p+=8; + v2 = XXH64_round(v2, XXH_get64bits(p)); p+=8; + v3 = XXH64_round(v3, XXH_get64bits(p)); p+=8; + v4 = XXH64_round(v4, XXH_get64bits(p)); p+=8; + } while (p<=limit); - h64 = XXH_rotl64(v1, 1) + XXH_rotl64(v2, 7) + XXH_rotl64(v3, 12) + XXH_rotl64(v4, 18); - h64 = XXH64_mergeRound(h64, v1); - h64 = XXH64_mergeRound(h64, v2); - h64 = XXH64_mergeRound(h64, v3); - h64 = XXH64_mergeRound(h64, v4); + h64 = XXH_rotl64(v1, 1) + XXH_rotl64(v2, 7) + XXH_rotl64(v3, 12) + XXH_rotl64(v4, 18); + h64 = XXH64_mergeRound(h64, v1); + h64 = XXH64_mergeRound(h64, v2); + h64 = XXH64_mergeRound(h64, v3); + h64 = XXH64_mergeRound(h64, v4); - } else { - h64 = seed + PRIME64_5; - } + } else { + h64 = seed + PRIME64_5; + } - h64 += (U64) len; + h64 += (U64) len; - while (p+8<=bEnd) { - U64 const k1 = XXH64_round(0, XXH_get64bits(p)); - h64 ^= k1; - h64 = XXH_rotl64(h64,27) * PRIME64_1 + PRIME64_4; - p+=8; - } + while (p+8<=bEnd) { + U64 const k1 = XXH64_round(0, XXH_get64bits(p)); + h64 ^= k1; + h64 = XXH_rotl64(h64,27) * PRIME64_1 + PRIME64_4; + p+=8; + } - if (p+4<=bEnd) { - h64 ^= (U64)(XXH_get32bits(p)) * PRIME64_1; - h64 = XXH_rotl64(h64, 23) * PRIME64_2 + PRIME64_3; - p+=4; - } + if (p+4<=bEnd) { + h64 ^= (U64)(XXH_get32bits(p)) * PRIME64_1; + h64 = XXH_rotl64(h64, 23) * PRIME64_2 + PRIME64_3; + p+=4; + } - while (p> 33; - h64 *= PRIME64_2; - h64 ^= h64 >> 29; - h64 *= PRIME64_3; - h64 ^= h64 >> 32; + h64 ^= h64 >> 33; + h64 *= PRIME64_2; + h64 ^= h64 >> 29; + h64 *= PRIME64_3; + h64 ^= h64 >> 32; - return h64; + return h64; } XXH_PUBLIC_API unsigned long long XXH64 (const void* input, size_t len, unsigned long long seed) { #if 0 - /* Simple version, good for code maintenance, but unfortunately slow for small inputs */ - XXH64_CREATESTATE_STATIC(state); - XXH64_reset(state, seed); - XXH64_update(state, input, len); - return XXH64_digest(state); + /* Simple version, good for code maintenance, but unfortunately slow for small inputs */ + XXH64_CREATESTATE_STATIC(state); + XXH64_reset(state, seed); + XXH64_update(state, input, len); + return XXH64_digest(state); #else - XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; - if (XXH_FORCE_ALIGN_CHECK) { - if ((((size_t)input) & 7)==0) { /* Input is aligned, let's leverage the speed advantage */ - if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) - return XXH64_endian_align(input, len, seed, XXH_littleEndian, XXH_aligned); - else - return XXH64_endian_align(input, len, seed, XXH_bigEndian, XXH_aligned); - } } + if (XXH_FORCE_ALIGN_CHECK) { + if ((((size_t)input) & 7)==0) { /* Input is aligned, let's leverage the speed advantage */ + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH64_endian_align(input, len, seed, XXH_littleEndian, XXH_aligned); + else + return XXH64_endian_align(input, len, seed, XXH_bigEndian, XXH_aligned); + } } - if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) - return XXH64_endian_align(input, len, seed, XXH_littleEndian, XXH_unaligned); - else - return XXH64_endian_align(input, len, seed, XXH_bigEndian, XXH_unaligned); + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH64_endian_align(input, len, seed, XXH_littleEndian, XXH_unaligned); + else + return XXH64_endian_align(input, len, seed, XXH_bigEndian, XXH_unaligned); #endif } @@ -537,22 +537,22 @@ XXH_PUBLIC_API unsigned long long XXH64 (const void* input, size_t len, unsigned XXH_PUBLIC_API XXH32_state_t* XXH32_createState(void) { - return (XXH32_state_t*)XXH_malloc(sizeof(XXH32_state_t)); + return (XXH32_state_t*)XXH_malloc(sizeof(XXH32_state_t)); } XXH_PUBLIC_API XXH_errorcode XXH32_freeState(XXH32_state_t* statePtr) { - XXH_free(statePtr); - return XXH_OK; + XXH_free(statePtr); + return XXH_OK; } XXH_PUBLIC_API XXH64_state_t* XXH64_createState(void) { - return (XXH64_state_t*)XXH_malloc(sizeof(XXH64_state_t)); + return (XXH64_state_t*)XXH_malloc(sizeof(XXH64_state_t)); } XXH_PUBLIC_API XXH_errorcode XXH64_freeState(XXH64_state_t* statePtr) { - XXH_free(statePtr); - return XXH_OK; + XXH_free(statePtr); + return XXH_OK; } @@ -560,144 +560,144 @@ XXH_PUBLIC_API XXH_errorcode XXH64_freeState(XXH64_state_t* statePtr) XXH_PUBLIC_API XXH_errorcode XXH32_reset(XXH32_state_t* statePtr, unsigned int seed) { - XXH32_state_t state; /* using a local state to memcpy() in order to avoid strict-aliasing warnings */ - memset(&state, 0, sizeof(state)-4); /* do not write into reserved, for future removal */ - state.v1 = seed + PRIME32_1 + PRIME32_2; - state.v2 = seed + PRIME32_2; - state.v3 = seed + 0; - state.v4 = seed - PRIME32_1; - memcpy(statePtr, &state, sizeof(state)); - return XXH_OK; + XXH32_state_t state; /* using a local state to memcpy() in order to avoid strict-aliasing warnings */ + memset(&state, 0, sizeof(state)-4); /* do not write into reserved, for future removal */ + state.v1 = seed + PRIME32_1 + PRIME32_2; + state.v2 = seed + PRIME32_2; + state.v3 = seed + 0; + state.v4 = seed - PRIME32_1; + memcpy(statePtr, &state, sizeof(state)); + return XXH_OK; } XXH_PUBLIC_API XXH_errorcode XXH64_reset(XXH64_state_t* statePtr, unsigned long long seed) { - XXH64_state_t state; /* using a local state to memcpy() in order to avoid strict-aliasing warnings */ - memset(&state, 0, sizeof(state)-8); /* do not write into reserved, for future removal */ - state.v1 = seed + PRIME64_1 + PRIME64_2; - state.v2 = seed + PRIME64_2; - state.v3 = seed + 0; - state.v4 = seed - PRIME64_1; - memcpy(statePtr, &state, sizeof(state)); - return XXH_OK; + XXH64_state_t state; /* using a local state to memcpy() in order to avoid strict-aliasing warnings */ + memset(&state, 0, sizeof(state)-8); /* do not write into reserved, for future removal */ + state.v1 = seed + PRIME64_1 + PRIME64_2; + state.v2 = seed + PRIME64_2; + state.v3 = seed + 0; + state.v4 = seed - PRIME64_1; + memcpy(statePtr, &state, sizeof(state)); + return XXH_OK; } FORCE_INLINE XXH_errorcode XXH32_update_endian (XXH32_state_t* state, const void* input, size_t len, XXH_endianess endian) { - const BYTE* p = (const BYTE*)input; - const BYTE* const bEnd = p + len; + const BYTE* p = (const BYTE*)input; + const BYTE* const bEnd = p + len; #ifdef XXH_ACCEPT_NULL_INPUT_POINTER - if (input==NULL) return XXH_ERROR; + if (input==NULL) return XXH_ERROR; #endif - state->total_len_32 += (unsigned)len; - state->large_len |= (len>=16) | (state->total_len_32>=16); + state->total_len_32 += (unsigned)len; + state->large_len |= (len>=16) | (state->total_len_32>=16); - if (state->memsize + len < 16) { /* fill in tmp buffer */ - XXH_memcpy((BYTE*)(state->mem32) + state->memsize, input, len); - state->memsize += (unsigned)len; - return XXH_OK; - } + if (state->memsize + len < 16) { /* fill in tmp buffer */ + XXH_memcpy((BYTE*)(state->mem32) + state->memsize, input, len); + state->memsize += (unsigned)len; + return XXH_OK; + } - if (state->memsize) { /* some data left from previous update */ - XXH_memcpy((BYTE*)(state->mem32) + state->memsize, input, 16-state->memsize); - { const U32* p32 = state->mem32; - state->v1 = XXH32_round(state->v1, XXH_readLE32(p32, endian)); p32++; - state->v2 = XXH32_round(state->v2, XXH_readLE32(p32, endian)); p32++; - state->v3 = XXH32_round(state->v3, XXH_readLE32(p32, endian)); p32++; - state->v4 = XXH32_round(state->v4, XXH_readLE32(p32, endian)); p32++; - } - p += 16-state->memsize; - state->memsize = 0; - } + if (state->memsize) { /* some data left from previous update */ + XXH_memcpy((BYTE*)(state->mem32) + state->memsize, input, 16-state->memsize); + { const U32* p32 = state->mem32; + state->v1 = XXH32_round(state->v1, XXH_readLE32(p32, endian)); p32++; + state->v2 = XXH32_round(state->v2, XXH_readLE32(p32, endian)); p32++; + state->v3 = XXH32_round(state->v3, XXH_readLE32(p32, endian)); p32++; + state->v4 = XXH32_round(state->v4, XXH_readLE32(p32, endian)); p32++; + } + p += 16-state->memsize; + state->memsize = 0; + } - if (p <= bEnd-16) { - const BYTE* const limit = bEnd - 16; - U32 v1 = state->v1; - U32 v2 = state->v2; - U32 v3 = state->v3; - U32 v4 = state->v4; + if (p <= bEnd-16) { + const BYTE* const limit = bEnd - 16; + U32 v1 = state->v1; + U32 v2 = state->v2; + U32 v3 = state->v3; + U32 v4 = state->v4; - do { - v1 = XXH32_round(v1, XXH_readLE32(p, endian)); p+=4; - v2 = XXH32_round(v2, XXH_readLE32(p, endian)); p+=4; - v3 = XXH32_round(v3, XXH_readLE32(p, endian)); p+=4; - v4 = XXH32_round(v4, XXH_readLE32(p, endian)); p+=4; - } while (p<=limit); + do { + v1 = XXH32_round(v1, XXH_readLE32(p, endian)); p+=4; + v2 = XXH32_round(v2, XXH_readLE32(p, endian)); p+=4; + v3 = XXH32_round(v3, XXH_readLE32(p, endian)); p+=4; + v4 = XXH32_round(v4, XXH_readLE32(p, endian)); p+=4; + } while (p<=limit); - state->v1 = v1; - state->v2 = v2; - state->v3 = v3; - state->v4 = v4; - } + state->v1 = v1; + state->v2 = v2; + state->v3 = v3; + state->v4 = v4; + } - if (p < bEnd) { - XXH_memcpy(state->mem32, p, (size_t)(bEnd-p)); - state->memsize = (unsigned)(bEnd-p); - } + if (p < bEnd) { + XXH_memcpy(state->mem32, p, (size_t)(bEnd-p)); + state->memsize = (unsigned)(bEnd-p); + } - return XXH_OK; + return XXH_OK; } XXH_PUBLIC_API XXH_errorcode XXH32_update (XXH32_state_t* state_in, const void* input, size_t len) { - XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; - if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) - return XXH32_update_endian(state_in, input, len, XXH_littleEndian); - else - return XXH32_update_endian(state_in, input, len, XXH_bigEndian); + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH32_update_endian(state_in, input, len, XXH_littleEndian); + else + return XXH32_update_endian(state_in, input, len, XXH_bigEndian); } FORCE_INLINE U32 XXH32_digest_endian (const XXH32_state_t* state, XXH_endianess endian) { - const BYTE * p = (const BYTE*)state->mem32; - const BYTE* const bEnd = (const BYTE*)(state->mem32) + state->memsize; - U32 h32; + const BYTE * p = (const BYTE*)state->mem32; + const BYTE* const bEnd = (const BYTE*)(state->mem32) + state->memsize; + U32 h32; - if (state->large_len) { - h32 = XXH_rotl32(state->v1, 1) + XXH_rotl32(state->v2, 7) + XXH_rotl32(state->v3, 12) + XXH_rotl32(state->v4, 18); - } else { - h32 = state->v3 /* == seed */ + PRIME32_5; - } + if (state->large_len) { + h32 = XXH_rotl32(state->v1, 1) + XXH_rotl32(state->v2, 7) + XXH_rotl32(state->v3, 12) + XXH_rotl32(state->v4, 18); + } else { + h32 = state->v3 /* == seed */ + PRIME32_5; + } - h32 += state->total_len_32; + h32 += state->total_len_32; - while (p+4<=bEnd) { - h32 += XXH_readLE32(p, endian) * PRIME32_3; - h32 = XXH_rotl32(h32, 17) * PRIME32_4; - p+=4; - } + while (p+4<=bEnd) { + h32 += XXH_readLE32(p, endian) * PRIME32_3; + h32 = XXH_rotl32(h32, 17) * PRIME32_4; + p+=4; + } - while (p> 15; - h32 *= PRIME32_2; - h32 ^= h32 >> 13; - h32 *= PRIME32_3; - h32 ^= h32 >> 16; + h32 ^= h32 >> 15; + h32 *= PRIME32_2; + h32 ^= h32 >> 13; + h32 *= PRIME32_3; + h32 ^= h32 >> 16; - return h32; + return h32; } XXH_PUBLIC_API unsigned int XXH32_digest (const XXH32_state_t* state_in) { - XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; - if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) - return XXH32_digest_endian(state_in, XXH_littleEndian); - else - return XXH32_digest_endian(state_in, XXH_bigEndian); + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH32_digest_endian(state_in, XXH_littleEndian); + else + return XXH32_digest_endian(state_in, XXH_bigEndian); } @@ -706,131 +706,131 @@ XXH_PUBLIC_API unsigned int XXH32_digest (const XXH32_state_t* state_in) FORCE_INLINE XXH_errorcode XXH64_update_endian (XXH64_state_t* state, const void* input, size_t len, XXH_endianess endian) { - const BYTE* p = (const BYTE*)input; - const BYTE* const bEnd = p + len; + const BYTE* p = (const BYTE*)input; + const BYTE* const bEnd = p + len; #ifdef XXH_ACCEPT_NULL_INPUT_POINTER - if (input==NULL) return XXH_ERROR; + if (input==NULL) return XXH_ERROR; #endif - state->total_len += len; + state->total_len += len; - if (state->memsize + len < 32) { /* fill in tmp buffer */ - XXH_memcpy(((BYTE*)state->mem64) + state->memsize, input, len); - state->memsize += (U32)len; - return XXH_OK; - } + if (state->memsize + len < 32) { /* fill in tmp buffer */ + XXH_memcpy(((BYTE*)state->mem64) + state->memsize, input, len); + state->memsize += (U32)len; + return XXH_OK; + } - if (state->memsize) { /* tmp buffer is full */ - XXH_memcpy(((BYTE*)state->mem64) + state->memsize, input, 32-state->memsize); - state->v1 = XXH64_round(state->v1, XXH_readLE64(state->mem64+0, endian)); - state->v2 = XXH64_round(state->v2, XXH_readLE64(state->mem64+1, endian)); - state->v3 = XXH64_round(state->v3, XXH_readLE64(state->mem64+2, endian)); - state->v4 = XXH64_round(state->v4, XXH_readLE64(state->mem64+3, endian)); - p += 32-state->memsize; - state->memsize = 0; - } + if (state->memsize) { /* tmp buffer is full */ + XXH_memcpy(((BYTE*)state->mem64) + state->memsize, input, 32-state->memsize); + state->v1 = XXH64_round(state->v1, XXH_readLE64(state->mem64+0, endian)); + state->v2 = XXH64_round(state->v2, XXH_readLE64(state->mem64+1, endian)); + state->v3 = XXH64_round(state->v3, XXH_readLE64(state->mem64+2, endian)); + state->v4 = XXH64_round(state->v4, XXH_readLE64(state->mem64+3, endian)); + p += 32-state->memsize; + state->memsize = 0; + } - if (p+32 <= bEnd) { - const BYTE* const limit = bEnd - 32; - U64 v1 = state->v1; - U64 v2 = state->v2; - U64 v3 = state->v3; - U64 v4 = state->v4; + if (p+32 <= bEnd) { + const BYTE* const limit = bEnd - 32; + U64 v1 = state->v1; + U64 v2 = state->v2; + U64 v3 = state->v3; + U64 v4 = state->v4; - do { - v1 = XXH64_round(v1, XXH_readLE64(p, endian)); p+=8; - v2 = XXH64_round(v2, XXH_readLE64(p, endian)); p+=8; - v3 = XXH64_round(v3, XXH_readLE64(p, endian)); p+=8; - v4 = XXH64_round(v4, XXH_readLE64(p, endian)); p+=8; - } while (p<=limit); + do { + v1 = XXH64_round(v1, XXH_readLE64(p, endian)); p+=8; + v2 = XXH64_round(v2, XXH_readLE64(p, endian)); p+=8; + v3 = XXH64_round(v3, XXH_readLE64(p, endian)); p+=8; + v4 = XXH64_round(v4, XXH_readLE64(p, endian)); p+=8; + } while (p<=limit); - state->v1 = v1; - state->v2 = v2; - state->v3 = v3; - state->v4 = v4; - } + state->v1 = v1; + state->v2 = v2; + state->v3 = v3; + state->v4 = v4; + } - if (p < bEnd) { - XXH_memcpy(state->mem64, p, (size_t)(bEnd-p)); - state->memsize = (unsigned)(bEnd-p); - } + if (p < bEnd) { + XXH_memcpy(state->mem64, p, (size_t)(bEnd-p)); + state->memsize = (unsigned)(bEnd-p); + } - return XXH_OK; + return XXH_OK; } XXH_PUBLIC_API XXH_errorcode XXH64_update (XXH64_state_t* state_in, const void* input, size_t len) { - XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; - if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) - return XXH64_update_endian(state_in, input, len, XXH_littleEndian); - else - return XXH64_update_endian(state_in, input, len, XXH_bigEndian); + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH64_update_endian(state_in, input, len, XXH_littleEndian); + else + return XXH64_update_endian(state_in, input, len, XXH_bigEndian); } FORCE_INLINE U64 XXH64_digest_endian (const XXH64_state_t* state, XXH_endianess endian) { - const BYTE * p = (const BYTE*)state->mem64; - const BYTE* const bEnd = (const BYTE*)state->mem64 + state->memsize; - U64 h64; + const BYTE * p = (const BYTE*)state->mem64; + const BYTE* const bEnd = (const BYTE*)state->mem64 + state->memsize; + U64 h64; - if (state->total_len >= 32) { - U64 const v1 = state->v1; - U64 const v2 = state->v2; - U64 const v3 = state->v3; - U64 const v4 = state->v4; + if (state->total_len >= 32) { + U64 const v1 = state->v1; + U64 const v2 = state->v2; + U64 const v3 = state->v3; + U64 const v4 = state->v4; - h64 = XXH_rotl64(v1, 1) + XXH_rotl64(v2, 7) + XXH_rotl64(v3, 12) + XXH_rotl64(v4, 18); - h64 = XXH64_mergeRound(h64, v1); - h64 = XXH64_mergeRound(h64, v2); - h64 = XXH64_mergeRound(h64, v3); - h64 = XXH64_mergeRound(h64, v4); - } else { - h64 = state->v3 + PRIME64_5; - } + h64 = XXH_rotl64(v1, 1) + XXH_rotl64(v2, 7) + XXH_rotl64(v3, 12) + XXH_rotl64(v4, 18); + h64 = XXH64_mergeRound(h64, v1); + h64 = XXH64_mergeRound(h64, v2); + h64 = XXH64_mergeRound(h64, v3); + h64 = XXH64_mergeRound(h64, v4); + } else { + h64 = state->v3 + PRIME64_5; + } - h64 += (U64) state->total_len; + h64 += (U64) state->total_len; - while (p+8<=bEnd) { - U64 const k1 = XXH64_round(0, XXH_readLE64(p, endian)); - h64 ^= k1; - h64 = XXH_rotl64(h64,27) * PRIME64_1 + PRIME64_4; - p+=8; - } + while (p+8<=bEnd) { + U64 const k1 = XXH64_round(0, XXH_readLE64(p, endian)); + h64 ^= k1; + h64 = XXH_rotl64(h64,27) * PRIME64_1 + PRIME64_4; + p+=8; + } - if (p+4<=bEnd) { - h64 ^= (U64)(XXH_readLE32(p, endian)) * PRIME64_1; - h64 = XXH_rotl64(h64, 23) * PRIME64_2 + PRIME64_3; - p+=4; - } + if (p+4<=bEnd) { + h64 ^= (U64)(XXH_readLE32(p, endian)) * PRIME64_1; + h64 = XXH_rotl64(h64, 23) * PRIME64_2 + PRIME64_3; + p+=4; + } - while (p> 33; - h64 *= PRIME64_2; - h64 ^= h64 >> 29; - h64 *= PRIME64_3; - h64 ^= h64 >> 32; + h64 ^= h64 >> 33; + h64 *= PRIME64_2; + h64 ^= h64 >> 29; + h64 *= PRIME64_3; + h64 ^= h64 >> 32; - return h64; + return h64; } XXH_PUBLIC_API unsigned long long XXH64_digest (const XXH64_state_t* state_in) { - XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; + XXH_endianess endian_detected = (XXH_endianess)XXH_CPU_LITTLE_ENDIAN; - if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) - return XXH64_digest_endian(state_in, XXH_littleEndian); - else - return XXH64_digest_endian(state_in, XXH_bigEndian); + if ((endian_detected==XXH_littleEndian) || XXH_FORCE_NATIVE_FORMAT) + return XXH64_digest_endian(state_in, XXH_littleEndian); + else + return XXH64_digest_endian(state_in, XXH_bigEndian); } @@ -846,24 +846,24 @@ XXH_PUBLIC_API unsigned long long XXH64_digest (const XXH64_state_t* state_in) XXH_PUBLIC_API void XXH32_canonicalFromHash(XXH32_canonical_t* dst, XXH32_hash_t hash) { - XXH_STATIC_ASSERT(sizeof(XXH32_canonical_t) == sizeof(XXH32_hash_t)); - if (XXH_CPU_LITTLE_ENDIAN) hash = XXH_swap32(hash); - memcpy(dst, &hash, sizeof(*dst)); + XXH_STATIC_ASSERT(sizeof(XXH32_canonical_t) == sizeof(XXH32_hash_t)); + if (XXH_CPU_LITTLE_ENDIAN) hash = XXH_swap32(hash); + memcpy(dst, &hash, sizeof(*dst)); } XXH_PUBLIC_API void XXH64_canonicalFromHash(XXH64_canonical_t* dst, XXH64_hash_t hash) { - XXH_STATIC_ASSERT(sizeof(XXH64_canonical_t) == sizeof(XXH64_hash_t)); - if (XXH_CPU_LITTLE_ENDIAN) hash = XXH_swap64(hash); - memcpy(dst, &hash, sizeof(*dst)); + XXH_STATIC_ASSERT(sizeof(XXH64_canonical_t) == sizeof(XXH64_hash_t)); + if (XXH_CPU_LITTLE_ENDIAN) hash = XXH_swap64(hash); + memcpy(dst, &hash, sizeof(*dst)); } XXH_PUBLIC_API XXH32_hash_t XXH32_hashFromCanonical(const XXH32_canonical_t* src) { - return XXH_readBE32(src); + return XXH_readBE32(src); } XXH_PUBLIC_API XXH64_hash_t XXH64_hashFromCanonical(const XXH64_canonical_t* src) { - return XXH_readBE64(src); + return XXH_readBE64(src); } diff --git a/contrib/linux-kernel/lib/xxhash.h b/contrib/linux-kernel/lib/xxhash.h index 9bad1f59f..82e24d17d 100644 --- a/contrib/linux-kernel/lib/xxhash.h +++ b/contrib/linux-kernel/lib/xxhash.h @@ -9,9 +9,9 @@ modification, are permitted provided that the following conditions are met: - * Redistributions of source code must retain the above copyright + * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - * Redistributions in binary form must reproduce the above + * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. @@ -165,14 +165,14 @@ XXH_PUBLIC_API XXH64_hash_t XXH64 (const void* input, size_t length, unsigned lo /*! XXH32() : - Calculate the 32-bits hash of sequence "length" bytes stored at memory address "input". - The memory between input & input+length must be valid (allocated and read-accessible). - "seed" can be used to alter the result predictably. - Speed on Core 2 Duo @ 3 GHz (single thread, SMHasher benchmark) : 5.4 GB/s + Calculate the 32-bits hash of sequence "length" bytes stored at memory address "input". + The memory between input & input+length must be valid (allocated and read-accessible). + "seed" can be used to alter the result predictably. + Speed on Core 2 Duo @ 3 GHz (single thread, SMHasher benchmark) : 5.4 GB/s XXH64() : - Calculate the 64-bits hash of sequence of length "len" stored at memory address "input". - "seed" can be used to alter the result predictably. - This function runs 2x faster on 64-bits systems, but slower on 32-bits systems (see benchmark). + Calculate the 64-bits hash of sequence of length "len" stored at memory address "input". + "seed" can be used to alter the result predictably. + This function runs 2x faster on 64-bits systems, but slower on 32-bits systems (see benchmark). */ @@ -270,26 +270,26 @@ XXH_PUBLIC_API XXH64_hash_t XXH64_hashFromCanonical(const XXH64_canonical_t* src Do not use members directly. */ struct XXH32_state_s { - unsigned total_len_32; - unsigned large_len; - unsigned v1; - unsigned v2; - unsigned v3; - unsigned v4; - unsigned mem32[4]; /* buffer defined as U32 for alignment */ - unsigned memsize; - unsigned reserved; /* never read nor write, will be removed in a future version */ + unsigned total_len_32; + unsigned large_len; + unsigned v1; + unsigned v2; + unsigned v3; + unsigned v4; + unsigned mem32[4]; /* buffer defined as U32 for alignment */ + unsigned memsize; + unsigned reserved; /* never read nor write, will be removed in a future version */ }; /* typedef'd to XXH32_state_t */ struct XXH64_state_s { - unsigned long long total_len; - unsigned long long v1; - unsigned long long v2; - unsigned long long v3; - unsigned long long v4; - unsigned long long mem64[4]; /* buffer defined as U64 for alignment */ - unsigned memsize; - unsigned reserved[2]; /* never read nor write, will be removed in a future version */ + unsigned long long total_len; + unsigned long long v1; + unsigned long long v2; + unsigned long long v3; + unsigned long long v4; + unsigned long long mem64[4]; /* buffer defined as U64 for alignment */ + unsigned memsize; + unsigned reserved[2]; /* never read nor write, will be removed in a future version */ }; /* typedef'd to XXH64_state_t */ diff --git a/contrib/linux-kernel/lib/zstd_common.c b/contrib/linux-kernel/lib/zstd_common.c index 8408a589a..69199cb6f 100644 --- a/contrib/linux-kernel/lib/zstd_common.c +++ b/contrib/linux-kernel/lib/zstd_common.c @@ -50,24 +50,24 @@ const char* ZSTD_getErrorString(ZSTD_ErrorCode code) { return ERR_getErrorString /* default uses stdlib */ void* ZSTD_defaultAllocFunction(void* opaque, size_t size) { - void* address = malloc(size); - (void)opaque; - return address; + void* address = malloc(size); + (void)opaque; + return address; } void ZSTD_defaultFreeFunction(void* opaque, void* address) { - (void)opaque; - free(address); + (void)opaque; + free(address); } void* ZSTD_malloc(size_t size, ZSTD_customMem customMem) { - return customMem.customAlloc(customMem.opaque, size); + return customMem.customAlloc(customMem.opaque, size); } void ZSTD_free(void* ptr, ZSTD_customMem customMem) { - if (ptr!=NULL) - customMem.customFree(customMem.opaque, ptr); + if (ptr!=NULL) + customMem.customFree(customMem.opaque, ptr); } diff --git a/contrib/linux-kernel/lib/zstd_compress.c b/contrib/linux-kernel/lib/zstd_compress.c index c31f8db91..84d898e57 100644 --- a/contrib/linux-kernel/lib/zstd_compress.c +++ b/contrib/linux-kernel/lib/zstd_compress.c @@ -40,9 +40,9 @@ size_t ZSTD_compressBound(size_t srcSize) { return FSE_compressBound(srcSize) + ***************************************/ static void ZSTD_resetSeqStore(seqStore_t* ssPtr) { - ssPtr->lit = ssPtr->litStart; - ssPtr->sequences = ssPtr->sequencesStart; - ssPtr->longLengthID = 0; + ssPtr->lit = ssPtr->litStart; + ssPtr->sequences = ssPtr->sequencesStart; + ssPtr->longLengthID = 0; } @@ -50,110 +50,110 @@ static void ZSTD_resetSeqStore(seqStore_t* ssPtr) * Context memory management ***************************************/ struct ZSTD_CCtx_s { - const BYTE* nextSrc; /* next block here to continue on current prefix */ - const BYTE* base; /* All regular indexes relative to this position */ - const BYTE* dictBase; /* extDict indexes relative to this position */ - U32 dictLimit; /* below that point, need extDict */ - U32 lowLimit; /* below that point, no more data */ - U32 nextToUpdate; /* index from which to continue dictionary update */ - U32 nextToUpdate3; /* index from which to continue dictionary update */ - U32 hashLog3; /* dispatch table : larger == faster, more memory */ - U32 loadedDictEnd; /* index of end of dictionary */ - U32 forceWindow; /* force back-references to respect limit of 1<customMem = customMem; - return cctx; + cctx = (ZSTD_CCtx*) ZSTD_malloc(sizeof(ZSTD_CCtx), customMem); + if (!cctx) return NULL; + memset(cctx, 0, sizeof(ZSTD_CCtx)); + cctx->customMem = customMem; + return cctx; } size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx) { - if (cctx==NULL) return 0; /* support free on NULL */ - ZSTD_free(cctx->workSpace, cctx->customMem); - ZSTD_free(cctx, cctx->customMem); - return 0; /* reserved as a potential error code in the future */ + if (cctx==NULL) return 0; /* support free on NULL */ + ZSTD_free(cctx->workSpace, cctx->customMem); + ZSTD_free(cctx, cctx->customMem); + return 0; /* reserved as a potential error code in the future */ } size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx) { - if (cctx==NULL) return 0; /* support sizeof on NULL */ - return sizeof(*cctx) + cctx->workSpaceSize; + if (cctx==NULL) return 0; /* support sizeof on NULL */ + return sizeof(*cctx) + cctx->workSpaceSize; } size_t ZSTD_setCCtxParameter(ZSTD_CCtx* cctx, ZSTD_CCtxParameter param, unsigned value) { - switch(param) - { - case ZSTD_p_forceWindow : cctx->forceWindow = value>0; cctx->loadedDictEnd = 0; return 0; - case ZSTD_p_forceRawDict : cctx->forceRawDict = value>0; return 0; - default: return ERROR(parameter_unknown); - } + switch(param) + { + case ZSTD_p_forceWindow : cctx->forceWindow = value>0; cctx->loadedDictEnd = 0; return 0; + case ZSTD_p_forceRawDict : cctx->forceRawDict = value>0; return 0; + default: return ERROR(parameter_unknown); + } } const seqStore_t* ZSTD_getSeqStore(const ZSTD_CCtx* ctx) /* hidden interface */ { - return &(ctx->seqStore); + return &(ctx->seqStore); } static ZSTD_parameters ZSTD_getParamsFromCCtx(const ZSTD_CCtx* cctx) { - return cctx->params; + return cctx->params; } /** ZSTD_checkParams() : - ensure param values remain within authorized range. - @return : 0, or an error code if one value is beyond authorized range */ + ensure param values remain within authorized range. + @return : 0, or an error code if one value is beyond authorized range */ size_t ZSTD_checkCParams(ZSTD_compressionParameters cParams) { # define CLAMPCHECK(val,min,max) { if ((valmax)) return ERROR(compressionParameter_unsupported); } - CLAMPCHECK(cParams.windowLog, ZSTD_WINDOWLOG_MIN, ZSTD_WINDOWLOG_MAX); - CLAMPCHECK(cParams.chainLog, ZSTD_CHAINLOG_MIN, ZSTD_CHAINLOG_MAX); - CLAMPCHECK(cParams.hashLog, ZSTD_HASHLOG_MIN, ZSTD_HASHLOG_MAX); - CLAMPCHECK(cParams.searchLog, ZSTD_SEARCHLOG_MIN, ZSTD_SEARCHLOG_MAX); - CLAMPCHECK(cParams.searchLength, ZSTD_SEARCHLENGTH_MIN, ZSTD_SEARCHLENGTH_MAX); - CLAMPCHECK(cParams.targetLength, ZSTD_TARGETLENGTH_MIN, ZSTD_TARGETLENGTH_MAX); - if ((U32)(cParams.strategy) > (U32)ZSTD_btopt2) return ERROR(compressionParameter_unsupported); - return 0; + CLAMPCHECK(cParams.windowLog, ZSTD_WINDOWLOG_MIN, ZSTD_WINDOWLOG_MAX); + CLAMPCHECK(cParams.chainLog, ZSTD_CHAINLOG_MIN, ZSTD_CHAINLOG_MAX); + CLAMPCHECK(cParams.hashLog, ZSTD_HASHLOG_MIN, ZSTD_HASHLOG_MAX); + CLAMPCHECK(cParams.searchLog, ZSTD_SEARCHLOG_MIN, ZSTD_SEARCHLOG_MAX); + CLAMPCHECK(cParams.searchLength, ZSTD_SEARCHLENGTH_MIN, ZSTD_SEARCHLENGTH_MAX); + CLAMPCHECK(cParams.targetLength, ZSTD_TARGETLENGTH_MIN, ZSTD_TARGETLENGTH_MAX); + if ((U32)(cParams.strategy) > (U32)ZSTD_btopt2) return ERROR(compressionParameter_unsupported); + return 0; } @@ -161,173 +161,173 @@ size_t ZSTD_checkCParams(ZSTD_compressionParameters cParams) * condition for correct operation : hashLog > 1 */ static U32 ZSTD_cycleLog(U32 hashLog, ZSTD_strategy strat) { - U32 const btScale = ((U32)strat >= (U32)ZSTD_btlazy2); - return hashLog - btScale; + U32 const btScale = ((U32)strat >= (U32)ZSTD_btlazy2); + return hashLog - btScale; } /** ZSTD_adjustCParams() : - optimize `cPar` for a given input (`srcSize` and `dictSize`). - mostly downsizing to reduce memory consumption and initialization. - Both `srcSize` and `dictSize` are optional (use 0 if unknown), - but if both are 0, no optimization can be done. - Note : cPar is considered validated at this stage. Use ZSTD_checkParams() to ensure that. */ + optimize `cPar` for a given input (`srcSize` and `dictSize`). + mostly downsizing to reduce memory consumption and initialization. + Both `srcSize` and `dictSize` are optional (use 0 if unknown), + but if both are 0, no optimization can be done. + Note : cPar is considered validated at this stage. Use ZSTD_checkParams() to ensure that. */ ZSTD_compressionParameters ZSTD_adjustCParams(ZSTD_compressionParameters cPar, unsigned long long srcSize, size_t dictSize) { - if (srcSize+dictSize == 0) return cPar; /* no size information available : no adjustment */ + if (srcSize+dictSize == 0) return cPar; /* no size information available : no adjustment */ - /* resize params, to use less memory when necessary */ - { U32 const minSrcSize = (srcSize==0) ? 500 : 0; - U64 const rSize = srcSize + dictSize + minSrcSize; - if (rSize < ((U64)1< srcLog) cPar.windowLog = srcLog; - } } - if (cPar.hashLog > cPar.windowLog) cPar.hashLog = cPar.windowLog; - { U32 const cycleLog = ZSTD_cycleLog(cPar.chainLog, cPar.strategy); - if (cycleLog > cPar.windowLog) cPar.chainLog -= (cycleLog - cPar.windowLog); - } + /* resize params, to use less memory when necessary */ + { U32 const minSrcSize = (srcSize==0) ? 500 : 0; + U64 const rSize = srcSize + dictSize + minSrcSize; + if (rSize < ((U64)1< srcLog) cPar.windowLog = srcLog; + } } + if (cPar.hashLog > cPar.windowLog) cPar.hashLog = cPar.windowLog; + { U32 const cycleLog = ZSTD_cycleLog(cPar.chainLog, cPar.strategy); + if (cycleLog > cPar.windowLog) cPar.chainLog -= (cycleLog - cPar.windowLog); + } - if (cPar.windowLog < ZSTD_WINDOWLOG_ABSOLUTEMIN) cPar.windowLog = ZSTD_WINDOWLOG_ABSOLUTEMIN; /* required for frame header */ + if (cPar.windowLog < ZSTD_WINDOWLOG_ABSOLUTEMIN) cPar.windowLog = ZSTD_WINDOWLOG_ABSOLUTEMIN; /* required for frame header */ - return cPar; + return cPar; } size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams) { - size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << cParams.windowLog); - U32 const divider = (cParams.searchLength==3) ? 3 : 4; - size_t const maxNbSeq = blockSize / divider; - size_t const tokenSpace = blockSize + 11*maxNbSeq; + size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << cParams.windowLog); + U32 const divider = (cParams.searchLength==3) ? 3 : 4; + size_t const maxNbSeq = blockSize / divider; + size_t const tokenSpace = blockSize + 11*maxNbSeq; - size_t const chainSize = (cParams.strategy == ZSTD_fast) ? 0 : (1 << cParams.chainLog); - size_t const hSize = ((size_t)1) << cParams.hashLog; - U32 const hashLog3 = (cParams.searchLength>3) ? 0 : MIN(ZSTD_HASHLOG3_MAX, cParams.windowLog); - size_t const h3Size = ((size_t)1) << hashLog3; - size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); + size_t const chainSize = (cParams.strategy == ZSTD_fast) ? 0 : (1 << cParams.chainLog); + size_t const hSize = ((size_t)1) << cParams.hashLog; + U32 const hashLog3 = (cParams.searchLength>3) ? 0 : MIN(ZSTD_HASHLOG3_MAX, cParams.windowLog); + size_t const h3Size = ((size_t)1) << hashLog3; + size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); - size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<nextSrc - cctx->base); - cctx->params = params; - cctx->frameContentSize = frameContentSize; - cctx->lowLimit = end; - cctx->dictLimit = end; - cctx->nextToUpdate = end+1; - cctx->stage = ZSTDcs_init; - cctx->dictID = 0; - cctx->loadedDictEnd = 0; - { int i; for (i=0; irep[i] = repStartValue[i]; } - cctx->seqStore.litLengthSum = 0; /* force reset of btopt stats */ - XXH64_reset(&cctx->xxhState, 0); - return 0; + U32 const end = (U32)(cctx->nextSrc - cctx->base); + cctx->params = params; + cctx->frameContentSize = frameContentSize; + cctx->lowLimit = end; + cctx->dictLimit = end; + cctx->nextToUpdate = end+1; + cctx->stage = ZSTDcs_init; + cctx->dictID = 0; + cctx->loadedDictEnd = 0; + { int i; for (i=0; irep[i] = repStartValue[i]; } + cctx->seqStore.litLengthSum = 0; /* force reset of btopt stats */ + XXH64_reset(&cctx->xxhState, 0); + return 0; } typedef enum { ZSTDcrp_continue, ZSTDcrp_noMemset, ZSTDcrp_fullReset } ZSTD_compResetPolicy_e; /*! ZSTD_resetCCtx_advanced() : - note : `params` must be validated */ + note : `params` must be validated */ static size_t ZSTD_resetCCtx_advanced (ZSTD_CCtx* zc, - ZSTD_parameters params, U64 frameContentSize, - ZSTD_compResetPolicy_e const crp) + ZSTD_parameters params, U64 frameContentSize, + ZSTD_compResetPolicy_e const crp) { - if (crp == ZSTDcrp_continue) - if (ZSTD_equivalentParams(params, zc->params)) { - zc->flagStaticTables = 0; - zc->flagStaticHufTable = HUF_repeat_none; - return ZSTD_continueCCtx(zc, params, frameContentSize); - } + if (crp == ZSTDcrp_continue) + if (ZSTD_equivalentParams(params, zc->params)) { + zc->flagStaticTables = 0; + zc->flagStaticHufTable = HUF_repeat_none; + return ZSTD_continueCCtx(zc, params, frameContentSize); + } - { size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << params.cParams.windowLog); - U32 const divider = (params.cParams.searchLength==3) ? 3 : 4; - size_t const maxNbSeq = blockSize / divider; - size_t const tokenSpace = blockSize + 11*maxNbSeq; - size_t const chainSize = (params.cParams.strategy == ZSTD_fast) ? 0 : (1 << params.cParams.chainLog); - size_t const hSize = ((size_t)1) << params.cParams.hashLog; - U32 const hashLog3 = (params.cParams.searchLength>3) ? 0 : MIN(ZSTD_HASHLOG3_MAX, params.cParams.windowLog); - size_t const h3Size = ((size_t)1) << hashLog3; - size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); - void* ptr; + { size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << params.cParams.windowLog); + U32 const divider = (params.cParams.searchLength==3) ? 3 : 4; + size_t const maxNbSeq = blockSize / divider; + size_t const tokenSpace = blockSize + 11*maxNbSeq; + size_t const chainSize = (params.cParams.strategy == ZSTD_fast) ? 0 : (1 << params.cParams.chainLog); + size_t const hSize = ((size_t)1) << params.cParams.hashLog; + U32 const hashLog3 = (params.cParams.searchLength>3) ? 0 : MIN(ZSTD_HASHLOG3_MAX, params.cParams.windowLog); + size_t const h3Size = ((size_t)1) << hashLog3; + size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); + void* ptr; - /* Check if workSpace is large enough, alloc a new one if needed */ - { size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<workSpaceSize < neededSpace) { - ZSTD_free(zc->workSpace, zc->customMem); - zc->workSpace = ZSTD_malloc(neededSpace, zc->customMem); - if (zc->workSpace == NULL) return ERROR(memory_allocation); - zc->workSpaceSize = neededSpace; - } } + /* Check if workSpace is large enough, alloc a new one if needed */ + { size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<workSpaceSize < neededSpace) { + ZSTD_free(zc->workSpace, zc->customMem); + zc->workSpace = ZSTD_malloc(neededSpace, zc->customMem); + if (zc->workSpace == NULL) return ERROR(memory_allocation); + zc->workSpaceSize = neededSpace; + } } - if (crp!=ZSTDcrp_noMemset) memset(zc->workSpace, 0, tableSpace); /* reset tables only */ - XXH64_reset(&zc->xxhState, 0); - zc->hashLog3 = hashLog3; - zc->hashTable = (U32*)(zc->workSpace); - zc->chainTable = zc->hashTable + hSize; - zc->hashTable3 = zc->chainTable + chainSize; - ptr = zc->hashTable3 + h3Size; - zc->hufTable = (HUF_CElt*)ptr; - zc->flagStaticTables = 0; - zc->flagStaticHufTable = HUF_repeat_none; - ptr = ((U32*)ptr) + 256; /* note : HUF_CElt* is incomplete type, size is simulated using U32 */ + if (crp!=ZSTDcrp_noMemset) memset(zc->workSpace, 0, tableSpace); /* reset tables only */ + XXH64_reset(&zc->xxhState, 0); + zc->hashLog3 = hashLog3; + zc->hashTable = (U32*)(zc->workSpace); + zc->chainTable = zc->hashTable + hSize; + zc->hashTable3 = zc->chainTable + chainSize; + ptr = zc->hashTable3 + h3Size; + zc->hufTable = (HUF_CElt*)ptr; + zc->flagStaticTables = 0; + zc->flagStaticHufTable = HUF_repeat_none; + ptr = ((U32*)ptr) + 256; /* note : HUF_CElt* is incomplete type, size is simulated using U32 */ - zc->nextToUpdate = 1; - zc->nextSrc = NULL; - zc->base = NULL; - zc->dictBase = NULL; - zc->dictLimit = 0; - zc->lowLimit = 0; - zc->params = params; - zc->blockSize = blockSize; - zc->frameContentSize = frameContentSize; - { int i; for (i=0; irep[i] = repStartValue[i]; } + zc->nextToUpdate = 1; + zc->nextSrc = NULL; + zc->base = NULL; + zc->dictBase = NULL; + zc->dictLimit = 0; + zc->lowLimit = 0; + zc->params = params; + zc->blockSize = blockSize; + zc->frameContentSize = frameContentSize; + { int i; for (i=0; irep[i] = repStartValue[i]; } - if ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) { - zc->seqStore.litFreq = (U32*)ptr; - zc->seqStore.litLengthFreq = zc->seqStore.litFreq + (1<seqStore.matchLengthFreq = zc->seqStore.litLengthFreq + (MaxLL+1); - zc->seqStore.offCodeFreq = zc->seqStore.matchLengthFreq + (MaxML+1); - ptr = zc->seqStore.offCodeFreq + (MaxOff+1); - zc->seqStore.matchTable = (ZSTD_match_t*)ptr; - ptr = zc->seqStore.matchTable + ZSTD_OPT_NUM+1; - zc->seqStore.priceTable = (ZSTD_optimal_t*)ptr; - ptr = zc->seqStore.priceTable + ZSTD_OPT_NUM+1; - zc->seqStore.litLengthSum = 0; - } - zc->seqStore.sequencesStart = (seqDef*)ptr; - ptr = zc->seqStore.sequencesStart + maxNbSeq; - zc->seqStore.llCode = (BYTE*) ptr; - zc->seqStore.mlCode = zc->seqStore.llCode + maxNbSeq; - zc->seqStore.ofCode = zc->seqStore.mlCode + maxNbSeq; - zc->seqStore.litStart = zc->seqStore.ofCode + maxNbSeq; + if ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) { + zc->seqStore.litFreq = (U32*)ptr; + zc->seqStore.litLengthFreq = zc->seqStore.litFreq + (1<seqStore.matchLengthFreq = zc->seqStore.litLengthFreq + (MaxLL+1); + zc->seqStore.offCodeFreq = zc->seqStore.matchLengthFreq + (MaxML+1); + ptr = zc->seqStore.offCodeFreq + (MaxOff+1); + zc->seqStore.matchTable = (ZSTD_match_t*)ptr; + ptr = zc->seqStore.matchTable + ZSTD_OPT_NUM+1; + zc->seqStore.priceTable = (ZSTD_optimal_t*)ptr; + ptr = zc->seqStore.priceTable + ZSTD_OPT_NUM+1; + zc->seqStore.litLengthSum = 0; + } + zc->seqStore.sequencesStart = (seqDef*)ptr; + ptr = zc->seqStore.sequencesStart + maxNbSeq; + zc->seqStore.llCode = (BYTE*) ptr; + zc->seqStore.mlCode = zc->seqStore.llCode + maxNbSeq; + zc->seqStore.ofCode = zc->seqStore.mlCode + maxNbSeq; + zc->seqStore.litStart = zc->seqStore.ofCode + maxNbSeq; - zc->stage = ZSTDcs_init; - zc->dictID = 0; - zc->loadedDictEnd = 0; + zc->stage = ZSTDcs_init; + zc->dictID = 0; + zc->loadedDictEnd = 0; - return 0; - } + return 0; + } } /* ZSTD_invalidateRepCodes() : @@ -335,8 +335,8 @@ static size_t ZSTD_resetCCtx_advanced (ZSTD_CCtx* zc, * Note : only works with regular variant; * do not use with extDict variant ! */ void ZSTD_invalidateRepCodes(ZSTD_CCtx* cctx) { - int i; - for (i=0; irep[i] = 0; + int i; + for (i=0; irep[i] = 0; } /*! ZSTD_copyCCtx() : @@ -345,47 +345,47 @@ void ZSTD_invalidateRepCodes(ZSTD_CCtx* cctx) { * @return : 0, or an error code */ size_t ZSTD_copyCCtx(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx, unsigned long long pledgedSrcSize) { - if (srcCCtx->stage!=ZSTDcs_init) return ERROR(stage_wrong); + if (srcCCtx->stage!=ZSTDcs_init) return ERROR(stage_wrong); - memcpy(&dstCCtx->customMem, &srcCCtx->customMem, sizeof(ZSTD_customMem)); - { ZSTD_parameters params = srcCCtx->params; - params.fParams.contentSizeFlag = (pledgedSrcSize > 0); - ZSTD_resetCCtx_advanced(dstCCtx, params, pledgedSrcSize, ZSTDcrp_noMemset); - } + memcpy(&dstCCtx->customMem, &srcCCtx->customMem, sizeof(ZSTD_customMem)); + { ZSTD_parameters params = srcCCtx->params; + params.fParams.contentSizeFlag = (pledgedSrcSize > 0); + ZSTD_resetCCtx_advanced(dstCCtx, params, pledgedSrcSize, ZSTDcrp_noMemset); + } - /* copy tables */ - { size_t const chainSize = (srcCCtx->params.cParams.strategy == ZSTD_fast) ? 0 : (1 << srcCCtx->params.cParams.chainLog); - size_t const hSize = ((size_t)1) << srcCCtx->params.cParams.hashLog; - size_t const h3Size = (size_t)1 << srcCCtx->hashLog3; - size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); - memcpy(dstCCtx->workSpace, srcCCtx->workSpace, tableSpace); - } + /* copy tables */ + { size_t const chainSize = (srcCCtx->params.cParams.strategy == ZSTD_fast) ? 0 : (1 << srcCCtx->params.cParams.chainLog); + size_t const hSize = ((size_t)1) << srcCCtx->params.cParams.hashLog; + size_t const h3Size = (size_t)1 << srcCCtx->hashLog3; + size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); + memcpy(dstCCtx->workSpace, srcCCtx->workSpace, tableSpace); + } - /* copy dictionary offsets */ - dstCCtx->nextToUpdate = srcCCtx->nextToUpdate; - dstCCtx->nextToUpdate3= srcCCtx->nextToUpdate3; - dstCCtx->nextSrc = srcCCtx->nextSrc; - dstCCtx->base = srcCCtx->base; - dstCCtx->dictBase = srcCCtx->dictBase; - dstCCtx->dictLimit = srcCCtx->dictLimit; - dstCCtx->lowLimit = srcCCtx->lowLimit; - dstCCtx->loadedDictEnd= srcCCtx->loadedDictEnd; - dstCCtx->dictID = srcCCtx->dictID; + /* copy dictionary offsets */ + dstCCtx->nextToUpdate = srcCCtx->nextToUpdate; + dstCCtx->nextToUpdate3= srcCCtx->nextToUpdate3; + dstCCtx->nextSrc = srcCCtx->nextSrc; + dstCCtx->base = srcCCtx->base; + dstCCtx->dictBase = srcCCtx->dictBase; + dstCCtx->dictLimit = srcCCtx->dictLimit; + dstCCtx->lowLimit = srcCCtx->lowLimit; + dstCCtx->loadedDictEnd= srcCCtx->loadedDictEnd; + dstCCtx->dictID = srcCCtx->dictID; - /* copy entropy tables */ - dstCCtx->flagStaticTables = srcCCtx->flagStaticTables; - dstCCtx->flagStaticHufTable = srcCCtx->flagStaticHufTable; - if (srcCCtx->flagStaticTables) { - memcpy(dstCCtx->litlengthCTable, srcCCtx->litlengthCTable, sizeof(dstCCtx->litlengthCTable)); - memcpy(dstCCtx->matchlengthCTable, srcCCtx->matchlengthCTable, sizeof(dstCCtx->matchlengthCTable)); - memcpy(dstCCtx->offcodeCTable, srcCCtx->offcodeCTable, sizeof(dstCCtx->offcodeCTable)); - } - if (srcCCtx->flagStaticHufTable) { - memcpy(dstCCtx->hufTable, srcCCtx->hufTable, 256*4); - } + /* copy entropy tables */ + dstCCtx->flagStaticTables = srcCCtx->flagStaticTables; + dstCCtx->flagStaticHufTable = srcCCtx->flagStaticHufTable; + if (srcCCtx->flagStaticTables) { + memcpy(dstCCtx->litlengthCTable, srcCCtx->litlengthCTable, sizeof(dstCCtx->litlengthCTable)); + memcpy(dstCCtx->matchlengthCTable, srcCCtx->matchlengthCTable, sizeof(dstCCtx->matchlengthCTable)); + memcpy(dstCCtx->offcodeCTable, srcCCtx->offcodeCTable, sizeof(dstCCtx->offcodeCTable)); + } + if (srcCCtx->flagStaticHufTable) { + memcpy(dstCCtx->hufTable, srcCCtx->hufTable, 256*4); + } - return 0; + return 0; } @@ -393,25 +393,25 @@ size_t ZSTD_copyCCtx(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx, unsigned long * reduce table indexes by `reducerValue` */ static void ZSTD_reduceTable (U32* const table, U32 const size, U32 const reducerValue) { - U32 u; - for (u=0 ; u < size ; u++) { - if (table[u] < reducerValue) table[u] = 0; - else table[u] -= reducerValue; - } + U32 u; + for (u=0 ; u < size ; u++) { + if (table[u] < reducerValue) table[u] = 0; + else table[u] -= reducerValue; + } } /*! ZSTD_reduceIndex() : * rescale all indexes to avoid future overflow (indexes are U32) */ static void ZSTD_reduceIndex (ZSTD_CCtx* zc, const U32 reducerValue) { - { U32 const hSize = 1 << zc->params.cParams.hashLog; - ZSTD_reduceTable(zc->hashTable, hSize, reducerValue); } + { U32 const hSize = 1 << zc->params.cParams.hashLog; + ZSTD_reduceTable(zc->hashTable, hSize, reducerValue); } - { U32 const chainSize = (zc->params.cParams.strategy == ZSTD_fast) ? 0 : (1 << zc->params.cParams.chainLog); - ZSTD_reduceTable(zc->chainTable, chainSize, reducerValue); } + { U32 const chainSize = (zc->params.cParams.strategy == ZSTD_fast) ? 0 : (1 << zc->params.cParams.chainLog); + ZSTD_reduceTable(zc->chainTable, chainSize, reducerValue); } - { U32 const h3Size = (zc->hashLog3) ? 1 << zc->hashLog3 : 0; - ZSTD_reduceTable(zc->hashTable3, h3Size, reducerValue); } + { U32 const h3Size = (zc->hashLog3) ? 1 << zc->hashLog3 : 0; + ZSTD_reduceTable(zc->hashTable3, h3Size, reducerValue); } } @@ -423,376 +423,376 @@ static void ZSTD_reduceIndex (ZSTD_CCtx* zc, const U32 reducerValue) size_t ZSTD_noCompressBlock (void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - if (srcSize + ZSTD_blockHeaderSize > dstCapacity) return ERROR(dstSize_tooSmall); - memcpy((BYTE*)dst + ZSTD_blockHeaderSize, src, srcSize); - MEM_writeLE24(dst, (U32)(srcSize << 2) + (U32)bt_raw); - return ZSTD_blockHeaderSize+srcSize; + if (srcSize + ZSTD_blockHeaderSize > dstCapacity) return ERROR(dstSize_tooSmall); + memcpy((BYTE*)dst + ZSTD_blockHeaderSize, src, srcSize); + MEM_writeLE24(dst, (U32)(srcSize << 2) + (U32)bt_raw); + return ZSTD_blockHeaderSize+srcSize; } static size_t ZSTD_noCompressLiterals (void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - BYTE* const ostart = (BYTE* const)dst; - U32 const flSize = 1 + (srcSize>31) + (srcSize>4095); + BYTE* const ostart = (BYTE* const)dst; + U32 const flSize = 1 + (srcSize>31) + (srcSize>4095); - if (srcSize + flSize > dstCapacity) return ERROR(dstSize_tooSmall); + if (srcSize + flSize > dstCapacity) return ERROR(dstSize_tooSmall); - switch(flSize) - { - case 1: /* 2 - 1 - 5 */ - ostart[0] = (BYTE)((U32)set_basic + (srcSize<<3)); - break; - case 2: /* 2 - 2 - 12 */ - MEM_writeLE16(ostart, (U16)((U32)set_basic + (1<<2) + (srcSize<<4))); - break; - default: /*note : should not be necessary : flSize is within {1,2,3} */ - case 3: /* 2 - 2 - 20 */ - MEM_writeLE32(ostart, (U32)((U32)set_basic + (3<<2) + (srcSize<<4))); - break; - } + switch(flSize) + { + case 1: /* 2 - 1 - 5 */ + ostart[0] = (BYTE)((U32)set_basic + (srcSize<<3)); + break; + case 2: /* 2 - 2 - 12 */ + MEM_writeLE16(ostart, (U16)((U32)set_basic + (1<<2) + (srcSize<<4))); + break; + default: /*note : should not be necessary : flSize is within {1,2,3} */ + case 3: /* 2 - 2 - 20 */ + MEM_writeLE32(ostart, (U32)((U32)set_basic + (3<<2) + (srcSize<<4))); + break; + } - memcpy(ostart + flSize, src, srcSize); - return srcSize + flSize; + memcpy(ostart + flSize, src, srcSize); + return srcSize + flSize; } static size_t ZSTD_compressRleLiteralsBlock (void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - BYTE* const ostart = (BYTE* const)dst; - U32 const flSize = 1 + (srcSize>31) + (srcSize>4095); + BYTE* const ostart = (BYTE* const)dst; + U32 const flSize = 1 + (srcSize>31) + (srcSize>4095); - (void)dstCapacity; /* dstCapacity already guaranteed to be >=4, hence large enough */ + (void)dstCapacity; /* dstCapacity already guaranteed to be >=4, hence large enough */ - switch(flSize) - { - case 1: /* 2 - 1 - 5 */ - ostart[0] = (BYTE)((U32)set_rle + (srcSize<<3)); - break; - case 2: /* 2 - 2 - 12 */ - MEM_writeLE16(ostart, (U16)((U32)set_rle + (1<<2) + (srcSize<<4))); - break; - default: /*note : should not be necessary : flSize is necessarily within {1,2,3} */ - case 3: /* 2 - 2 - 20 */ - MEM_writeLE32(ostart, (U32)((U32)set_rle + (3<<2) + (srcSize<<4))); - break; - } + switch(flSize) + { + case 1: /* 2 - 1 - 5 */ + ostart[0] = (BYTE)((U32)set_rle + (srcSize<<3)); + break; + case 2: /* 2 - 2 - 12 */ + MEM_writeLE16(ostart, (U16)((U32)set_rle + (1<<2) + (srcSize<<4))); + break; + default: /*note : should not be necessary : flSize is necessarily within {1,2,3} */ + case 3: /* 2 - 2 - 20 */ + MEM_writeLE32(ostart, (U32)((U32)set_rle + (3<<2) + (srcSize<<4))); + break; + } - ostart[flSize] = *(const BYTE*)src; - return flSize+1; + ostart[flSize] = *(const BYTE*)src; + return flSize+1; } static size_t ZSTD_minGain(size_t srcSize) { return (srcSize >> 6) + 2; } static size_t ZSTD_compressLiterals (ZSTD_CCtx* zc, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize) { - size_t const minGain = ZSTD_minGain(srcSize); - size_t const lhSize = 3 + (srcSize >= 1 KB) + (srcSize >= 16 KB); - BYTE* const ostart = (BYTE*)dst; - U32 singleStream = srcSize < 256; - symbolEncodingType_e hType = set_compressed; - size_t cLitSize; + size_t const minGain = ZSTD_minGain(srcSize); + size_t const lhSize = 3 + (srcSize >= 1 KB) + (srcSize >= 16 KB); + BYTE* const ostart = (BYTE*)dst; + U32 singleStream = srcSize < 256; + symbolEncodingType_e hType = set_compressed; + size_t cLitSize; - /* small ? don't even attempt compression (speed opt) */ + /* small ? don't even attempt compression (speed opt) */ # define LITERAL_NOENTROPY 63 - { size_t const minLitSize = zc->flagStaticHufTable == HUF_repeat_valid ? 6 : LITERAL_NOENTROPY; - if (srcSize <= minLitSize) return ZSTD_noCompressLiterals(dst, dstCapacity, src, srcSize); - } + { size_t const minLitSize = zc->flagStaticHufTable == HUF_repeat_valid ? 6 : LITERAL_NOENTROPY; + if (srcSize <= minLitSize) return ZSTD_noCompressLiterals(dst, dstCapacity, src, srcSize); + } - if (dstCapacity < lhSize+1) return ERROR(dstSize_tooSmall); /* not enough space for compression */ - { HUF_repeat repeat = zc->flagStaticHufTable; - int const preferRepeat = zc->params.cParams.strategy < ZSTD_lazy ? srcSize <= 1024 : 0; - if (repeat == HUF_repeat_valid && lhSize == 3) singleStream = 1; - cLitSize = singleStream ? HUF_compress1X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11, zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat) - : HUF_compress4X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11, zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat); - if (repeat != HUF_repeat_none) { hType = set_repeat; } /* reused the existing table */ - else { zc->flagStaticHufTable = HUF_repeat_check; } /* now have a table to reuse */ - } + if (dstCapacity < lhSize+1) return ERROR(dstSize_tooSmall); /* not enough space for compression */ + { HUF_repeat repeat = zc->flagStaticHufTable; + int const preferRepeat = zc->params.cParams.strategy < ZSTD_lazy ? srcSize <= 1024 : 0; + if (repeat == HUF_repeat_valid && lhSize == 3) singleStream = 1; + cLitSize = singleStream ? HUF_compress1X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11, zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat) + : HUF_compress4X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11, zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat); + if (repeat != HUF_repeat_none) { hType = set_repeat; } /* reused the existing table */ + else { zc->flagStaticHufTable = HUF_repeat_check; } /* now have a table to reuse */ + } - if ((cLitSize==0) | (cLitSize >= srcSize - minGain)) { - zc->flagStaticHufTable = HUF_repeat_none; - return ZSTD_noCompressLiterals(dst, dstCapacity, src, srcSize); - } - if (cLitSize==1) { - zc->flagStaticHufTable = HUF_repeat_none; - return ZSTD_compressRleLiteralsBlock(dst, dstCapacity, src, srcSize); - } + if ((cLitSize==0) | (cLitSize >= srcSize - minGain)) { + zc->flagStaticHufTable = HUF_repeat_none; + return ZSTD_noCompressLiterals(dst, dstCapacity, src, srcSize); + } + if (cLitSize==1) { + zc->flagStaticHufTable = HUF_repeat_none; + return ZSTD_compressRleLiteralsBlock(dst, dstCapacity, src, srcSize); + } - /* Build header */ - switch(lhSize) - { - case 3: /* 2 - 2 - 10 - 10 */ - { U32 const lhc = hType + ((!singleStream) << 2) + ((U32)srcSize<<4) + ((U32)cLitSize<<14); - MEM_writeLE24(ostart, lhc); - break; - } - case 4: /* 2 - 2 - 14 - 14 */ - { U32 const lhc = hType + (2 << 2) + ((U32)srcSize<<4) + ((U32)cLitSize<<18); - MEM_writeLE32(ostart, lhc); - break; - } - default: /* should not be necessary, lhSize is only {3,4,5} */ - case 5: /* 2 - 2 - 18 - 18 */ - { U32 const lhc = hType + (3 << 2) + ((U32)srcSize<<4) + ((U32)cLitSize<<22); - MEM_writeLE32(ostart, lhc); - ostart[4] = (BYTE)(cLitSize >> 10); - break; - } - } - return lhSize+cLitSize; + /* Build header */ + switch(lhSize) + { + case 3: /* 2 - 2 - 10 - 10 */ + { U32 const lhc = hType + ((!singleStream) << 2) + ((U32)srcSize<<4) + ((U32)cLitSize<<14); + MEM_writeLE24(ostart, lhc); + break; + } + case 4: /* 2 - 2 - 14 - 14 */ + { U32 const lhc = hType + (2 << 2) + ((U32)srcSize<<4) + ((U32)cLitSize<<18); + MEM_writeLE32(ostart, lhc); + break; + } + default: /* should not be necessary, lhSize is only {3,4,5} */ + case 5: /* 2 - 2 - 18 - 18 */ + { U32 const lhc = hType + (3 << 2) + ((U32)srcSize<<4) + ((U32)cLitSize<<22); + MEM_writeLE32(ostart, lhc); + ostart[4] = (BYTE)(cLitSize >> 10); + break; + } + } + return lhSize+cLitSize; } static const BYTE LL_Code[64] = { 0, 1, 2, 3, 4, 5, 6, 7, - 8, 9, 10, 11, 12, 13, 14, 15, - 16, 16, 17, 17, 18, 18, 19, 19, - 20, 20, 20, 20, 21, 21, 21, 21, - 22, 22, 22, 22, 22, 22, 22, 22, - 23, 23, 23, 23, 23, 23, 23, 23, - 24, 24, 24, 24, 24, 24, 24, 24, - 24, 24, 24, 24, 24, 24, 24, 24 }; + 8, 9, 10, 11, 12, 13, 14, 15, + 16, 16, 17, 17, 18, 18, 19, 19, + 20, 20, 20, 20, 21, 21, 21, 21, + 22, 22, 22, 22, 22, 22, 22, 22, + 23, 23, 23, 23, 23, 23, 23, 23, + 24, 24, 24, 24, 24, 24, 24, 24, + 24, 24, 24, 24, 24, 24, 24, 24 }; static const BYTE ML_Code[128] = { 0, 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, 32, 33, 33, 34, 34, 35, 35, 36, 36, 36, 36, 37, 37, 37, 37, - 38, 38, 38, 38, 38, 38, 38, 38, 39, 39, 39, 39, 39, 39, 39, 39, - 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, - 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, - 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, - 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42 }; + 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, + 32, 32, 33, 33, 34, 34, 35, 35, 36, 36, 36, 36, 37, 37, 37, 37, + 38, 38, 38, 38, 38, 38, 38, 38, 39, 39, 39, 39, 39, 39, 39, 39, + 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, 40, + 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, 41, + 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, + 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42, 42 }; void ZSTD_seqToCodes(const seqStore_t* seqStorePtr) { - BYTE const LL_deltaCode = 19; - BYTE const ML_deltaCode = 36; - const seqDef* const sequences = seqStorePtr->sequencesStart; - BYTE* const llCodeTable = seqStorePtr->llCode; - BYTE* const ofCodeTable = seqStorePtr->ofCode; - BYTE* const mlCodeTable = seqStorePtr->mlCode; - U32 const nbSeq = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); - U32 u; - for (u=0; u 63) ? (BYTE)ZSTD_highbit32(llv) + LL_deltaCode : LL_Code[llv]; - ofCodeTable[u] = (BYTE)ZSTD_highbit32(sequences[u].offset); - mlCodeTable[u] = (mlv>127) ? (BYTE)ZSTD_highbit32(mlv) + ML_deltaCode : ML_Code[mlv]; - } - if (seqStorePtr->longLengthID==1) - llCodeTable[seqStorePtr->longLengthPos] = MaxLL; - if (seqStorePtr->longLengthID==2) - mlCodeTable[seqStorePtr->longLengthPos] = MaxML; + BYTE const LL_deltaCode = 19; + BYTE const ML_deltaCode = 36; + const seqDef* const sequences = seqStorePtr->sequencesStart; + BYTE* const llCodeTable = seqStorePtr->llCode; + BYTE* const ofCodeTable = seqStorePtr->ofCode; + BYTE* const mlCodeTable = seqStorePtr->mlCode; + U32 const nbSeq = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); + U32 u; + for (u=0; u 63) ? (BYTE)ZSTD_highbit32(llv) + LL_deltaCode : LL_Code[llv]; + ofCodeTable[u] = (BYTE)ZSTD_highbit32(sequences[u].offset); + mlCodeTable[u] = (mlv>127) ? (BYTE)ZSTD_highbit32(mlv) + ML_deltaCode : ML_Code[mlv]; + } + if (seqStorePtr->longLengthID==1) + llCodeTable[seqStorePtr->longLengthPos] = MaxLL; + if (seqStorePtr->longLengthID==2) + mlCodeTable[seqStorePtr->longLengthPos] = MaxML; } MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc, - void* dst, size_t dstCapacity, - size_t srcSize) + void* dst, size_t dstCapacity, + size_t srcSize) { - const int longOffsets = zc->params.cParams.windowLog > STREAM_ACCUMULATOR_MIN; - const seqStore_t* seqStorePtr = &(zc->seqStore); - U32 count[MaxSeq+1]; - S16 norm[MaxSeq+1]; - FSE_CTable* CTable_LitLength = zc->litlengthCTable; - FSE_CTable* CTable_OffsetBits = zc->offcodeCTable; - FSE_CTable* CTable_MatchLength = zc->matchlengthCTable; - U32 LLtype, Offtype, MLtype; /* compressed, raw or rle */ - const seqDef* const sequences = seqStorePtr->sequencesStart; - const BYTE* const ofCodeTable = seqStorePtr->ofCode; - const BYTE* const llCodeTable = seqStorePtr->llCode; - const BYTE* const mlCodeTable = seqStorePtr->mlCode; - BYTE* const ostart = (BYTE*)dst; - BYTE* const oend = ostart + dstCapacity; - BYTE* op = ostart; - size_t const nbSeq = seqStorePtr->sequences - seqStorePtr->sequencesStart; - BYTE* seqHead; - BYTE scratchBuffer[1<params.cParams.windowLog > STREAM_ACCUMULATOR_MIN; + const seqStore_t* seqStorePtr = &(zc->seqStore); + U32 count[MaxSeq+1]; + S16 norm[MaxSeq+1]; + FSE_CTable* CTable_LitLength = zc->litlengthCTable; + FSE_CTable* CTable_OffsetBits = zc->offcodeCTable; + FSE_CTable* CTable_MatchLength = zc->matchlengthCTable; + U32 LLtype, Offtype, MLtype; /* compressed, raw or rle */ + const seqDef* const sequences = seqStorePtr->sequencesStart; + const BYTE* const ofCodeTable = seqStorePtr->ofCode; + const BYTE* const llCodeTable = seqStorePtr->llCode; + const BYTE* const mlCodeTable = seqStorePtr->mlCode; + BYTE* const ostart = (BYTE*)dst; + BYTE* const oend = ostart + dstCapacity; + BYTE* op = ostart; + size_t const nbSeq = seqStorePtr->sequences - seqStorePtr->sequencesStart; + BYTE* seqHead; + BYTE scratchBuffer[1<litStart; - size_t const litSize = seqStorePtr->lit - literals; - size_t const cSize = ZSTD_compressLiterals(zc, op, dstCapacity, literals, litSize); - if (ZSTD_isError(cSize)) return cSize; - op += cSize; - } + /* Compress literals */ + { const BYTE* const literals = seqStorePtr->litStart; + size_t const litSize = seqStorePtr->lit - literals; + size_t const cSize = ZSTD_compressLiterals(zc, op, dstCapacity, literals, litSize); + if (ZSTD_isError(cSize)) return cSize; + op += cSize; + } - /* Sequences Header */ - if ((oend-op) < 3 /*max nbSeq Size*/ + 1 /*seqHead */) return ERROR(dstSize_tooSmall); - if (nbSeq < 0x7F) *op++ = (BYTE)nbSeq; - else if (nbSeq < LONGNBSEQ) op[0] = (BYTE)((nbSeq>>8) + 0x80), op[1] = (BYTE)nbSeq, op+=2; - else op[0]=0xFF, MEM_writeLE16(op+1, (U16)(nbSeq - LONGNBSEQ)), op+=3; - if (nbSeq==0) goto _check_compressibility; + /* Sequences Header */ + if ((oend-op) < 3 /*max nbSeq Size*/ + 1 /*seqHead */) return ERROR(dstSize_tooSmall); + if (nbSeq < 0x7F) *op++ = (BYTE)nbSeq; + else if (nbSeq < LONGNBSEQ) op[0] = (BYTE)((nbSeq>>8) + 0x80), op[1] = (BYTE)nbSeq, op+=2; + else op[0]=0xFF, MEM_writeLE16(op+1, (U16)(nbSeq - LONGNBSEQ)), op+=3; + if (nbSeq==0) goto _check_compressibility; - /* seqHead : flags for FSE encoding type */ - seqHead = op++; + /* seqHead : flags for FSE encoding type */ + seqHead = op++; #define MIN_SEQ_FOR_DYNAMIC_FSE 64 #define MAX_SEQ_FOR_STATIC_FSE 1000 - /* convert length/distances into codes */ - ZSTD_seqToCodes(seqStorePtr); + /* convert length/distances into codes */ + ZSTD_seqToCodes(seqStorePtr); - /* CTable for Literal Lengths */ - { U32 max = MaxLL; - size_t const mostFrequent = FSE_countFast_wksp(count, &max, llCodeTable, nbSeq, zc->tmpCounters); - if ((mostFrequent == nbSeq) && (nbSeq > 2)) { - *op++ = llCodeTable[0]; - FSE_buildCTable_rle(CTable_LitLength, (BYTE)max); - LLtype = set_rle; - } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) { - LLtype = set_repeat; - } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (LL_defaultNormLog-1)))) { - FSE_buildCTable_wksp(CTable_LitLength, LL_defaultNorm, MaxLL, LL_defaultNormLog, scratchBuffer, sizeof(scratchBuffer)); - LLtype = set_basic; - } else { - size_t nbSeq_1 = nbSeq; - const U32 tableLog = FSE_optimalTableLog(LLFSELog, nbSeq, max); - if (count[llCodeTable[nbSeq-1]]>1) { count[llCodeTable[nbSeq-1]]--; nbSeq_1--; } - FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); - { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ - if (FSE_isError(NCountSize)) return NCountSize; - op += NCountSize; } - FSE_buildCTable_wksp(CTable_LitLength, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); - LLtype = set_compressed; - } } + /* CTable for Literal Lengths */ + { U32 max = MaxLL; + size_t const mostFrequent = FSE_countFast_wksp(count, &max, llCodeTable, nbSeq, zc->tmpCounters); + if ((mostFrequent == nbSeq) && (nbSeq > 2)) { + *op++ = llCodeTable[0]; + FSE_buildCTable_rle(CTable_LitLength, (BYTE)max); + LLtype = set_rle; + } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) { + LLtype = set_repeat; + } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (LL_defaultNormLog-1)))) { + FSE_buildCTable_wksp(CTable_LitLength, LL_defaultNorm, MaxLL, LL_defaultNormLog, scratchBuffer, sizeof(scratchBuffer)); + LLtype = set_basic; + } else { + size_t nbSeq_1 = nbSeq; + const U32 tableLog = FSE_optimalTableLog(LLFSELog, nbSeq, max); + if (count[llCodeTable[nbSeq-1]]>1) { count[llCodeTable[nbSeq-1]]--; nbSeq_1--; } + FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); + { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ + if (FSE_isError(NCountSize)) return NCountSize; + op += NCountSize; } + FSE_buildCTable_wksp(CTable_LitLength, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); + LLtype = set_compressed; + } } - /* CTable for Offsets */ - { U32 max = MaxOff; - size_t const mostFrequent = FSE_countFast_wksp(count, &max, ofCodeTable, nbSeq, zc->tmpCounters); - if ((mostFrequent == nbSeq) && (nbSeq > 2)) { - *op++ = ofCodeTable[0]; - FSE_buildCTable_rle(CTable_OffsetBits, (BYTE)max); - Offtype = set_rle; - } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) { - Offtype = set_repeat; - } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (OF_defaultNormLog-1)))) { - FSE_buildCTable_wksp(CTable_OffsetBits, OF_defaultNorm, MaxOff, OF_defaultNormLog, scratchBuffer, sizeof(scratchBuffer)); - Offtype = set_basic; - } else { - size_t nbSeq_1 = nbSeq; - const U32 tableLog = FSE_optimalTableLog(OffFSELog, nbSeq, max); - if (count[ofCodeTable[nbSeq-1]]>1) { count[ofCodeTable[nbSeq-1]]--; nbSeq_1--; } - FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); - { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ - if (FSE_isError(NCountSize)) return NCountSize; - op += NCountSize; } - FSE_buildCTable_wksp(CTable_OffsetBits, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); - Offtype = set_compressed; - } } + /* CTable for Offsets */ + { U32 max = MaxOff; + size_t const mostFrequent = FSE_countFast_wksp(count, &max, ofCodeTable, nbSeq, zc->tmpCounters); + if ((mostFrequent == nbSeq) && (nbSeq > 2)) { + *op++ = ofCodeTable[0]; + FSE_buildCTable_rle(CTable_OffsetBits, (BYTE)max); + Offtype = set_rle; + } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) { + Offtype = set_repeat; + } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (OF_defaultNormLog-1)))) { + FSE_buildCTable_wksp(CTable_OffsetBits, OF_defaultNorm, MaxOff, OF_defaultNormLog, scratchBuffer, sizeof(scratchBuffer)); + Offtype = set_basic; + } else { + size_t nbSeq_1 = nbSeq; + const U32 tableLog = FSE_optimalTableLog(OffFSELog, nbSeq, max); + if (count[ofCodeTable[nbSeq-1]]>1) { count[ofCodeTable[nbSeq-1]]--; nbSeq_1--; } + FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); + { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ + if (FSE_isError(NCountSize)) return NCountSize; + op += NCountSize; } + FSE_buildCTable_wksp(CTable_OffsetBits, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); + Offtype = set_compressed; + } } - /* CTable for MatchLengths */ - { U32 max = MaxML; - size_t const mostFrequent = FSE_countFast_wksp(count, &max, mlCodeTable, nbSeq, zc->tmpCounters); - if ((mostFrequent == nbSeq) && (nbSeq > 2)) { - *op++ = *mlCodeTable; - FSE_buildCTable_rle(CTable_MatchLength, (BYTE)max); - MLtype = set_rle; - } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) { - MLtype = set_repeat; - } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (ML_defaultNormLog-1)))) { - FSE_buildCTable_wksp(CTable_MatchLength, ML_defaultNorm, MaxML, ML_defaultNormLog, scratchBuffer, sizeof(scratchBuffer)); - MLtype = set_basic; - } else { - size_t nbSeq_1 = nbSeq; - const U32 tableLog = FSE_optimalTableLog(MLFSELog, nbSeq, max); - if (count[mlCodeTable[nbSeq-1]]>1) { count[mlCodeTable[nbSeq-1]]--; nbSeq_1--; } - FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); - { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ - if (FSE_isError(NCountSize)) return NCountSize; - op += NCountSize; } - FSE_buildCTable_wksp(CTable_MatchLength, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); - MLtype = set_compressed; - } } + /* CTable for MatchLengths */ + { U32 max = MaxML; + size_t const mostFrequent = FSE_countFast_wksp(count, &max, mlCodeTable, nbSeq, zc->tmpCounters); + if ((mostFrequent == nbSeq) && (nbSeq > 2)) { + *op++ = *mlCodeTable; + FSE_buildCTable_rle(CTable_MatchLength, (BYTE)max); + MLtype = set_rle; + } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) { + MLtype = set_repeat; + } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (ML_defaultNormLog-1)))) { + FSE_buildCTable_wksp(CTable_MatchLength, ML_defaultNorm, MaxML, ML_defaultNormLog, scratchBuffer, sizeof(scratchBuffer)); + MLtype = set_basic; + } else { + size_t nbSeq_1 = nbSeq; + const U32 tableLog = FSE_optimalTableLog(MLFSELog, nbSeq, max); + if (count[mlCodeTable[nbSeq-1]]>1) { count[mlCodeTable[nbSeq-1]]--; nbSeq_1--; } + FSE_normalizeCount(norm, tableLog, count, nbSeq_1, max); + { size_t const NCountSize = FSE_writeNCount(op, oend-op, norm, max, tableLog); /* overflow protected */ + if (FSE_isError(NCountSize)) return NCountSize; + op += NCountSize; } + FSE_buildCTable_wksp(CTable_MatchLength, norm, max, tableLog, scratchBuffer, sizeof(scratchBuffer)); + MLtype = set_compressed; + } } - *seqHead = (BYTE)((LLtype<<6) + (Offtype<<4) + (MLtype<<2)); - zc->flagStaticTables = 0; + *seqHead = (BYTE)((LLtype<<6) + (Offtype<<4) + (MLtype<<2)); + zc->flagStaticTables = 0; - /* Encoding Sequences */ - { BIT_CStream_t blockStream; - FSE_CState_t stateMatchLength; - FSE_CState_t stateOffsetBits; - FSE_CState_t stateLitLength; + /* Encoding Sequences */ + { BIT_CStream_t blockStream; + FSE_CState_t stateMatchLength; + FSE_CState_t stateOffsetBits; + FSE_CState_t stateLitLength; - CHECK_E(BIT_initCStream(&blockStream, op, oend-op), dstSize_tooSmall); /* not enough space remaining */ + CHECK_E(BIT_initCStream(&blockStream, op, oend-op), dstSize_tooSmall); /* not enough space remaining */ - /* first symbols */ - FSE_initCState2(&stateMatchLength, CTable_MatchLength, mlCodeTable[nbSeq-1]); - FSE_initCState2(&stateOffsetBits, CTable_OffsetBits, ofCodeTable[nbSeq-1]); - FSE_initCState2(&stateLitLength, CTable_LitLength, llCodeTable[nbSeq-1]); - BIT_addBits(&blockStream, sequences[nbSeq-1].litLength, LL_bits[llCodeTable[nbSeq-1]]); - if (MEM_32bits()) BIT_flushBits(&blockStream); - BIT_addBits(&blockStream, sequences[nbSeq-1].matchLength, ML_bits[mlCodeTable[nbSeq-1]]); - if (MEM_32bits()) BIT_flushBits(&blockStream); - if (longOffsets) { - U32 const ofBits = ofCodeTable[nbSeq-1]; - int const extraBits = ofBits - MIN(ofBits, STREAM_ACCUMULATOR_MIN-1); - if (extraBits) { - BIT_addBits(&blockStream, sequences[nbSeq-1].offset, extraBits); - BIT_flushBits(&blockStream); - } - BIT_addBits(&blockStream, sequences[nbSeq-1].offset >> extraBits, - ofBits - extraBits); - } else { - BIT_addBits(&blockStream, sequences[nbSeq-1].offset, ofCodeTable[nbSeq-1]); - } - BIT_flushBits(&blockStream); + /* first symbols */ + FSE_initCState2(&stateMatchLength, CTable_MatchLength, mlCodeTable[nbSeq-1]); + FSE_initCState2(&stateOffsetBits, CTable_OffsetBits, ofCodeTable[nbSeq-1]); + FSE_initCState2(&stateLitLength, CTable_LitLength, llCodeTable[nbSeq-1]); + BIT_addBits(&blockStream, sequences[nbSeq-1].litLength, LL_bits[llCodeTable[nbSeq-1]]); + if (MEM_32bits()) BIT_flushBits(&blockStream); + BIT_addBits(&blockStream, sequences[nbSeq-1].matchLength, ML_bits[mlCodeTable[nbSeq-1]]); + if (MEM_32bits()) BIT_flushBits(&blockStream); + if (longOffsets) { + U32 const ofBits = ofCodeTable[nbSeq-1]; + int const extraBits = ofBits - MIN(ofBits, STREAM_ACCUMULATOR_MIN-1); + if (extraBits) { + BIT_addBits(&blockStream, sequences[nbSeq-1].offset, extraBits); + BIT_flushBits(&blockStream); + } + BIT_addBits(&blockStream, sequences[nbSeq-1].offset >> extraBits, + ofBits - extraBits); + } else { + BIT_addBits(&blockStream, sequences[nbSeq-1].offset, ofCodeTable[nbSeq-1]); + } + BIT_flushBits(&blockStream); - { size_t n; - for (n=nbSeq-2 ; n= 64-7-(LLFSELog+MLFSELog+OffFSELog))) - BIT_flushBits(&blockStream); /* (7)*/ - BIT_addBits(&blockStream, sequences[n].litLength, llBits); - if (MEM_32bits() && ((llBits+mlBits)>24)) BIT_flushBits(&blockStream); - BIT_addBits(&blockStream, sequences[n].matchLength, mlBits); - if (MEM_32bits()) BIT_flushBits(&blockStream); /* (7)*/ - if (longOffsets) { - int const extraBits = ofBits - MIN(ofBits, STREAM_ACCUMULATOR_MIN-1); - if (extraBits) { - BIT_addBits(&blockStream, sequences[n].offset, extraBits); - BIT_flushBits(&blockStream); /* (7)*/ - } - BIT_addBits(&blockStream, sequences[n].offset >> extraBits, - ofBits - extraBits); /* 31 */ - } else { - BIT_addBits(&blockStream, sequences[n].offset, ofBits); /* 31 */ - } - BIT_flushBits(&blockStream); /* (7)*/ - } } + { size_t n; + for (n=nbSeq-2 ; n= 64-7-(LLFSELog+MLFSELog+OffFSELog))) + BIT_flushBits(&blockStream); /* (7)*/ + BIT_addBits(&blockStream, sequences[n].litLength, llBits); + if (MEM_32bits() && ((llBits+mlBits)>24)) BIT_flushBits(&blockStream); + BIT_addBits(&blockStream, sequences[n].matchLength, mlBits); + if (MEM_32bits()) BIT_flushBits(&blockStream); /* (7)*/ + if (longOffsets) { + int const extraBits = ofBits - MIN(ofBits, STREAM_ACCUMULATOR_MIN-1); + if (extraBits) { + BIT_addBits(&blockStream, sequences[n].offset, extraBits); + BIT_flushBits(&blockStream); /* (7)*/ + } + BIT_addBits(&blockStream, sequences[n].offset >> extraBits, + ofBits - extraBits); /* 31 */ + } else { + BIT_addBits(&blockStream, sequences[n].offset, ofBits); /* 31 */ + } + BIT_flushBits(&blockStream); /* (7)*/ + } } - FSE_flushCState(&blockStream, &stateMatchLength); - FSE_flushCState(&blockStream, &stateOffsetBits); - FSE_flushCState(&blockStream, &stateLitLength); + FSE_flushCState(&blockStream, &stateMatchLength); + FSE_flushCState(&blockStream, &stateOffsetBits); + FSE_flushCState(&blockStream, &stateLitLength); - { size_t const streamSize = BIT_closeCStream(&blockStream); - if (streamSize==0) return ERROR(dstSize_tooSmall); /* not enough space */ - op += streamSize; - } } + { size_t const streamSize = BIT_closeCStream(&blockStream); + if (streamSize==0) return ERROR(dstSize_tooSmall); /* not enough space */ + op += streamSize; + } } - /* check compressibility */ + /* check compressibility */ _check_compressibility: - { size_t const minGain = ZSTD_minGain(srcSize); - size_t const maxCSize = srcSize - minGain; - if ((size_t)(op-ostart) >= maxCSize) { - zc->flagStaticHufTable = HUF_repeat_none; - return 0; - } } + { size_t const minGain = ZSTD_minGain(srcSize); + size_t const maxCSize = srcSize - minGain; + if ((size_t)(op-ostart) >= maxCSize) { + zc->flagStaticHufTable = HUF_repeat_none; + return 0; + } } - /* confirm repcodes */ - { int i; for (i=0; irep[i] = zc->repToConfirm[i]; } + /* confirm repcodes */ + { int i; for (i=0; irep[i] = zc->repToConfirm[i]; } - return op - ostart; + return op - ostart; } #if 0 /* for debug */ @@ -803,37 +803,37 @@ const BYTE* g_start = NULL; #endif /*! ZSTD_storeSeq() : - Store a sequence (literal length, literals, offset code and match length code) into seqStore_t. - `offsetCode` : distance to match, or 0 == repCode. - `matchCode` : matchLength - MINMATCH + Store a sequence (literal length, literals, offset code and match length code) into seqStore_t. + `offsetCode` : distance to match, or 0 == repCode. + `matchCode` : matchLength - MINMATCH */ MEM_STATIC void ZSTD_storeSeq(seqStore_t* seqStorePtr, size_t litLength, const void* literals, U32 offsetCode, size_t matchCode) { #ifdef STORESEQ_DEBUG - if (g_startDebug) { - const U32 pos = (U32)((const BYTE*)literals - g_start); - if (g_start==NULL) g_start = (const BYTE*)literals; - if ((pos > 1895000) && (pos < 1895300)) - fprintf(stderr, "Cpos %6u :%5u literals & match %3u bytes at distance %6u \n", - pos, (U32)litLength, (U32)matchCode+MINMATCH, (U32)offsetCode); - } + if (g_startDebug) { + const U32 pos = (U32)((const BYTE*)literals - g_start); + if (g_start==NULL) g_start = (const BYTE*)literals; + if ((pos > 1895000) && (pos < 1895300)) + fprintf(stderr, "Cpos %6u :%5u literals & match %3u bytes at distance %6u \n", + pos, (U32)litLength, (U32)matchCode+MINMATCH, (U32)offsetCode); + } #endif - /* copy Literals */ - ZSTD_wildcopy(seqStorePtr->lit, literals, litLength); - seqStorePtr->lit += litLength; + /* copy Literals */ + ZSTD_wildcopy(seqStorePtr->lit, literals, litLength); + seqStorePtr->lit += litLength; - /* literal Length */ - if (litLength>0xFFFF) { seqStorePtr->longLengthID = 1; seqStorePtr->longLengthPos = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); } - seqStorePtr->sequences[0].litLength = (U16)litLength; + /* literal Length */ + if (litLength>0xFFFF) { seqStorePtr->longLengthID = 1; seqStorePtr->longLengthPos = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); } + seqStorePtr->sequences[0].litLength = (U16)litLength; - /* match offset */ - seqStorePtr->sequences[0].offset = offsetCode + 1; + /* match offset */ + seqStorePtr->sequences[0].offset = offsetCode + 1; - /* match Length */ - if (matchCode>0xFFFF) { seqStorePtr->longLengthID = 2; seqStorePtr->longLengthPos = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); } - seqStorePtr->sequences[0].matchLength = (U16)matchCode; + /* match Length */ + if (matchCode>0xFFFF) { seqStorePtr->longLengthID = 2; seqStorePtr->longLengthPos = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); } + seqStorePtr->sequences[0].matchLength = (U16)matchCode; - seqStorePtr->sequences++; + seqStorePtr->sequences++; } @@ -842,78 +842,78 @@ MEM_STATIC void ZSTD_storeSeq(seqStore_t* seqStorePtr, size_t litLength, const v ***************************************/ static unsigned ZSTD_NbCommonBytes (register size_t val) { - if (MEM_isLittleEndian()) { - if (MEM_64bits()) { + if (MEM_isLittleEndian()) { + if (MEM_64bits()) { # if defined(_MSC_VER) && defined(_WIN64) - unsigned long r = 0; - _BitScanForward64( &r, (U64)val ); - return (unsigned)(r>>3); + unsigned long r = 0; + _BitScanForward64( &r, (U64)val ); + return (unsigned)(r>>3); # elif defined(__GNUC__) && (__GNUC__ >= 3) - return (__builtin_ctzll((U64)val) >> 3); + return (__builtin_ctzll((U64)val) >> 3); # else - static const int DeBruijnBytePos[64] = { 0, 0, 0, 0, 0, 1, 1, 2, 0, 3, 1, 3, 1, 4, 2, 7, 0, 2, 3, 6, 1, 5, 3, 5, 1, 3, 4, 4, 2, 5, 6, 7, 7, 0, 1, 2, 3, 3, 4, 6, 2, 6, 5, 5, 3, 4, 5, 6, 7, 1, 2, 4, 6, 4, 4, 5, 7, 2, 6, 5, 7, 6, 7, 7 }; - return DeBruijnBytePos[((U64)((val & -(long long)val) * 0x0218A392CDABBD3FULL)) >> 58]; + static const int DeBruijnBytePos[64] = { 0, 0, 0, 0, 0, 1, 1, 2, 0, 3, 1, 3, 1, 4, 2, 7, 0, 2, 3, 6, 1, 5, 3, 5, 1, 3, 4, 4, 2, 5, 6, 7, 7, 0, 1, 2, 3, 3, 4, 6, 2, 6, 5, 5, 3, 4, 5, 6, 7, 1, 2, 4, 6, 4, 4, 5, 7, 2, 6, 5, 7, 6, 7, 7 }; + return DeBruijnBytePos[((U64)((val & -(long long)val) * 0x0218A392CDABBD3FULL)) >> 58]; # endif - } else { /* 32 bits */ + } else { /* 32 bits */ # if defined(_MSC_VER) - unsigned long r=0; - _BitScanForward( &r, (U32)val ); - return (unsigned)(r>>3); + unsigned long r=0; + _BitScanForward( &r, (U32)val ); + return (unsigned)(r>>3); # elif defined(__GNUC__) && (__GNUC__ >= 3) - return (__builtin_ctz((U32)val) >> 3); + return (__builtin_ctz((U32)val) >> 3); # else - static const int DeBruijnBytePos[32] = { 0, 0, 3, 0, 3, 1, 3, 0, 3, 2, 2, 1, 3, 2, 0, 1, 3, 3, 1, 2, 2, 2, 2, 0, 3, 1, 2, 0, 1, 0, 1, 1 }; - return DeBruijnBytePos[((U32)((val & -(S32)val) * 0x077CB531U)) >> 27]; + static const int DeBruijnBytePos[32] = { 0, 0, 3, 0, 3, 1, 3, 0, 3, 2, 2, 1, 3, 2, 0, 1, 3, 3, 1, 2, 2, 2, 2, 0, 3, 1, 2, 0, 1, 0, 1, 1 }; + return DeBruijnBytePos[((U32)((val & -(S32)val) * 0x077CB531U)) >> 27]; # endif - } - } else { /* Big Endian CPU */ - if (MEM_64bits()) { + } + } else { /* Big Endian CPU */ + if (MEM_64bits()) { # if defined(_MSC_VER) && defined(_WIN64) - unsigned long r = 0; - _BitScanReverse64( &r, val ); - return (unsigned)(r>>3); + unsigned long r = 0; + _BitScanReverse64( &r, val ); + return (unsigned)(r>>3); # elif defined(__GNUC__) && (__GNUC__ >= 3) - return (__builtin_clzll(val) >> 3); + return (__builtin_clzll(val) >> 3); # else - unsigned r; - const unsigned n32 = sizeof(size_t)*4; /* calculate this way due to compiler complaining in 32-bits mode */ - if (!(val>>n32)) { r=4; } else { r=0; val>>=n32; } - if (!(val>>16)) { r+=2; val>>=8; } else { val>>=24; } - r += (!val); - return r; + unsigned r; + const unsigned n32 = sizeof(size_t)*4; /* calculate this way due to compiler complaining in 32-bits mode */ + if (!(val>>n32)) { r=4; } else { r=0; val>>=n32; } + if (!(val>>16)) { r+=2; val>>=8; } else { val>>=24; } + r += (!val); + return r; # endif - } else { /* 32 bits */ + } else { /* 32 bits */ # if defined(_MSC_VER) - unsigned long r = 0; - _BitScanReverse( &r, (unsigned long)val ); - return (unsigned)(r>>3); + unsigned long r = 0; + _BitScanReverse( &r, (unsigned long)val ); + return (unsigned)(r>>3); # elif defined(__GNUC__) && (__GNUC__ >= 3) - return (__builtin_clz((U32)val) >> 3); + return (__builtin_clz((U32)val) >> 3); # else - unsigned r; - if (!(val>>16)) { r=2; val>>=8; } else { r=0; val>>=24; } - r += (!val); - return r; + unsigned r; + if (!(val>>16)) { r=2; val>>=8; } else { r=0; val>>=24; } + r += (!val); + return r; # endif - } } + } } } static size_t ZSTD_count(const BYTE* pIn, const BYTE* pMatch, const BYTE* const pInLimit) { - const BYTE* const pStart = pIn; - const BYTE* const pInLoopLimit = pInLimit - (sizeof(size_t)-1); + const BYTE* const pStart = pIn; + const BYTE* const pInLoopLimit = pInLimit - (sizeof(size_t)-1); - while (pIn < pInLoopLimit) { - size_t const diff = MEM_readST(pMatch) ^ MEM_readST(pIn); - if (!diff) { pIn+=sizeof(size_t); pMatch+=sizeof(size_t); continue; } - pIn += ZSTD_NbCommonBytes(diff); - return (size_t)(pIn - pStart); - } - if (MEM_64bits()) if ((pIn<(pInLimit-3)) && (MEM_read32(pMatch) == MEM_read32(pIn))) { pIn+=4; pMatch+=4; } - if ((pIn<(pInLimit-1)) && (MEM_read16(pMatch) == MEM_read16(pIn))) { pIn+=2; pMatch+=2; } - if ((pInhashTable; - U32 const hBits = zc->params.cParams.hashLog; - const BYTE* const base = zc->base; - const BYTE* ip = base + zc->nextToUpdate; - const BYTE* const iend = ((const BYTE*)end) - HASH_READ_SIZE; - const size_t fastHashFillStep = 3; + U32* const hashTable = zc->hashTable; + U32 const hBits = zc->params.cParams.hashLog; + const BYTE* const base = zc->base; + const BYTE* ip = base + zc->nextToUpdate; + const BYTE* const iend = ((const BYTE*)end) - HASH_READ_SIZE; + const size_t fastHashFillStep = 3; - while(ip <= iend) { - hashTable[ZSTD_hashPtr(ip, hBits, mls)] = (U32)(ip - base); - ip += fastHashFillStep; - } + while(ip <= iend) { + hashTable[ZSTD_hashPtr(ip, hBits, mls)] = (U32)(ip - base); + ip += fastHashFillStep; + } } FORCE_INLINE void ZSTD_compressBlock_fast_generic(ZSTD_CCtx* cctx, - const void* src, size_t srcSize, - const U32 mls) + const void* src, size_t srcSize, + const U32 mls) { - U32* const hashTable = cctx->hashTable; - U32 const hBits = cctx->params.cParams.hashLog; - seqStore_t* seqStorePtr = &(cctx->seqStore); - const BYTE* const base = cctx->base; - const BYTE* const istart = (const BYTE*)src; - const BYTE* ip = istart; - const BYTE* anchor = istart; - const U32 lowestIndex = cctx->dictLimit; - const BYTE* const lowest = base + lowestIndex; - const BYTE* const iend = istart + srcSize; - const BYTE* const ilimit = iend - HASH_READ_SIZE; - U32 offset_1=cctx->rep[0], offset_2=cctx->rep[1]; - U32 offsetSaved = 0; + U32* const hashTable = cctx->hashTable; + U32 const hBits = cctx->params.cParams.hashLog; + seqStore_t* seqStorePtr = &(cctx->seqStore); + const BYTE* const base = cctx->base; + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const U32 lowestIndex = cctx->dictLimit; + const BYTE* const lowest = base + lowestIndex; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - HASH_READ_SIZE; + U32 offset_1=cctx->rep[0], offset_2=cctx->rep[1]; + U32 offsetSaved = 0; - /* init */ - ip += (ip==lowest); - { U32 const maxRep = (U32)(ip-lowest); - if (offset_2 > maxRep) offsetSaved = offset_2, offset_2 = 0; - if (offset_1 > maxRep) offsetSaved = offset_1, offset_1 = 0; - } + /* init */ + ip += (ip==lowest); + { U32 const maxRep = (U32)(ip-lowest); + if (offset_2 > maxRep) offsetSaved = offset_2, offset_2 = 0; + if (offset_1 > maxRep) offsetSaved = offset_1, offset_1 = 0; + } - /* Main Search Loop */ - while (ip < ilimit) { /* < instead of <=, because repcode check at (ip+1) */ - size_t mLength; - size_t const h = ZSTD_hashPtr(ip, hBits, mls); - U32 const current = (U32)(ip-base); - U32 const matchIndex = hashTable[h]; - const BYTE* match = base + matchIndex; - hashTable[h] = current; /* update hash table */ + /* Main Search Loop */ + while (ip < ilimit) { /* < instead of <=, because repcode check at (ip+1) */ + size_t mLength; + size_t const h = ZSTD_hashPtr(ip, hBits, mls); + U32 const current = (U32)(ip-base); + U32 const matchIndex = hashTable[h]; + const BYTE* match = base + matchIndex; + hashTable[h] = current; /* update hash table */ - if ((offset_1 > 0) & (MEM_read32(ip+1-offset_1) == MEM_read32(ip+1))) { - mLength = ZSTD_count(ip+1+4, ip+1+4-offset_1, iend) + 4; - ip++; - ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); - } else { - U32 offset; - if ( (matchIndex <= lowestIndex) || (MEM_read32(match) != MEM_read32(ip)) ) { - ip += ((ip-anchor) >> g_searchStrength) + 1; - continue; - } - mLength = ZSTD_count(ip+4, match+4, iend) + 4; - offset = (U32)(ip-match); - while (((ip>anchor) & (match>lowest)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ - offset_2 = offset_1; - offset_1 = offset; + if ((offset_1 > 0) & (MEM_read32(ip+1-offset_1) == MEM_read32(ip+1))) { + mLength = ZSTD_count(ip+1+4, ip+1+4-offset_1, iend) + 4; + ip++; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); + } else { + U32 offset; + if ( (matchIndex <= lowestIndex) || (MEM_read32(match) != MEM_read32(ip)) ) { + ip += ((ip-anchor) >> g_searchStrength) + 1; + continue; + } + mLength = ZSTD_count(ip+4, match+4, iend) + 4; + offset = (U32)(ip-match); + while (((ip>anchor) & (match>lowest)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ + offset_2 = offset_1; + offset_1 = offset; - ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); - } + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); + } - /* match found */ - ip += mLength; - anchor = ip; + /* match found */ + ip += mLength; + anchor = ip; - if (ip <= ilimit) { - /* Fill Table */ - hashTable[ZSTD_hashPtr(base+current+2, hBits, mls)] = current+2; /* here because current+2 could be > iend-8 */ - hashTable[ZSTD_hashPtr(ip-2, hBits, mls)] = (U32)(ip-2-base); - /* check immediate repcode */ - while ( (ip <= ilimit) - && ( (offset_2>0) - & (MEM_read32(ip) == MEM_read32(ip - offset_2)) )) { - /* store sequence */ - size_t const rLength = ZSTD_count(ip+4, ip+4-offset_2, iend) + 4; - { U32 const tmpOff = offset_2; offset_2 = offset_1; offset_1 = tmpOff; } /* swap offset_2 <=> offset_1 */ - hashTable[ZSTD_hashPtr(ip, hBits, mls)] = (U32)(ip-base); - ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, rLength-MINMATCH); - ip += rLength; - anchor = ip; - continue; /* faster when present ... (?) */ - } } } + if (ip <= ilimit) { + /* Fill Table */ + hashTable[ZSTD_hashPtr(base+current+2, hBits, mls)] = current+2; /* here because current+2 could be > iend-8 */ + hashTable[ZSTD_hashPtr(ip-2, hBits, mls)] = (U32)(ip-2-base); + /* check immediate repcode */ + while ( (ip <= ilimit) + && ( (offset_2>0) + & (MEM_read32(ip) == MEM_read32(ip - offset_2)) )) { + /* store sequence */ + size_t const rLength = ZSTD_count(ip+4, ip+4-offset_2, iend) + 4; + { U32 const tmpOff = offset_2; offset_2 = offset_1; offset_1 = tmpOff; } /* swap offset_2 <=> offset_1 */ + hashTable[ZSTD_hashPtr(ip, hBits, mls)] = (U32)(ip-base); + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, rLength-MINMATCH); + ip += rLength; + anchor = ip; + continue; /* faster when present ... (?) */ + } } } - /* save reps for next block */ - cctx->repToConfirm[0] = offset_1 ? offset_1 : offsetSaved; - cctx->repToConfirm[1] = offset_2 ? offset_2 : offsetSaved; + /* save reps for next block */ + cctx->repToConfirm[0] = offset_1 ? offset_1 : offsetSaved; + cctx->repToConfirm[1] = offset_2 ? offset_2 : offsetSaved; - /* Last Literals */ - { size_t const lastLLSize = iend - anchor; - memcpy(seqStorePtr->lit, anchor, lastLLSize); - seqStorePtr->lit += lastLLSize; - } + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } } static void ZSTD_compressBlock_fast(ZSTD_CCtx* ctx, - const void* src, size_t srcSize) + const void* src, size_t srcSize) { - const U32 mls = ctx->params.cParams.searchLength; - switch(mls) - { - default: /* includes case 3 */ - case 4 : - ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 4); return; - case 5 : - ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 5); return; - case 6 : - ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 6); return; - case 7 : - ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 7); return; - } + const U32 mls = ctx->params.cParams.searchLength; + switch(mls) + { + default: /* includes case 3 */ + case 4 : + ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 4); return; + case 5 : + ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 5); return; + case 6 : + ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 6); return; + case 7 : + ZSTD_compressBlock_fast_generic(ctx, src, srcSize, 7); return; + } } static void ZSTD_compressBlock_fast_extDict_generic(ZSTD_CCtx* ctx, - const void* src, size_t srcSize, - const U32 mls) + const void* src, size_t srcSize, + const U32 mls) { - U32* hashTable = ctx->hashTable; - const U32 hBits = ctx->params.cParams.hashLog; - seqStore_t* seqStorePtr = &(ctx->seqStore); - const BYTE* const base = ctx->base; - const BYTE* const dictBase = ctx->dictBase; - const BYTE* const istart = (const BYTE*)src; - const BYTE* ip = istart; - const BYTE* anchor = istart; - const U32 lowestIndex = ctx->lowLimit; - const BYTE* const dictStart = dictBase + lowestIndex; - const U32 dictLimit = ctx->dictLimit; - const BYTE* const lowPrefixPtr = base + dictLimit; - const BYTE* const dictEnd = dictBase + dictLimit; - const BYTE* const iend = istart + srcSize; - const BYTE* const ilimit = iend - 8; - U32 offset_1=ctx->rep[0], offset_2=ctx->rep[1]; + U32* hashTable = ctx->hashTable; + const U32 hBits = ctx->params.cParams.hashLog; + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const base = ctx->base; + const BYTE* const dictBase = ctx->dictBase; + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const U32 lowestIndex = ctx->lowLimit; + const BYTE* const dictStart = dictBase + lowestIndex; + const U32 dictLimit = ctx->dictLimit; + const BYTE* const lowPrefixPtr = base + dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + U32 offset_1=ctx->rep[0], offset_2=ctx->rep[1]; - /* Search Loop */ - while (ip < ilimit) { /* < instead of <=, because (ip+1) */ - const size_t h = ZSTD_hashPtr(ip, hBits, mls); - const U32 matchIndex = hashTable[h]; - const BYTE* matchBase = matchIndex < dictLimit ? dictBase : base; - const BYTE* match = matchBase + matchIndex; - const U32 current = (U32)(ip-base); - const U32 repIndex = current + 1 - offset_1; /* offset_1 expected <= current +1 */ - const BYTE* repBase = repIndex < dictLimit ? dictBase : base; - const BYTE* repMatch = repBase + repIndex; - size_t mLength; - hashTable[h] = current; /* update hash table */ + /* Search Loop */ + while (ip < ilimit) { /* < instead of <=, because (ip+1) */ + const size_t h = ZSTD_hashPtr(ip, hBits, mls); + const U32 matchIndex = hashTable[h]; + const BYTE* matchBase = matchIndex < dictLimit ? dictBase : base; + const BYTE* match = matchBase + matchIndex; + const U32 current = (U32)(ip-base); + const U32 repIndex = current + 1 - offset_1; /* offset_1 expected <= current +1 */ + const BYTE* repBase = repIndex < dictLimit ? dictBase : base; + const BYTE* repMatch = repBase + repIndex; + size_t mLength; + hashTable[h] = current; /* update hash table */ - if ( (((U32)((dictLimit-1) - repIndex) >= 3) /* intentional underflow */ & (repIndex > lowestIndex)) - && (MEM_read32(repMatch) == MEM_read32(ip+1)) ) { - const BYTE* repMatchEnd = repIndex < dictLimit ? dictEnd : iend; - mLength = ZSTD_count_2segments(ip+1+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repMatchEnd, lowPrefixPtr) + EQUAL_READ32; - ip++; - ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); - } else { - if ( (matchIndex < lowestIndex) || - (MEM_read32(match) != MEM_read32(ip)) ) { - ip += ((ip-anchor) >> g_searchStrength) + 1; - continue; - } - { const BYTE* matchEnd = matchIndex < dictLimit ? dictEnd : iend; - const BYTE* lowMatchPtr = matchIndex < dictLimit ? dictStart : lowPrefixPtr; - U32 offset; - mLength = ZSTD_count_2segments(ip+EQUAL_READ32, match+EQUAL_READ32, iend, matchEnd, lowPrefixPtr) + EQUAL_READ32; - while (((ip>anchor) & (match>lowMatchPtr)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ - offset = current - matchIndex; - offset_2 = offset_1; - offset_1 = offset; - ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); - } } + if ( (((U32)((dictLimit-1) - repIndex) >= 3) /* intentional underflow */ & (repIndex > lowestIndex)) + && (MEM_read32(repMatch) == MEM_read32(ip+1)) ) { + const BYTE* repMatchEnd = repIndex < dictLimit ? dictEnd : iend; + mLength = ZSTD_count_2segments(ip+1+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repMatchEnd, lowPrefixPtr) + EQUAL_READ32; + ip++; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); + } else { + if ( (matchIndex < lowestIndex) || + (MEM_read32(match) != MEM_read32(ip)) ) { + ip += ((ip-anchor) >> g_searchStrength) + 1; + continue; + } + { const BYTE* matchEnd = matchIndex < dictLimit ? dictEnd : iend; + const BYTE* lowMatchPtr = matchIndex < dictLimit ? dictStart : lowPrefixPtr; + U32 offset; + mLength = ZSTD_count_2segments(ip+EQUAL_READ32, match+EQUAL_READ32, iend, matchEnd, lowPrefixPtr) + EQUAL_READ32; + while (((ip>anchor) & (match>lowMatchPtr)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ + offset = current - matchIndex; + offset_2 = offset_1; + offset_1 = offset; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); + } } - /* found a match : store it */ - ip += mLength; - anchor = ip; + /* found a match : store it */ + ip += mLength; + anchor = ip; - if (ip <= ilimit) { - /* Fill Table */ - hashTable[ZSTD_hashPtr(base+current+2, hBits, mls)] = current+2; - hashTable[ZSTD_hashPtr(ip-2, hBits, mls)] = (U32)(ip-2-base); - /* check immediate repcode */ - while (ip <= ilimit) { - U32 const current2 = (U32)(ip-base); - U32 const repIndex2 = current2 - offset_2; - const BYTE* repMatch2 = repIndex2 < dictLimit ? dictBase + repIndex2 : base + repIndex2; - if ( (((U32)((dictLimit-1) - repIndex2) >= 3) & (repIndex2 > lowestIndex)) /* intentional overflow */ - && (MEM_read32(repMatch2) == MEM_read32(ip)) ) { - const BYTE* const repEnd2 = repIndex2 < dictLimit ? dictEnd : iend; - size_t repLength2 = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch2+EQUAL_READ32, iend, repEnd2, lowPrefixPtr) + EQUAL_READ32; - U32 tmpOffset = offset_2; offset_2 = offset_1; offset_1 = tmpOffset; /* swap offset_2 <=> offset_1 */ - ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, repLength2-MINMATCH); - hashTable[ZSTD_hashPtr(ip, hBits, mls)] = current2; - ip += repLength2; - anchor = ip; - continue; - } - break; - } } } + if (ip <= ilimit) { + /* Fill Table */ + hashTable[ZSTD_hashPtr(base+current+2, hBits, mls)] = current+2; + hashTable[ZSTD_hashPtr(ip-2, hBits, mls)] = (U32)(ip-2-base); + /* check immediate repcode */ + while (ip <= ilimit) { + U32 const current2 = (U32)(ip-base); + U32 const repIndex2 = current2 - offset_2; + const BYTE* repMatch2 = repIndex2 < dictLimit ? dictBase + repIndex2 : base + repIndex2; + if ( (((U32)((dictLimit-1) - repIndex2) >= 3) & (repIndex2 > lowestIndex)) /* intentional overflow */ + && (MEM_read32(repMatch2) == MEM_read32(ip)) ) { + const BYTE* const repEnd2 = repIndex2 < dictLimit ? dictEnd : iend; + size_t repLength2 = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch2+EQUAL_READ32, iend, repEnd2, lowPrefixPtr) + EQUAL_READ32; + U32 tmpOffset = offset_2; offset_2 = offset_1; offset_1 = tmpOffset; /* swap offset_2 <=> offset_1 */ + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, repLength2-MINMATCH); + hashTable[ZSTD_hashPtr(ip, hBits, mls)] = current2; + ip += repLength2; + anchor = ip; + continue; + } + break; + } } } - /* save reps for next block */ - ctx->repToConfirm[0] = offset_1; ctx->repToConfirm[1] = offset_2; + /* save reps for next block */ + ctx->repToConfirm[0] = offset_1; ctx->repToConfirm[1] = offset_2; - /* Last Literals */ - { size_t const lastLLSize = iend - anchor; - memcpy(seqStorePtr->lit, anchor, lastLLSize); - seqStorePtr->lit += lastLLSize; - } + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } } static void ZSTD_compressBlock_fast_extDict(ZSTD_CCtx* ctx, - const void* src, size_t srcSize) + const void* src, size_t srcSize) { - U32 const mls = ctx->params.cParams.searchLength; - switch(mls) - { - default: /* includes case 3 */ - case 4 : - ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 4); return; - case 5 : - ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 5); return; - case 6 : - ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 6); return; - case 7 : - ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 7); return; - } + U32 const mls = ctx->params.cParams.searchLength; + switch(mls) + { + default: /* includes case 3 */ + case 4 : + ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 4); return; + case 5 : + ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 5); return; + case 6 : + ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 6); return; + case 7 : + ZSTD_compressBlock_fast_extDict_generic(ctx, src, srcSize, 7); return; + } } @@ -1216,302 +1216,302 @@ static void ZSTD_compressBlock_fast_extDict(ZSTD_CCtx* ctx, ***************************************/ static void ZSTD_fillDoubleHashTable (ZSTD_CCtx* cctx, const void* end, const U32 mls) { - U32* const hashLarge = cctx->hashTable; - U32 const hBitsL = cctx->params.cParams.hashLog; - U32* const hashSmall = cctx->chainTable; - U32 const hBitsS = cctx->params.cParams.chainLog; - const BYTE* const base = cctx->base; - const BYTE* ip = base + cctx->nextToUpdate; - const BYTE* const iend = ((const BYTE*)end) - HASH_READ_SIZE; - const size_t fastHashFillStep = 3; + U32* const hashLarge = cctx->hashTable; + U32 const hBitsL = cctx->params.cParams.hashLog; + U32* const hashSmall = cctx->chainTable; + U32 const hBitsS = cctx->params.cParams.chainLog; + const BYTE* const base = cctx->base; + const BYTE* ip = base + cctx->nextToUpdate; + const BYTE* const iend = ((const BYTE*)end) - HASH_READ_SIZE; + const size_t fastHashFillStep = 3; - while(ip <= iend) { - hashSmall[ZSTD_hashPtr(ip, hBitsS, mls)] = (U32)(ip - base); - hashLarge[ZSTD_hashPtr(ip, hBitsL, 8)] = (U32)(ip - base); - ip += fastHashFillStep; - } + while(ip <= iend) { + hashSmall[ZSTD_hashPtr(ip, hBitsS, mls)] = (U32)(ip - base); + hashLarge[ZSTD_hashPtr(ip, hBitsL, 8)] = (U32)(ip - base); + ip += fastHashFillStep; + } } FORCE_INLINE void ZSTD_compressBlock_doubleFast_generic(ZSTD_CCtx* cctx, - const void* src, size_t srcSize, - const U32 mls) + const void* src, size_t srcSize, + const U32 mls) { - U32* const hashLong = cctx->hashTable; - const U32 hBitsL = cctx->params.cParams.hashLog; - U32* const hashSmall = cctx->chainTable; - const U32 hBitsS = cctx->params.cParams.chainLog; - seqStore_t* seqStorePtr = &(cctx->seqStore); - const BYTE* const base = cctx->base; - const BYTE* const istart = (const BYTE*)src; - const BYTE* ip = istart; - const BYTE* anchor = istart; - const U32 lowestIndex = cctx->dictLimit; - const BYTE* const lowest = base + lowestIndex; - const BYTE* const iend = istart + srcSize; - const BYTE* const ilimit = iend - HASH_READ_SIZE; - U32 offset_1=cctx->rep[0], offset_2=cctx->rep[1]; - U32 offsetSaved = 0; + U32* const hashLong = cctx->hashTable; + const U32 hBitsL = cctx->params.cParams.hashLog; + U32* const hashSmall = cctx->chainTable; + const U32 hBitsS = cctx->params.cParams.chainLog; + seqStore_t* seqStorePtr = &(cctx->seqStore); + const BYTE* const base = cctx->base; + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const U32 lowestIndex = cctx->dictLimit; + const BYTE* const lowest = base + lowestIndex; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - HASH_READ_SIZE; + U32 offset_1=cctx->rep[0], offset_2=cctx->rep[1]; + U32 offsetSaved = 0; - /* init */ - ip += (ip==lowest); - { U32 const maxRep = (U32)(ip-lowest); - if (offset_2 > maxRep) offsetSaved = offset_2, offset_2 = 0; - if (offset_1 > maxRep) offsetSaved = offset_1, offset_1 = 0; - } + /* init */ + ip += (ip==lowest); + { U32 const maxRep = (U32)(ip-lowest); + if (offset_2 > maxRep) offsetSaved = offset_2, offset_2 = 0; + if (offset_1 > maxRep) offsetSaved = offset_1, offset_1 = 0; + } - /* Main Search Loop */ - while (ip < ilimit) { /* < instead of <=, because repcode check at (ip+1) */ - size_t mLength; - size_t const h2 = ZSTD_hashPtr(ip, hBitsL, 8); - size_t const h = ZSTD_hashPtr(ip, hBitsS, mls); - U32 const current = (U32)(ip-base); - U32 const matchIndexL = hashLong[h2]; - U32 const matchIndexS = hashSmall[h]; - const BYTE* matchLong = base + matchIndexL; - const BYTE* match = base + matchIndexS; - hashLong[h2] = hashSmall[h] = current; /* update hash tables */ + /* Main Search Loop */ + while (ip < ilimit) { /* < instead of <=, because repcode check at (ip+1) */ + size_t mLength; + size_t const h2 = ZSTD_hashPtr(ip, hBitsL, 8); + size_t const h = ZSTD_hashPtr(ip, hBitsS, mls); + U32 const current = (U32)(ip-base); + U32 const matchIndexL = hashLong[h2]; + U32 const matchIndexS = hashSmall[h]; + const BYTE* matchLong = base + matchIndexL; + const BYTE* match = base + matchIndexS; + hashLong[h2] = hashSmall[h] = current; /* update hash tables */ - if ((offset_1 > 0) & (MEM_read32(ip+1-offset_1) == MEM_read32(ip+1))) { /* note : by construction, offset_1 <= current */ - mLength = ZSTD_count(ip+1+4, ip+1+4-offset_1, iend) + 4; - ip++; - ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); - } else { - U32 offset; - if ( (matchIndexL > lowestIndex) && (MEM_read64(matchLong) == MEM_read64(ip)) ) { - mLength = ZSTD_count(ip+8, matchLong+8, iend) + 8; - offset = (U32)(ip-matchLong); - while (((ip>anchor) & (matchLong>lowest)) && (ip[-1] == matchLong[-1])) { ip--; matchLong--; mLength++; } /* catch up */ - } else if ( (matchIndexS > lowestIndex) && (MEM_read32(match) == MEM_read32(ip)) ) { - size_t const h3 = ZSTD_hashPtr(ip+1, hBitsL, 8); - U32 const matchIndex3 = hashLong[h3]; - const BYTE* match3 = base + matchIndex3; - hashLong[h3] = current + 1; - if ( (matchIndex3 > lowestIndex) && (MEM_read64(match3) == MEM_read64(ip+1)) ) { - mLength = ZSTD_count(ip+9, match3+8, iend) + 8; - ip++; - offset = (U32)(ip-match3); - while (((ip>anchor) & (match3>lowest)) && (ip[-1] == match3[-1])) { ip--; match3--; mLength++; } /* catch up */ - } else { - mLength = ZSTD_count(ip+4, match+4, iend) + 4; - offset = (U32)(ip-match); - while (((ip>anchor) & (match>lowest)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ - } - } else { - ip += ((ip-anchor) >> g_searchStrength) + 1; - continue; - } + if ((offset_1 > 0) & (MEM_read32(ip+1-offset_1) == MEM_read32(ip+1))) { /* note : by construction, offset_1 <= current */ + mLength = ZSTD_count(ip+1+4, ip+1+4-offset_1, iend) + 4; + ip++; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); + } else { + U32 offset; + if ( (matchIndexL > lowestIndex) && (MEM_read64(matchLong) == MEM_read64(ip)) ) { + mLength = ZSTD_count(ip+8, matchLong+8, iend) + 8; + offset = (U32)(ip-matchLong); + while (((ip>anchor) & (matchLong>lowest)) && (ip[-1] == matchLong[-1])) { ip--; matchLong--; mLength++; } /* catch up */ + } else if ( (matchIndexS > lowestIndex) && (MEM_read32(match) == MEM_read32(ip)) ) { + size_t const h3 = ZSTD_hashPtr(ip+1, hBitsL, 8); + U32 const matchIndex3 = hashLong[h3]; + const BYTE* match3 = base + matchIndex3; + hashLong[h3] = current + 1; + if ( (matchIndex3 > lowestIndex) && (MEM_read64(match3) == MEM_read64(ip+1)) ) { + mLength = ZSTD_count(ip+9, match3+8, iend) + 8; + ip++; + offset = (U32)(ip-match3); + while (((ip>anchor) & (match3>lowest)) && (ip[-1] == match3[-1])) { ip--; match3--; mLength++; } /* catch up */ + } else { + mLength = ZSTD_count(ip+4, match+4, iend) + 4; + offset = (U32)(ip-match); + while (((ip>anchor) & (match>lowest)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ + } + } else { + ip += ((ip-anchor) >> g_searchStrength) + 1; + continue; + } - offset_2 = offset_1; - offset_1 = offset; + offset_2 = offset_1; + offset_1 = offset; - ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); - } + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); + } - /* match found */ - ip += mLength; - anchor = ip; + /* match found */ + ip += mLength; + anchor = ip; - if (ip <= ilimit) { - /* Fill Table */ - hashLong[ZSTD_hashPtr(base+current+2, hBitsL, 8)] = - hashSmall[ZSTD_hashPtr(base+current+2, hBitsS, mls)] = current+2; /* here because current+2 could be > iend-8 */ - hashLong[ZSTD_hashPtr(ip-2, hBitsL, 8)] = - hashSmall[ZSTD_hashPtr(ip-2, hBitsS, mls)] = (U32)(ip-2-base); + if (ip <= ilimit) { + /* Fill Table */ + hashLong[ZSTD_hashPtr(base+current+2, hBitsL, 8)] = + hashSmall[ZSTD_hashPtr(base+current+2, hBitsS, mls)] = current+2; /* here because current+2 could be > iend-8 */ + hashLong[ZSTD_hashPtr(ip-2, hBitsL, 8)] = + hashSmall[ZSTD_hashPtr(ip-2, hBitsS, mls)] = (U32)(ip-2-base); - /* check immediate repcode */ - while ( (ip <= ilimit) - && ( (offset_2>0) - & (MEM_read32(ip) == MEM_read32(ip - offset_2)) )) { - /* store sequence */ - size_t const rLength = ZSTD_count(ip+4, ip+4-offset_2, iend) + 4; - { U32 const tmpOff = offset_2; offset_2 = offset_1; offset_1 = tmpOff; } /* swap offset_2 <=> offset_1 */ - hashSmall[ZSTD_hashPtr(ip, hBitsS, mls)] = (U32)(ip-base); - hashLong[ZSTD_hashPtr(ip, hBitsL, 8)] = (U32)(ip-base); - ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, rLength-MINMATCH); - ip += rLength; - anchor = ip; - continue; /* faster when present ... (?) */ - } } } + /* check immediate repcode */ + while ( (ip <= ilimit) + && ( (offset_2>0) + & (MEM_read32(ip) == MEM_read32(ip - offset_2)) )) { + /* store sequence */ + size_t const rLength = ZSTD_count(ip+4, ip+4-offset_2, iend) + 4; + { U32 const tmpOff = offset_2; offset_2 = offset_1; offset_1 = tmpOff; } /* swap offset_2 <=> offset_1 */ + hashSmall[ZSTD_hashPtr(ip, hBitsS, mls)] = (U32)(ip-base); + hashLong[ZSTD_hashPtr(ip, hBitsL, 8)] = (U32)(ip-base); + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, rLength-MINMATCH); + ip += rLength; + anchor = ip; + continue; /* faster when present ... (?) */ + } } } - /* save reps for next block */ - cctx->repToConfirm[0] = offset_1 ? offset_1 : offsetSaved; - cctx->repToConfirm[1] = offset_2 ? offset_2 : offsetSaved; + /* save reps for next block */ + cctx->repToConfirm[0] = offset_1 ? offset_1 : offsetSaved; + cctx->repToConfirm[1] = offset_2 ? offset_2 : offsetSaved; - /* Last Literals */ - { size_t const lastLLSize = iend - anchor; - memcpy(seqStorePtr->lit, anchor, lastLLSize); - seqStorePtr->lit += lastLLSize; - } + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } } static void ZSTD_compressBlock_doubleFast(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { - const U32 mls = ctx->params.cParams.searchLength; - switch(mls) - { - default: /* includes case 3 */ - case 4 : - ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 4); return; - case 5 : - ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 5); return; - case 6 : - ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 6); return; - case 7 : - ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 7); return; - } + const U32 mls = ctx->params.cParams.searchLength; + switch(mls) + { + default: /* includes case 3 */ + case 4 : + ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 4); return; + case 5 : + ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 5); return; + case 6 : + ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 6); return; + case 7 : + ZSTD_compressBlock_doubleFast_generic(ctx, src, srcSize, 7); return; + } } static void ZSTD_compressBlock_doubleFast_extDict_generic(ZSTD_CCtx* ctx, - const void* src, size_t srcSize, - const U32 mls) + const void* src, size_t srcSize, + const U32 mls) { - U32* const hashLong = ctx->hashTable; - U32 const hBitsL = ctx->params.cParams.hashLog; - U32* const hashSmall = ctx->chainTable; - U32 const hBitsS = ctx->params.cParams.chainLog; - seqStore_t* seqStorePtr = &(ctx->seqStore); - const BYTE* const base = ctx->base; - const BYTE* const dictBase = ctx->dictBase; - const BYTE* const istart = (const BYTE*)src; - const BYTE* ip = istart; - const BYTE* anchor = istart; - const U32 lowestIndex = ctx->lowLimit; - const BYTE* const dictStart = dictBase + lowestIndex; - const U32 dictLimit = ctx->dictLimit; - const BYTE* const lowPrefixPtr = base + dictLimit; - const BYTE* const dictEnd = dictBase + dictLimit; - const BYTE* const iend = istart + srcSize; - const BYTE* const ilimit = iend - 8; - U32 offset_1=ctx->rep[0], offset_2=ctx->rep[1]; + U32* const hashLong = ctx->hashTable; + U32 const hBitsL = ctx->params.cParams.hashLog; + U32* const hashSmall = ctx->chainTable; + U32 const hBitsS = ctx->params.cParams.chainLog; + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const base = ctx->base; + const BYTE* const dictBase = ctx->dictBase; + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const U32 lowestIndex = ctx->lowLimit; + const BYTE* const dictStart = dictBase + lowestIndex; + const U32 dictLimit = ctx->dictLimit; + const BYTE* const lowPrefixPtr = base + dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + U32 offset_1=ctx->rep[0], offset_2=ctx->rep[1]; - /* Search Loop */ - while (ip < ilimit) { /* < instead of <=, because (ip+1) */ - const size_t hSmall = ZSTD_hashPtr(ip, hBitsS, mls); - const U32 matchIndex = hashSmall[hSmall]; - const BYTE* matchBase = matchIndex < dictLimit ? dictBase : base; - const BYTE* match = matchBase + matchIndex; + /* Search Loop */ + while (ip < ilimit) { /* < instead of <=, because (ip+1) */ + const size_t hSmall = ZSTD_hashPtr(ip, hBitsS, mls); + const U32 matchIndex = hashSmall[hSmall]; + const BYTE* matchBase = matchIndex < dictLimit ? dictBase : base; + const BYTE* match = matchBase + matchIndex; - const size_t hLong = ZSTD_hashPtr(ip, hBitsL, 8); - const U32 matchLongIndex = hashLong[hLong]; - const BYTE* matchLongBase = matchLongIndex < dictLimit ? dictBase : base; - const BYTE* matchLong = matchLongBase + matchLongIndex; + const size_t hLong = ZSTD_hashPtr(ip, hBitsL, 8); + const U32 matchLongIndex = hashLong[hLong]; + const BYTE* matchLongBase = matchLongIndex < dictLimit ? dictBase : base; + const BYTE* matchLong = matchLongBase + matchLongIndex; - const U32 current = (U32)(ip-base); - const U32 repIndex = current + 1 - offset_1; /* offset_1 expected <= current +1 */ - const BYTE* repBase = repIndex < dictLimit ? dictBase : base; - const BYTE* repMatch = repBase + repIndex; - size_t mLength; - hashSmall[hSmall] = hashLong[hLong] = current; /* update hash table */ + const U32 current = (U32)(ip-base); + const U32 repIndex = current + 1 - offset_1; /* offset_1 expected <= current +1 */ + const BYTE* repBase = repIndex < dictLimit ? dictBase : base; + const BYTE* repMatch = repBase + repIndex; + size_t mLength; + hashSmall[hSmall] = hashLong[hLong] = current; /* update hash table */ - if ( (((U32)((dictLimit-1) - repIndex) >= 3) /* intentional underflow */ & (repIndex > lowestIndex)) - && (MEM_read32(repMatch) == MEM_read32(ip+1)) ) { - const BYTE* repMatchEnd = repIndex < dictLimit ? dictEnd : iend; - mLength = ZSTD_count_2segments(ip+1+4, repMatch+4, iend, repMatchEnd, lowPrefixPtr) + 4; - ip++; - ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); - } else { - if ((matchLongIndex > lowestIndex) && (MEM_read64(matchLong) == MEM_read64(ip))) { - const BYTE* matchEnd = matchLongIndex < dictLimit ? dictEnd : iend; - const BYTE* lowMatchPtr = matchLongIndex < dictLimit ? dictStart : lowPrefixPtr; - U32 offset; - mLength = ZSTD_count_2segments(ip+8, matchLong+8, iend, matchEnd, lowPrefixPtr) + 8; - offset = current - matchLongIndex; - while (((ip>anchor) & (matchLong>lowMatchPtr)) && (ip[-1] == matchLong[-1])) { ip--; matchLong--; mLength++; } /* catch up */ - offset_2 = offset_1; - offset_1 = offset; - ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); + if ( (((U32)((dictLimit-1) - repIndex) >= 3) /* intentional underflow */ & (repIndex > lowestIndex)) + && (MEM_read32(repMatch) == MEM_read32(ip+1)) ) { + const BYTE* repMatchEnd = repIndex < dictLimit ? dictEnd : iend; + mLength = ZSTD_count_2segments(ip+1+4, repMatch+4, iend, repMatchEnd, lowPrefixPtr) + 4; + ip++; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH); + } else { + if ((matchLongIndex > lowestIndex) && (MEM_read64(matchLong) == MEM_read64(ip))) { + const BYTE* matchEnd = matchLongIndex < dictLimit ? dictEnd : iend; + const BYTE* lowMatchPtr = matchLongIndex < dictLimit ? dictStart : lowPrefixPtr; + U32 offset; + mLength = ZSTD_count_2segments(ip+8, matchLong+8, iend, matchEnd, lowPrefixPtr) + 8; + offset = current - matchLongIndex; + while (((ip>anchor) & (matchLong>lowMatchPtr)) && (ip[-1] == matchLong[-1])) { ip--; matchLong--; mLength++; } /* catch up */ + offset_2 = offset_1; + offset_1 = offset; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); - } else if ((matchIndex > lowestIndex) && (MEM_read32(match) == MEM_read32(ip))) { - size_t const h3 = ZSTD_hashPtr(ip+1, hBitsL, 8); - U32 const matchIndex3 = hashLong[h3]; - const BYTE* const match3Base = matchIndex3 < dictLimit ? dictBase : base; - const BYTE* match3 = match3Base + matchIndex3; - U32 offset; - hashLong[h3] = current + 1; - if ( (matchIndex3 > lowestIndex) && (MEM_read64(match3) == MEM_read64(ip+1)) ) { - const BYTE* matchEnd = matchIndex3 < dictLimit ? dictEnd : iend; - const BYTE* lowMatchPtr = matchIndex3 < dictLimit ? dictStart : lowPrefixPtr; - mLength = ZSTD_count_2segments(ip+9, match3+8, iend, matchEnd, lowPrefixPtr) + 8; - ip++; - offset = current+1 - matchIndex3; - while (((ip>anchor) & (match3>lowMatchPtr)) && (ip[-1] == match3[-1])) { ip--; match3--; mLength++; } /* catch up */ - } else { - const BYTE* matchEnd = matchIndex < dictLimit ? dictEnd : iend; - const BYTE* lowMatchPtr = matchIndex < dictLimit ? dictStart : lowPrefixPtr; - mLength = ZSTD_count_2segments(ip+4, match+4, iend, matchEnd, lowPrefixPtr) + 4; - offset = current - matchIndex; - while (((ip>anchor) & (match>lowMatchPtr)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ - } - offset_2 = offset_1; - offset_1 = offset; - ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); + } else if ((matchIndex > lowestIndex) && (MEM_read32(match) == MEM_read32(ip))) { + size_t const h3 = ZSTD_hashPtr(ip+1, hBitsL, 8); + U32 const matchIndex3 = hashLong[h3]; + const BYTE* const match3Base = matchIndex3 < dictLimit ? dictBase : base; + const BYTE* match3 = match3Base + matchIndex3; + U32 offset; + hashLong[h3] = current + 1; + if ( (matchIndex3 > lowestIndex) && (MEM_read64(match3) == MEM_read64(ip+1)) ) { + const BYTE* matchEnd = matchIndex3 < dictLimit ? dictEnd : iend; + const BYTE* lowMatchPtr = matchIndex3 < dictLimit ? dictStart : lowPrefixPtr; + mLength = ZSTD_count_2segments(ip+9, match3+8, iend, matchEnd, lowPrefixPtr) + 8; + ip++; + offset = current+1 - matchIndex3; + while (((ip>anchor) & (match3>lowMatchPtr)) && (ip[-1] == match3[-1])) { ip--; match3--; mLength++; } /* catch up */ + } else { + const BYTE* matchEnd = matchIndex < dictLimit ? dictEnd : iend; + const BYTE* lowMatchPtr = matchIndex < dictLimit ? dictStart : lowPrefixPtr; + mLength = ZSTD_count_2segments(ip+4, match+4, iend, matchEnd, lowPrefixPtr) + 4; + offset = current - matchIndex; + while (((ip>anchor) & (match>lowMatchPtr)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; } /* catch up */ + } + offset_2 = offset_1; + offset_1 = offset; + ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, offset + ZSTD_REP_MOVE, mLength-MINMATCH); - } else { - ip += ((ip-anchor) >> g_searchStrength) + 1; - continue; - } } + } else { + ip += ((ip-anchor) >> g_searchStrength) + 1; + continue; + } } - /* found a match : store it */ - ip += mLength; - anchor = ip; + /* found a match : store it */ + ip += mLength; + anchor = ip; - if (ip <= ilimit) { - /* Fill Table */ - hashSmall[ZSTD_hashPtr(base+current+2, hBitsS, mls)] = current+2; - hashLong[ZSTD_hashPtr(base+current+2, hBitsL, 8)] = current+2; - hashSmall[ZSTD_hashPtr(ip-2, hBitsS, mls)] = (U32)(ip-2-base); - hashLong[ZSTD_hashPtr(ip-2, hBitsL, 8)] = (U32)(ip-2-base); - /* check immediate repcode */ - while (ip <= ilimit) { - U32 const current2 = (U32)(ip-base); - U32 const repIndex2 = current2 - offset_2; - const BYTE* repMatch2 = repIndex2 < dictLimit ? dictBase + repIndex2 : base + repIndex2; - if ( (((U32)((dictLimit-1) - repIndex2) >= 3) & (repIndex2 > lowestIndex)) /* intentional overflow */ - && (MEM_read32(repMatch2) == MEM_read32(ip)) ) { - const BYTE* const repEnd2 = repIndex2 < dictLimit ? dictEnd : iend; - size_t const repLength2 = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch2+EQUAL_READ32, iend, repEnd2, lowPrefixPtr) + EQUAL_READ32; - U32 tmpOffset = offset_2; offset_2 = offset_1; offset_1 = tmpOffset; /* swap offset_2 <=> offset_1 */ - ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, repLength2-MINMATCH); - hashSmall[ZSTD_hashPtr(ip, hBitsS, mls)] = current2; - hashLong[ZSTD_hashPtr(ip, hBitsL, 8)] = current2; - ip += repLength2; - anchor = ip; - continue; - } - break; - } } } + if (ip <= ilimit) { + /* Fill Table */ + hashSmall[ZSTD_hashPtr(base+current+2, hBitsS, mls)] = current+2; + hashLong[ZSTD_hashPtr(base+current+2, hBitsL, 8)] = current+2; + hashSmall[ZSTD_hashPtr(ip-2, hBitsS, mls)] = (U32)(ip-2-base); + hashLong[ZSTD_hashPtr(ip-2, hBitsL, 8)] = (U32)(ip-2-base); + /* check immediate repcode */ + while (ip <= ilimit) { + U32 const current2 = (U32)(ip-base); + U32 const repIndex2 = current2 - offset_2; + const BYTE* repMatch2 = repIndex2 < dictLimit ? dictBase + repIndex2 : base + repIndex2; + if ( (((U32)((dictLimit-1) - repIndex2) >= 3) & (repIndex2 > lowestIndex)) /* intentional overflow */ + && (MEM_read32(repMatch2) == MEM_read32(ip)) ) { + const BYTE* const repEnd2 = repIndex2 < dictLimit ? dictEnd : iend; + size_t const repLength2 = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch2+EQUAL_READ32, iend, repEnd2, lowPrefixPtr) + EQUAL_READ32; + U32 tmpOffset = offset_2; offset_2 = offset_1; offset_1 = tmpOffset; /* swap offset_2 <=> offset_1 */ + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, repLength2-MINMATCH); + hashSmall[ZSTD_hashPtr(ip, hBitsS, mls)] = current2; + hashLong[ZSTD_hashPtr(ip, hBitsL, 8)] = current2; + ip += repLength2; + anchor = ip; + continue; + } + break; + } } } - /* save reps for next block */ - ctx->repToConfirm[0] = offset_1; ctx->repToConfirm[1] = offset_2; + /* save reps for next block */ + ctx->repToConfirm[0] = offset_1; ctx->repToConfirm[1] = offset_2; - /* Last Literals */ - { size_t const lastLLSize = iend - anchor; - memcpy(seqStorePtr->lit, anchor, lastLLSize); - seqStorePtr->lit += lastLLSize; - } + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } } static void ZSTD_compressBlock_doubleFast_extDict(ZSTD_CCtx* ctx, - const void* src, size_t srcSize) + const void* src, size_t srcSize) { - U32 const mls = ctx->params.cParams.searchLength; - switch(mls) - { - default: /* includes case 3 */ - case 4 : - ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 4); return; - case 5 : - ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 5); return; - case 6 : - ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 6); return; - case 7 : - ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 7); return; - } + U32 const mls = ctx->params.cParams.searchLength; + switch(mls) + { + default: /* includes case 3 */ + case 4 : + ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 4); return; + case 5 : + ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 5); return; + case 6 : + ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 6); return; + case 7 : + ZSTD_compressBlock_doubleFast_extDict_generic(ctx, src, srcSize, 7); return; + } } @@ -1522,262 +1522,262 @@ static void ZSTD_compressBlock_doubleFast_extDict(ZSTD_CCtx* ctx, * ip : assumed <= iend-8 . * @return : nb of positions added */ static U32 ZSTD_insertBt1(ZSTD_CCtx* zc, const BYTE* const ip, const U32 mls, const BYTE* const iend, U32 nbCompares, - U32 extDict) + U32 extDict) { - U32* const hashTable = zc->hashTable; - U32 const hashLog = zc->params.cParams.hashLog; - size_t const h = ZSTD_hashPtr(ip, hashLog, mls); - U32* const bt = zc->chainTable; - U32 const btLog = zc->params.cParams.chainLog - 1; - U32 const btMask = (1 << btLog) - 1; - U32 matchIndex = hashTable[h]; - size_t commonLengthSmaller=0, commonLengthLarger=0; - const BYTE* const base = zc->base; - const BYTE* const dictBase = zc->dictBase; - const U32 dictLimit = zc->dictLimit; - const BYTE* const dictEnd = dictBase + dictLimit; - const BYTE* const prefixStart = base + dictLimit; - const BYTE* match; - const U32 current = (U32)(ip-base); - const U32 btLow = btMask >= current ? 0 : current - btMask; - U32* smallerPtr = bt + 2*(current&btMask); - U32* largerPtr = smallerPtr + 1; - U32 dummy32; /* to be nullified at the end */ - U32 const windowLow = zc->lowLimit; - U32 matchEndIdx = current+8; - size_t bestLength = 8; + U32* const hashTable = zc->hashTable; + U32 const hashLog = zc->params.cParams.hashLog; + size_t const h = ZSTD_hashPtr(ip, hashLog, mls); + U32* const bt = zc->chainTable; + U32 const btLog = zc->params.cParams.chainLog - 1; + U32 const btMask = (1 << btLog) - 1; + U32 matchIndex = hashTable[h]; + size_t commonLengthSmaller=0, commonLengthLarger=0; + const BYTE* const base = zc->base; + const BYTE* const dictBase = zc->dictBase; + const U32 dictLimit = zc->dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const prefixStart = base + dictLimit; + const BYTE* match; + const U32 current = (U32)(ip-base); + const U32 btLow = btMask >= current ? 0 : current - btMask; + U32* smallerPtr = bt + 2*(current&btMask); + U32* largerPtr = smallerPtr + 1; + U32 dummy32; /* to be nullified at the end */ + U32 const windowLow = zc->lowLimit; + U32 matchEndIdx = current+8; + size_t bestLength = 8; #ifdef ZSTD_C_PREDICT - U32 predictedSmall = *(bt + 2*((current-1)&btMask) + 0); - U32 predictedLarge = *(bt + 2*((current-1)&btMask) + 1); - predictedSmall += (predictedSmall>0); - predictedLarge += (predictedLarge>0); + U32 predictedSmall = *(bt + 2*((current-1)&btMask) + 0); + U32 predictedLarge = *(bt + 2*((current-1)&btMask) + 1); + predictedSmall += (predictedSmall>0); + predictedLarge += (predictedLarge>0); #endif /* ZSTD_C_PREDICT */ - hashTable[h] = current; /* Update Hash Table */ + hashTable[h] = current; /* Update Hash Table */ - while (nbCompares-- && (matchIndex > windowLow)) { - U32* const nextPtr = bt + 2*(matchIndex & btMask); - size_t matchLength = MIN(commonLengthSmaller, commonLengthLarger); /* guaranteed minimum nb of common bytes */ + while (nbCompares-- && (matchIndex > windowLow)) { + U32* const nextPtr = bt + 2*(matchIndex & btMask); + size_t matchLength = MIN(commonLengthSmaller, commonLengthLarger); /* guaranteed minimum nb of common bytes */ #ifdef ZSTD_C_PREDICT /* note : can create issues when hlog small <= 11 */ - const U32* predictPtr = bt + 2*((matchIndex-1) & btMask); /* written this way, as bt is a roll buffer */ - if (matchIndex == predictedSmall) { - /* no need to check length, result known */ - *smallerPtr = matchIndex; - if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ - smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ - matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ - predictedSmall = predictPtr[1] + (predictPtr[1]>0); - continue; - } - if (matchIndex == predictedLarge) { - *largerPtr = matchIndex; - if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ - largerPtr = nextPtr; - matchIndex = nextPtr[0]; - predictedLarge = predictPtr[0] + (predictPtr[0]>0); - continue; - } + const U32* predictPtr = bt + 2*((matchIndex-1) & btMask); /* written this way, as bt is a roll buffer */ + if (matchIndex == predictedSmall) { + /* no need to check length, result known */ + *smallerPtr = matchIndex; + if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ + matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ + predictedSmall = predictPtr[1] + (predictPtr[1]>0); + continue; + } + if (matchIndex == predictedLarge) { + *largerPtr = matchIndex; + if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + largerPtr = nextPtr; + matchIndex = nextPtr[0]; + predictedLarge = predictPtr[0] + (predictPtr[0]>0); + continue; + } #endif - if ((!extDict) || (matchIndex+matchLength >= dictLimit)) { - match = base + matchIndex; - if (match[matchLength] == ip[matchLength]) - matchLength += ZSTD_count(ip+matchLength+1, match+matchLength+1, iend) +1; - } else { - match = dictBase + matchIndex; - matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iend, dictEnd, prefixStart); - if (matchIndex+matchLength >= dictLimit) - match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ - } + if ((!extDict) || (matchIndex+matchLength >= dictLimit)) { + match = base + matchIndex; + if (match[matchLength] == ip[matchLength]) + matchLength += ZSTD_count(ip+matchLength+1, match+matchLength+1, iend) +1; + } else { + match = dictBase + matchIndex; + matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iend, dictEnd, prefixStart); + if (matchIndex+matchLength >= dictLimit) + match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ + } - if (matchLength > bestLength) { - bestLength = matchLength; - if (matchLength > matchEndIdx - matchIndex) - matchEndIdx = matchIndex + (U32)matchLength; - } + if (matchLength > bestLength) { + bestLength = matchLength; + if (matchLength > matchEndIdx - matchIndex) + matchEndIdx = matchIndex + (U32)matchLength; + } - if (ip+matchLength == iend) /* equal : no way to know if inf or sup */ - break; /* drop , to guarantee consistency ; miss a bit of compression, but other solutions can corrupt the tree */ + if (ip+matchLength == iend) /* equal : no way to know if inf or sup */ + break; /* drop , to guarantee consistency ; miss a bit of compression, but other solutions can corrupt the tree */ - if (match[matchLength] < ip[matchLength]) { /* necessarily within correct buffer */ - /* match is smaller than current */ - *smallerPtr = matchIndex; /* update smaller idx */ - commonLengthSmaller = matchLength; /* all smaller will now have at least this guaranteed common length */ - if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ - smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ - matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ - } else { - /* match is larger than current */ - *largerPtr = matchIndex; - commonLengthLarger = matchLength; - if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ - largerPtr = nextPtr; - matchIndex = nextPtr[0]; - } } + if (match[matchLength] < ip[matchLength]) { /* necessarily within correct buffer */ + /* match is smaller than current */ + *smallerPtr = matchIndex; /* update smaller idx */ + commonLengthSmaller = matchLength; /* all smaller will now have at least this guaranteed common length */ + if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ + matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ + } else { + /* match is larger than current */ + *largerPtr = matchIndex; + commonLengthLarger = matchLength; + if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + largerPtr = nextPtr; + matchIndex = nextPtr[0]; + } } - *smallerPtr = *largerPtr = 0; - if (bestLength > 384) return MIN(192, (U32)(bestLength - 384)); /* speed optimization */ - if (matchEndIdx > current + 8) return matchEndIdx - current - 8; - return 1; + *smallerPtr = *largerPtr = 0; + if (bestLength > 384) return MIN(192, (U32)(bestLength - 384)); /* speed optimization */ + if (matchEndIdx > current + 8) return matchEndIdx - current - 8; + return 1; } static size_t ZSTD_insertBtAndFindBestMatch ( - ZSTD_CCtx* zc, - const BYTE* const ip, const BYTE* const iend, - size_t* offsetPtr, - U32 nbCompares, const U32 mls, - U32 extDict) + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iend, + size_t* offsetPtr, + U32 nbCompares, const U32 mls, + U32 extDict) { - U32* const hashTable = zc->hashTable; - U32 const hashLog = zc->params.cParams.hashLog; - size_t const h = ZSTD_hashPtr(ip, hashLog, mls); - U32* const bt = zc->chainTable; - U32 const btLog = zc->params.cParams.chainLog - 1; - U32 const btMask = (1 << btLog) - 1; - U32 matchIndex = hashTable[h]; - size_t commonLengthSmaller=0, commonLengthLarger=0; - const BYTE* const base = zc->base; - const BYTE* const dictBase = zc->dictBase; - const U32 dictLimit = zc->dictLimit; - const BYTE* const dictEnd = dictBase + dictLimit; - const BYTE* const prefixStart = base + dictLimit; - const U32 current = (U32)(ip-base); - const U32 btLow = btMask >= current ? 0 : current - btMask; - const U32 windowLow = zc->lowLimit; - U32* smallerPtr = bt + 2*(current&btMask); - U32* largerPtr = bt + 2*(current&btMask) + 1; - U32 matchEndIdx = current+8; - U32 dummy32; /* to be nullified at the end */ - size_t bestLength = 0; + U32* const hashTable = zc->hashTable; + U32 const hashLog = zc->params.cParams.hashLog; + size_t const h = ZSTD_hashPtr(ip, hashLog, mls); + U32* const bt = zc->chainTable; + U32 const btLog = zc->params.cParams.chainLog - 1; + U32 const btMask = (1 << btLog) - 1; + U32 matchIndex = hashTable[h]; + size_t commonLengthSmaller=0, commonLengthLarger=0; + const BYTE* const base = zc->base; + const BYTE* const dictBase = zc->dictBase; + const U32 dictLimit = zc->dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const prefixStart = base + dictLimit; + const U32 current = (U32)(ip-base); + const U32 btLow = btMask >= current ? 0 : current - btMask; + const U32 windowLow = zc->lowLimit; + U32* smallerPtr = bt + 2*(current&btMask); + U32* largerPtr = bt + 2*(current&btMask) + 1; + U32 matchEndIdx = current+8; + U32 dummy32; /* to be nullified at the end */ + size_t bestLength = 0; - hashTable[h] = current; /* Update Hash Table */ + hashTable[h] = current; /* Update Hash Table */ - while (nbCompares-- && (matchIndex > windowLow)) { - U32* const nextPtr = bt + 2*(matchIndex & btMask); - size_t matchLength = MIN(commonLengthSmaller, commonLengthLarger); /* guaranteed minimum nb of common bytes */ - const BYTE* match; + while (nbCompares-- && (matchIndex > windowLow)) { + U32* const nextPtr = bt + 2*(matchIndex & btMask); + size_t matchLength = MIN(commonLengthSmaller, commonLengthLarger); /* guaranteed minimum nb of common bytes */ + const BYTE* match; - if ((!extDict) || (matchIndex+matchLength >= dictLimit)) { - match = base + matchIndex; - if (match[matchLength] == ip[matchLength]) - matchLength += ZSTD_count(ip+matchLength+1, match+matchLength+1, iend) +1; - } else { - match = dictBase + matchIndex; - matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iend, dictEnd, prefixStart); - if (matchIndex+matchLength >= dictLimit) - match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ - } + if ((!extDict) || (matchIndex+matchLength >= dictLimit)) { + match = base + matchIndex; + if (match[matchLength] == ip[matchLength]) + matchLength += ZSTD_count(ip+matchLength+1, match+matchLength+1, iend) +1; + } else { + match = dictBase + matchIndex; + matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iend, dictEnd, prefixStart); + if (matchIndex+matchLength >= dictLimit) + match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ + } - if (matchLength > bestLength) { - if (matchLength > matchEndIdx - matchIndex) - matchEndIdx = matchIndex + (U32)matchLength; - if ( (4*(int)(matchLength-bestLength)) > (int)(ZSTD_highbit32(current-matchIndex+1) - ZSTD_highbit32((U32)offsetPtr[0]+1)) ) - bestLength = matchLength, *offsetPtr = ZSTD_REP_MOVE + current - matchIndex; - if (ip+matchLength == iend) /* equal : no way to know if inf or sup */ - break; /* drop, to guarantee consistency (miss a little bit of compression) */ - } + if (matchLength > bestLength) { + if (matchLength > matchEndIdx - matchIndex) + matchEndIdx = matchIndex + (U32)matchLength; + if ( (4*(int)(matchLength-bestLength)) > (int)(ZSTD_highbit32(current-matchIndex+1) - ZSTD_highbit32((U32)offsetPtr[0]+1)) ) + bestLength = matchLength, *offsetPtr = ZSTD_REP_MOVE + current - matchIndex; + if (ip+matchLength == iend) /* equal : no way to know if inf or sup */ + break; /* drop, to guarantee consistency (miss a little bit of compression) */ + } - if (match[matchLength] < ip[matchLength]) { - /* match is smaller than current */ - *smallerPtr = matchIndex; /* update smaller idx */ - commonLengthSmaller = matchLength; /* all smaller will now have at least this guaranteed common length */ - if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ - smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ - matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ - } else { - /* match is larger than current */ - *largerPtr = matchIndex; - commonLengthLarger = matchLength; - if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ - largerPtr = nextPtr; - matchIndex = nextPtr[0]; - } } + if (match[matchLength] < ip[matchLength]) { + /* match is smaller than current */ + *smallerPtr = matchIndex; /* update smaller idx */ + commonLengthSmaller = matchLength; /* all smaller will now have at least this guaranteed common length */ + if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ + matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ + } else { + /* match is larger than current */ + *largerPtr = matchIndex; + commonLengthLarger = matchLength; + if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + largerPtr = nextPtr; + matchIndex = nextPtr[0]; + } } - *smallerPtr = *largerPtr = 0; + *smallerPtr = *largerPtr = 0; - zc->nextToUpdate = (matchEndIdx > current + 8) ? matchEndIdx - 8 : current+1; - return bestLength; + zc->nextToUpdate = (matchEndIdx > current + 8) ? matchEndIdx - 8 : current+1; + return bestLength; } static void ZSTD_updateTree(ZSTD_CCtx* zc, const BYTE* const ip, const BYTE* const iend, const U32 nbCompares, const U32 mls) { - const BYTE* const base = zc->base; - const U32 target = (U32)(ip - base); - U32 idx = zc->nextToUpdate; + const BYTE* const base = zc->base; + const U32 target = (U32)(ip - base); + U32 idx = zc->nextToUpdate; - while(idx < target) - idx += ZSTD_insertBt1(zc, base+idx, mls, iend, nbCompares, 0); + while(idx < target) + idx += ZSTD_insertBt1(zc, base+idx, mls, iend, nbCompares, 0); } /** ZSTD_BtFindBestMatch() : Tree updater, providing best match */ static size_t ZSTD_BtFindBestMatch ( - ZSTD_CCtx* zc, - const BYTE* const ip, const BYTE* const iLimit, - size_t* offsetPtr, - const U32 maxNbAttempts, const U32 mls) + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 mls) { - if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ - ZSTD_updateTree(zc, ip, iLimit, maxNbAttempts, mls); - return ZSTD_insertBtAndFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, mls, 0); + if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ + ZSTD_updateTree(zc, ip, iLimit, maxNbAttempts, mls); + return ZSTD_insertBtAndFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, mls, 0); } static size_t ZSTD_BtFindBestMatch_selectMLS ( - ZSTD_CCtx* zc, /* Index table will be updated */ - const BYTE* ip, const BYTE* const iLimit, - size_t* offsetPtr, - const U32 maxNbAttempts, const U32 matchLengthSearch) + ZSTD_CCtx* zc, /* Index table will be updated */ + const BYTE* ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 matchLengthSearch) { - switch(matchLengthSearch) - { - default : /* includes case 3 */ - case 4 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4); - case 5 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5); - case 7 : - case 6 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6); - } + switch(matchLengthSearch) + { + default : /* includes case 3 */ + case 4 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4); + case 5 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5); + case 7 : + case 6 : return ZSTD_BtFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6); + } } static void ZSTD_updateTree_extDict(ZSTD_CCtx* zc, const BYTE* const ip, const BYTE* const iend, const U32 nbCompares, const U32 mls) { - const BYTE* const base = zc->base; - const U32 target = (U32)(ip - base); - U32 idx = zc->nextToUpdate; + const BYTE* const base = zc->base; + const U32 target = (U32)(ip - base); + U32 idx = zc->nextToUpdate; - while (idx < target) idx += ZSTD_insertBt1(zc, base+idx, mls, iend, nbCompares, 1); + while (idx < target) idx += ZSTD_insertBt1(zc, base+idx, mls, iend, nbCompares, 1); } /** Tree updater, providing best match */ static size_t ZSTD_BtFindBestMatch_extDict ( - ZSTD_CCtx* zc, - const BYTE* const ip, const BYTE* const iLimit, - size_t* offsetPtr, - const U32 maxNbAttempts, const U32 mls) + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 mls) { - if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ - ZSTD_updateTree_extDict(zc, ip, iLimit, maxNbAttempts, mls); - return ZSTD_insertBtAndFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, mls, 1); + if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ + ZSTD_updateTree_extDict(zc, ip, iLimit, maxNbAttempts, mls); + return ZSTD_insertBtAndFindBestMatch(zc, ip, iLimit, offsetPtr, maxNbAttempts, mls, 1); } static size_t ZSTD_BtFindBestMatch_selectMLS_extDict ( - ZSTD_CCtx* zc, /* Index table will be updated */ - const BYTE* ip, const BYTE* const iLimit, - size_t* offsetPtr, - const U32 maxNbAttempts, const U32 matchLengthSearch) + ZSTD_CCtx* zc, /* Index table will be updated */ + const BYTE* ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 matchLengthSearch) { - switch(matchLengthSearch) - { - default : /* includes case 3 */ - case 4 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4); - case 5 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5); - case 7 : - case 6 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6); - } + switch(matchLengthSearch) + { + default : /* includes case 3 */ + case 4 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4); + case 5 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5); + case 7 : + case 6 : return ZSTD_BtFindBestMatch_extDict(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6); + } } @@ -1792,106 +1792,106 @@ static size_t ZSTD_BtFindBestMatch_selectMLS_extDict ( FORCE_INLINE U32 ZSTD_insertAndFindFirstIndex (ZSTD_CCtx* zc, const BYTE* ip, U32 mls) { - U32* const hashTable = zc->hashTable; - const U32 hashLog = zc->params.cParams.hashLog; - U32* const chainTable = zc->chainTable; - const U32 chainMask = (1 << zc->params.cParams.chainLog) - 1; - const BYTE* const base = zc->base; - const U32 target = (U32)(ip - base); - U32 idx = zc->nextToUpdate; + U32* const hashTable = zc->hashTable; + const U32 hashLog = zc->params.cParams.hashLog; + U32* const chainTable = zc->chainTable; + const U32 chainMask = (1 << zc->params.cParams.chainLog) - 1; + const BYTE* const base = zc->base; + const U32 target = (U32)(ip - base); + U32 idx = zc->nextToUpdate; - while(idx < target) { /* catch up */ - size_t const h = ZSTD_hashPtr(base+idx, hashLog, mls); - NEXT_IN_CHAIN(idx, chainMask) = hashTable[h]; - hashTable[h] = idx; - idx++; - } + while(idx < target) { /* catch up */ + size_t const h = ZSTD_hashPtr(base+idx, hashLog, mls); + NEXT_IN_CHAIN(idx, chainMask) = hashTable[h]; + hashTable[h] = idx; + idx++; + } - zc->nextToUpdate = target; - return hashTable[ZSTD_hashPtr(ip, hashLog, mls)]; + zc->nextToUpdate = target; + return hashTable[ZSTD_hashPtr(ip, hashLog, mls)]; } FORCE_INLINE /* inlining is important to hardwire a hot branch (template emulation) */ size_t ZSTD_HcFindBestMatch_generic ( - ZSTD_CCtx* zc, /* Index table will be updated */ - const BYTE* const ip, const BYTE* const iLimit, - size_t* offsetPtr, - const U32 maxNbAttempts, const U32 mls, const U32 extDict) + ZSTD_CCtx* zc, /* Index table will be updated */ + const BYTE* const ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 mls, const U32 extDict) { - U32* const chainTable = zc->chainTable; - const U32 chainSize = (1 << zc->params.cParams.chainLog); - const U32 chainMask = chainSize-1; - const BYTE* const base = zc->base; - const BYTE* const dictBase = zc->dictBase; - const U32 dictLimit = zc->dictLimit; - const BYTE* const prefixStart = base + dictLimit; - const BYTE* const dictEnd = dictBase + dictLimit; - const U32 lowLimit = zc->lowLimit; - const U32 current = (U32)(ip-base); - const U32 minChain = current > chainSize ? current - chainSize : 0; - int nbAttempts=maxNbAttempts; - size_t ml=EQUAL_READ32-1; + U32* const chainTable = zc->chainTable; + const U32 chainSize = (1 << zc->params.cParams.chainLog); + const U32 chainMask = chainSize-1; + const BYTE* const base = zc->base; + const BYTE* const dictBase = zc->dictBase; + const U32 dictLimit = zc->dictLimit; + const BYTE* const prefixStart = base + dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const U32 lowLimit = zc->lowLimit; + const U32 current = (U32)(ip-base); + const U32 minChain = current > chainSize ? current - chainSize : 0; + int nbAttempts=maxNbAttempts; + size_t ml=EQUAL_READ32-1; - /* HC4 match finder */ - U32 matchIndex = ZSTD_insertAndFindFirstIndex (zc, ip, mls); + /* HC4 match finder */ + U32 matchIndex = ZSTD_insertAndFindFirstIndex (zc, ip, mls); - for ( ; (matchIndex>lowLimit) & (nbAttempts>0) ; nbAttempts--) { - const BYTE* match; - size_t currentMl=0; - if ((!extDict) || matchIndex >= dictLimit) { - match = base + matchIndex; - if (match[ml] == ip[ml]) /* potentially better */ - currentMl = ZSTD_count(ip, match, iLimit); - } else { - match = dictBase + matchIndex; - if (MEM_read32(match) == MEM_read32(ip)) /* assumption : matchIndex <= dictLimit-4 (by table construction) */ - currentMl = ZSTD_count_2segments(ip+EQUAL_READ32, match+EQUAL_READ32, iLimit, dictEnd, prefixStart) + EQUAL_READ32; - } + for ( ; (matchIndex>lowLimit) & (nbAttempts>0) ; nbAttempts--) { + const BYTE* match; + size_t currentMl=0; + if ((!extDict) || matchIndex >= dictLimit) { + match = base + matchIndex; + if (match[ml] == ip[ml]) /* potentially better */ + currentMl = ZSTD_count(ip, match, iLimit); + } else { + match = dictBase + matchIndex; + if (MEM_read32(match) == MEM_read32(ip)) /* assumption : matchIndex <= dictLimit-4 (by table construction) */ + currentMl = ZSTD_count_2segments(ip+EQUAL_READ32, match+EQUAL_READ32, iLimit, dictEnd, prefixStart) + EQUAL_READ32; + } - /* save best solution */ - if (currentMl > ml) { ml = currentMl; *offsetPtr = current - matchIndex + ZSTD_REP_MOVE; if (ip+currentMl == iLimit) break; /* best possible, and avoid read overflow*/ } + /* save best solution */ + if (currentMl > ml) { ml = currentMl; *offsetPtr = current - matchIndex + ZSTD_REP_MOVE; if (ip+currentMl == iLimit) break; /* best possible, and avoid read overflow*/ } - if (matchIndex <= minChain) break; - matchIndex = NEXT_IN_CHAIN(matchIndex, chainMask); - } + if (matchIndex <= minChain) break; + matchIndex = NEXT_IN_CHAIN(matchIndex, chainMask); + } - return ml; + return ml; } FORCE_INLINE size_t ZSTD_HcFindBestMatch_selectMLS ( - ZSTD_CCtx* zc, - const BYTE* ip, const BYTE* const iLimit, - size_t* offsetPtr, - const U32 maxNbAttempts, const U32 matchLengthSearch) + ZSTD_CCtx* zc, + const BYTE* ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 matchLengthSearch) { - switch(matchLengthSearch) - { - default : /* includes case 3 */ - case 4 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4, 0); - case 5 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5, 0); - case 7 : - case 6 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6, 0); - } + switch(matchLengthSearch) + { + default : /* includes case 3 */ + case 4 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4, 0); + case 5 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5, 0); + case 7 : + case 6 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6, 0); + } } FORCE_INLINE size_t ZSTD_HcFindBestMatch_extDict_selectMLS ( - ZSTD_CCtx* zc, - const BYTE* ip, const BYTE* const iLimit, - size_t* offsetPtr, - const U32 maxNbAttempts, const U32 matchLengthSearch) + ZSTD_CCtx* zc, + const BYTE* ip, const BYTE* const iLimit, + size_t* offsetPtr, + const U32 maxNbAttempts, const U32 matchLengthSearch) { - switch(matchLengthSearch) - { - default : /* includes case 3 */ - case 4 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4, 1); - case 5 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5, 1); - case 7 : - case 6 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6, 1); - } + switch(matchLengthSearch) + { + default : /* includes case 3 */ + case 4 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 4, 1); + case 5 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 5, 1); + case 7 : + case 6 : return ZSTD_HcFindBestMatch_generic(zc, ip, iLimit, offsetPtr, maxNbAttempts, 6, 1); + } } @@ -1900,351 +1900,351 @@ FORCE_INLINE size_t ZSTD_HcFindBestMatch_extDict_selectMLS ( *********************************/ FORCE_INLINE void ZSTD_compressBlock_lazy_generic(ZSTD_CCtx* ctx, - const void* src, size_t srcSize, - const U32 searchMethod, const U32 depth) + const void* src, size_t srcSize, + const U32 searchMethod, const U32 depth) { - seqStore_t* seqStorePtr = &(ctx->seqStore); - const BYTE* const istart = (const BYTE*)src; - const BYTE* ip = istart; - const BYTE* anchor = istart; - const BYTE* const iend = istart + srcSize; - const BYTE* const ilimit = iend - 8; - const BYTE* const base = ctx->base + ctx->dictLimit; + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + const BYTE* const base = ctx->base + ctx->dictLimit; - U32 const maxSearches = 1 << ctx->params.cParams.searchLog; - U32 const mls = ctx->params.cParams.searchLength; + U32 const maxSearches = 1 << ctx->params.cParams.searchLog; + U32 const mls = ctx->params.cParams.searchLength; - typedef size_t (*searchMax_f)(ZSTD_CCtx* zc, const BYTE* ip, const BYTE* iLimit, - size_t* offsetPtr, - U32 maxNbAttempts, U32 matchLengthSearch); - searchMax_f const searchMax = searchMethod ? ZSTD_BtFindBestMatch_selectMLS : ZSTD_HcFindBestMatch_selectMLS; - U32 offset_1 = ctx->rep[0], offset_2 = ctx->rep[1], savedOffset=0; + typedef size_t (*searchMax_f)(ZSTD_CCtx* zc, const BYTE* ip, const BYTE* iLimit, + size_t* offsetPtr, + U32 maxNbAttempts, U32 matchLengthSearch); + searchMax_f const searchMax = searchMethod ? ZSTD_BtFindBestMatch_selectMLS : ZSTD_HcFindBestMatch_selectMLS; + U32 offset_1 = ctx->rep[0], offset_2 = ctx->rep[1], savedOffset=0; - /* init */ - ip += (ip==base); - ctx->nextToUpdate3 = ctx->nextToUpdate; - { U32 const maxRep = (U32)(ip-base); - if (offset_2 > maxRep) savedOffset = offset_2, offset_2 = 0; - if (offset_1 > maxRep) savedOffset = offset_1, offset_1 = 0; - } + /* init */ + ip += (ip==base); + ctx->nextToUpdate3 = ctx->nextToUpdate; + { U32 const maxRep = (U32)(ip-base); + if (offset_2 > maxRep) savedOffset = offset_2, offset_2 = 0; + if (offset_1 > maxRep) savedOffset = offset_1, offset_1 = 0; + } - /* Match Loop */ - while (ip < ilimit) { - size_t matchLength=0; - size_t offset=0; - const BYTE* start=ip+1; + /* Match Loop */ + while (ip < ilimit) { + size_t matchLength=0; + size_t offset=0; + const BYTE* start=ip+1; - /* check repCode */ - if ((offset_1>0) & (MEM_read32(ip+1) == MEM_read32(ip+1 - offset_1))) { - /* repcode : we take it */ - matchLength = ZSTD_count(ip+1+EQUAL_READ32, ip+1+EQUAL_READ32-offset_1, iend) + EQUAL_READ32; - if (depth==0) goto _storeSequence; - } + /* check repCode */ + if ((offset_1>0) & (MEM_read32(ip+1) == MEM_read32(ip+1 - offset_1))) { + /* repcode : we take it */ + matchLength = ZSTD_count(ip+1+EQUAL_READ32, ip+1+EQUAL_READ32-offset_1, iend) + EQUAL_READ32; + if (depth==0) goto _storeSequence; + } - /* first search (depth 0) */ - { size_t offsetFound = 99999999; - size_t const ml2 = searchMax(ctx, ip, iend, &offsetFound, maxSearches, mls); - if (ml2 > matchLength) - matchLength = ml2, start = ip, offset=offsetFound; - } + /* first search (depth 0) */ + { size_t offsetFound = 99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offsetFound, maxSearches, mls); + if (ml2 > matchLength) + matchLength = ml2, start = ip, offset=offsetFound; + } - if (matchLength < EQUAL_READ32) { - ip += ((ip-anchor) >> g_searchStrength) + 1; /* jump faster over incompressible sections */ - continue; - } + if (matchLength < EQUAL_READ32) { + ip += ((ip-anchor) >> g_searchStrength) + 1; /* jump faster over incompressible sections */ + continue; + } - /* let's try to find a better solution */ - if (depth>=1) - while (ip0) & (MEM_read32(ip) == MEM_read32(ip - offset_1)))) { - size_t const mlRep = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_1, iend) + EQUAL_READ32; - int const gain2 = (int)(mlRep * 3); - int const gain1 = (int)(matchLength*3 - ZSTD_highbit32((U32)offset+1) + 1); - if ((mlRep >= EQUAL_READ32) && (gain2 > gain1)) - matchLength = mlRep, offset = 0, start = ip; - } - { size_t offset2=99999999; - size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); - int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ - int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 4); - if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { - matchLength = ml2, offset = offset2, start = ip; - continue; /* search a better one */ - } } + /* let's try to find a better solution */ + if (depth>=1) + while (ip0) & (MEM_read32(ip) == MEM_read32(ip - offset_1)))) { + size_t const mlRep = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_1, iend) + EQUAL_READ32; + int const gain2 = (int)(mlRep * 3); + int const gain1 = (int)(matchLength*3 - ZSTD_highbit32((U32)offset+1) + 1); + if ((mlRep >= EQUAL_READ32) && (gain2 > gain1)) + matchLength = mlRep, offset = 0, start = ip; + } + { size_t offset2=99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); + int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ + int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 4); + if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { + matchLength = ml2, offset = offset2, start = ip; + continue; /* search a better one */ + } } - /* let's find an even better one */ - if ((depth==2) && (ip0) & (MEM_read32(ip) == MEM_read32(ip - offset_1)))) { - size_t const ml2 = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_1, iend) + EQUAL_READ32; - int const gain2 = (int)(ml2 * 4); - int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 1); - if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) - matchLength = ml2, offset = 0, start = ip; - } - { size_t offset2=99999999; - size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); - int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ - int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 7); - if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { - matchLength = ml2, offset = offset2, start = ip; - continue; - } } } - break; /* nothing found : store previous solution */ - } + /* let's find an even better one */ + if ((depth==2) && (ip0) & (MEM_read32(ip) == MEM_read32(ip - offset_1)))) { + size_t const ml2 = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_1, iend) + EQUAL_READ32; + int const gain2 = (int)(ml2 * 4); + int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 1); + if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) + matchLength = ml2, offset = 0, start = ip; + } + { size_t offset2=99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); + int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ + int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 7); + if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { + matchLength = ml2, offset = offset2, start = ip; + continue; + } } } + break; /* nothing found : store previous solution */ + } - /* catch up */ - if (offset) { - while ((start>anchor) && (start>base+offset-ZSTD_REP_MOVE) && (start[-1] == start[-1-offset+ZSTD_REP_MOVE])) /* only search for offset within prefix */ - { start--; matchLength++; } - offset_2 = offset_1; offset_1 = (U32)(offset - ZSTD_REP_MOVE); - } + /* catch up */ + if (offset) { + while ((start>anchor) && (start>base+offset-ZSTD_REP_MOVE) && (start[-1] == start[-1-offset+ZSTD_REP_MOVE])) /* only search for offset within prefix */ + { start--; matchLength++; } + offset_2 = offset_1; offset_1 = (U32)(offset - ZSTD_REP_MOVE); + } - /* store sequence */ + /* store sequence */ _storeSequence: - { size_t const litLength = start - anchor; - ZSTD_storeSeq(seqStorePtr, litLength, anchor, (U32)offset, matchLength-MINMATCH); - anchor = ip = start + matchLength; - } + { size_t const litLength = start - anchor; + ZSTD_storeSeq(seqStorePtr, litLength, anchor, (U32)offset, matchLength-MINMATCH); + anchor = ip = start + matchLength; + } - /* check immediate repcode */ - while ( (ip <= ilimit) - && ((offset_2>0) - & (MEM_read32(ip) == MEM_read32(ip - offset_2)) )) { - /* store sequence */ - matchLength = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_2, iend) + EQUAL_READ32; - offset = offset_2; offset_2 = offset_1; offset_1 = (U32)offset; /* swap repcodes */ - ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, matchLength-MINMATCH); - ip += matchLength; - anchor = ip; - continue; /* faster when present ... (?) */ - } } + /* check immediate repcode */ + while ( (ip <= ilimit) + && ((offset_2>0) + & (MEM_read32(ip) == MEM_read32(ip - offset_2)) )) { + /* store sequence */ + matchLength = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_2, iend) + EQUAL_READ32; + offset = offset_2; offset_2 = offset_1; offset_1 = (U32)offset; /* swap repcodes */ + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, matchLength-MINMATCH); + ip += matchLength; + anchor = ip; + continue; /* faster when present ... (?) */ + } } - /* Save reps for next block */ - ctx->repToConfirm[0] = offset_1 ? offset_1 : savedOffset; - ctx->repToConfirm[1] = offset_2 ? offset_2 : savedOffset; + /* Save reps for next block */ + ctx->repToConfirm[0] = offset_1 ? offset_1 : savedOffset; + ctx->repToConfirm[1] = offset_2 ? offset_2 : savedOffset; - /* Last Literals */ - { size_t const lastLLSize = iend - anchor; - memcpy(seqStorePtr->lit, anchor, lastLLSize); - seqStorePtr->lit += lastLLSize; - } + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } } static void ZSTD_compressBlock_btlazy2(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { - ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 1, 2); + ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 1, 2); } static void ZSTD_compressBlock_lazy2(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { - ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 0, 2); + ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 0, 2); } static void ZSTD_compressBlock_lazy(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { - ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 0, 1); + ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 0, 1); } static void ZSTD_compressBlock_greedy(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { - ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 0, 0); + ZSTD_compressBlock_lazy_generic(ctx, src, srcSize, 0, 0); } FORCE_INLINE void ZSTD_compressBlock_lazy_extDict_generic(ZSTD_CCtx* ctx, - const void* src, size_t srcSize, - const U32 searchMethod, const U32 depth) + const void* src, size_t srcSize, + const U32 searchMethod, const U32 depth) { - seqStore_t* seqStorePtr = &(ctx->seqStore); - const BYTE* const istart = (const BYTE*)src; - const BYTE* ip = istart; - const BYTE* anchor = istart; - const BYTE* const iend = istart + srcSize; - const BYTE* const ilimit = iend - 8; - const BYTE* const base = ctx->base; - const U32 dictLimit = ctx->dictLimit; - const U32 lowestIndex = ctx->lowLimit; - const BYTE* const prefixStart = base + dictLimit; - const BYTE* const dictBase = ctx->dictBase; - const BYTE* const dictEnd = dictBase + dictLimit; - const BYTE* const dictStart = dictBase + ctx->lowLimit; + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + const BYTE* const base = ctx->base; + const U32 dictLimit = ctx->dictLimit; + const U32 lowestIndex = ctx->lowLimit; + const BYTE* const prefixStart = base + dictLimit; + const BYTE* const dictBase = ctx->dictBase; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const dictStart = dictBase + ctx->lowLimit; - const U32 maxSearches = 1 << ctx->params.cParams.searchLog; - const U32 mls = ctx->params.cParams.searchLength; + const U32 maxSearches = 1 << ctx->params.cParams.searchLog; + const U32 mls = ctx->params.cParams.searchLength; - typedef size_t (*searchMax_f)(ZSTD_CCtx* zc, const BYTE* ip, const BYTE* iLimit, - size_t* offsetPtr, - U32 maxNbAttempts, U32 matchLengthSearch); - searchMax_f searchMax = searchMethod ? ZSTD_BtFindBestMatch_selectMLS_extDict : ZSTD_HcFindBestMatch_extDict_selectMLS; + typedef size_t (*searchMax_f)(ZSTD_CCtx* zc, const BYTE* ip, const BYTE* iLimit, + size_t* offsetPtr, + U32 maxNbAttempts, U32 matchLengthSearch); + searchMax_f searchMax = searchMethod ? ZSTD_BtFindBestMatch_selectMLS_extDict : ZSTD_HcFindBestMatch_extDict_selectMLS; - U32 offset_1 = ctx->rep[0], offset_2 = ctx->rep[1]; + U32 offset_1 = ctx->rep[0], offset_2 = ctx->rep[1]; - /* init */ - ctx->nextToUpdate3 = ctx->nextToUpdate; - ip += (ip == prefixStart); + /* init */ + ctx->nextToUpdate3 = ctx->nextToUpdate; + ip += (ip == prefixStart); - /* Match Loop */ - while (ip < ilimit) { - size_t matchLength=0; - size_t offset=0; - const BYTE* start=ip+1; - U32 current = (U32)(ip-base); + /* Match Loop */ + while (ip < ilimit) { + size_t matchLength=0; + size_t offset=0; + const BYTE* start=ip+1; + U32 current = (U32)(ip-base); - /* check repCode */ - { const U32 repIndex = (U32)(current+1 - offset_1); - const BYTE* const repBase = repIndex < dictLimit ? dictBase : base; - const BYTE* const repMatch = repBase + repIndex; - if (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex > lowestIndex)) /* intentional overflow */ - if (MEM_read32(ip+1) == MEM_read32(repMatch)) { - /* repcode detected we should take it */ - const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; - matchLength = ZSTD_count_2segments(ip+1+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; - if (depth==0) goto _storeSequence; - } } + /* check repCode */ + { const U32 repIndex = (U32)(current+1 - offset_1); + const BYTE* const repBase = repIndex < dictLimit ? dictBase : base; + const BYTE* const repMatch = repBase + repIndex; + if (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex > lowestIndex)) /* intentional overflow */ + if (MEM_read32(ip+1) == MEM_read32(repMatch)) { + /* repcode detected we should take it */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + matchLength = ZSTD_count_2segments(ip+1+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; + if (depth==0) goto _storeSequence; + } } - /* first search (depth 0) */ - { size_t offsetFound = 99999999; - size_t const ml2 = searchMax(ctx, ip, iend, &offsetFound, maxSearches, mls); - if (ml2 > matchLength) - matchLength = ml2, start = ip, offset=offsetFound; - } + /* first search (depth 0) */ + { size_t offsetFound = 99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offsetFound, maxSearches, mls); + if (ml2 > matchLength) + matchLength = ml2, start = ip, offset=offsetFound; + } - if (matchLength < EQUAL_READ32) { - ip += ((ip-anchor) >> g_searchStrength) + 1; /* jump faster over incompressible sections */ - continue; - } + if (matchLength < EQUAL_READ32) { + ip += ((ip-anchor) >> g_searchStrength) + 1; /* jump faster over incompressible sections */ + continue; + } - /* let's try to find a better solution */ - if (depth>=1) - while (ip= 3) & (repIndex > lowestIndex)) /* intentional overflow */ - if (MEM_read32(ip) == MEM_read32(repMatch)) { - /* repcode detected */ - const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; - size_t const repLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; - int const gain2 = (int)(repLength * 3); - int const gain1 = (int)(matchLength*3 - ZSTD_highbit32((U32)offset+1) + 1); - if ((repLength >= EQUAL_READ32) && (gain2 > gain1)) - matchLength = repLength, offset = 0, start = ip; - } } + /* let's try to find a better solution */ + if (depth>=1) + while (ip= 3) & (repIndex > lowestIndex)) /* intentional overflow */ + if (MEM_read32(ip) == MEM_read32(repMatch)) { + /* repcode detected */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + size_t const repLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; + int const gain2 = (int)(repLength * 3); + int const gain1 = (int)(matchLength*3 - ZSTD_highbit32((U32)offset+1) + 1); + if ((repLength >= EQUAL_READ32) && (gain2 > gain1)) + matchLength = repLength, offset = 0, start = ip; + } } - /* search match, depth 1 */ - { size_t offset2=99999999; - size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); - int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ - int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 4); - if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { - matchLength = ml2, offset = offset2, start = ip; - continue; /* search a better one */ - } } + /* search match, depth 1 */ + { size_t offset2=99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); + int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ + int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 4); + if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { + matchLength = ml2, offset = offset2, start = ip; + continue; /* search a better one */ + } } - /* let's find an even better one */ - if ((depth==2) && (ip= 3) & (repIndex > lowestIndex)) /* intentional overflow */ - if (MEM_read32(ip) == MEM_read32(repMatch)) { - /* repcode detected */ - const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; - size_t repLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; - int gain2 = (int)(repLength * 4); - int gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 1); - if ((repLength >= EQUAL_READ32) && (gain2 > gain1)) - matchLength = repLength, offset = 0, start = ip; - } } + /* let's find an even better one */ + if ((depth==2) && (ip= 3) & (repIndex > lowestIndex)) /* intentional overflow */ + if (MEM_read32(ip) == MEM_read32(repMatch)) { + /* repcode detected */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + size_t repLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; + int gain2 = (int)(repLength * 4); + int gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 1); + if ((repLength >= EQUAL_READ32) && (gain2 > gain1)) + matchLength = repLength, offset = 0, start = ip; + } } - /* search match, depth 2 */ - { size_t offset2=99999999; - size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); - int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ - int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 7); - if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { - matchLength = ml2, offset = offset2, start = ip; - continue; - } } } - break; /* nothing found : store previous solution */ - } + /* search match, depth 2 */ + { size_t offset2=99999999; + size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls); + int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1)); /* raw approx */ + int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 7); + if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) { + matchLength = ml2, offset = offset2, start = ip; + continue; + } } } + break; /* nothing found : store previous solution */ + } - /* catch up */ - if (offset) { - U32 const matchIndex = (U32)((start-base) - (offset - ZSTD_REP_MOVE)); - const BYTE* match = (matchIndex < dictLimit) ? dictBase + matchIndex : base + matchIndex; - const BYTE* const mStart = (matchIndex < dictLimit) ? dictStart : prefixStart; - while ((start>anchor) && (match>mStart) && (start[-1] == match[-1])) { start--; match--; matchLength++; } /* catch up */ - offset_2 = offset_1; offset_1 = (U32)(offset - ZSTD_REP_MOVE); - } + /* catch up */ + if (offset) { + U32 const matchIndex = (U32)((start-base) - (offset - ZSTD_REP_MOVE)); + const BYTE* match = (matchIndex < dictLimit) ? dictBase + matchIndex : base + matchIndex; + const BYTE* const mStart = (matchIndex < dictLimit) ? dictStart : prefixStart; + while ((start>anchor) && (match>mStart) && (start[-1] == match[-1])) { start--; match--; matchLength++; } /* catch up */ + offset_2 = offset_1; offset_1 = (U32)(offset - ZSTD_REP_MOVE); + } - /* store sequence */ + /* store sequence */ _storeSequence: - { size_t const litLength = start - anchor; - ZSTD_storeSeq(seqStorePtr, litLength, anchor, (U32)offset, matchLength-MINMATCH); - anchor = ip = start + matchLength; - } + { size_t const litLength = start - anchor; + ZSTD_storeSeq(seqStorePtr, litLength, anchor, (U32)offset, matchLength-MINMATCH); + anchor = ip = start + matchLength; + } - /* check immediate repcode */ - while (ip <= ilimit) { - const U32 repIndex = (U32)((ip-base) - offset_2); - const BYTE* const repBase = repIndex < dictLimit ? dictBase : base; - const BYTE* const repMatch = repBase + repIndex; - if (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex > lowestIndex)) /* intentional overflow */ - if (MEM_read32(ip) == MEM_read32(repMatch)) { - /* repcode detected we should take it */ - const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; - matchLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; - offset = offset_2; offset_2 = offset_1; offset_1 = (U32)offset; /* swap offset history */ - ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, matchLength-MINMATCH); - ip += matchLength; - anchor = ip; - continue; /* faster when present ... (?) */ - } - break; - } } + /* check immediate repcode */ + while (ip <= ilimit) { + const U32 repIndex = (U32)((ip-base) - offset_2); + const BYTE* const repBase = repIndex < dictLimit ? dictBase : base; + const BYTE* const repMatch = repBase + repIndex; + if (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex > lowestIndex)) /* intentional overflow */ + if (MEM_read32(ip) == MEM_read32(repMatch)) { + /* repcode detected we should take it */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + matchLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32; + offset = offset_2; offset_2 = offset_1; offset_1 = (U32)offset; /* swap offset history */ + ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, matchLength-MINMATCH); + ip += matchLength; + anchor = ip; + continue; /* faster when present ... (?) */ + } + break; + } } - /* Save reps for next block */ - ctx->repToConfirm[0] = offset_1; ctx->repToConfirm[1] = offset_2; + /* Save reps for next block */ + ctx->repToConfirm[0] = offset_1; ctx->repToConfirm[1] = offset_2; - /* Last Literals */ - { size_t const lastLLSize = iend - anchor; - memcpy(seqStorePtr->lit, anchor, lastLLSize); - seqStorePtr->lit += lastLLSize; - } + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } } void ZSTD_compressBlock_greedy_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { - ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 0, 0); + ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 0, 0); } static void ZSTD_compressBlock_lazy_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { - ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 0, 1); + ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 0, 1); } static void ZSTD_compressBlock_lazy2_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { - ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 0, 2); + ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 0, 2); } static void ZSTD_compressBlock_btlazy2_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { - ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 1, 2); + ZSTD_compressBlock_lazy_extDict_generic(ctx, src, srcSize, 1, 2); } @@ -2254,40 +2254,40 @@ static void ZSTD_compressBlock_btlazy2_extDict(ZSTD_CCtx* ctx, const void* src, static void ZSTD_compressBlock_btopt(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { #ifdef ZSTD_OPT_H_91842398743 - ZSTD_compressBlock_opt_generic(ctx, src, srcSize, 0); + ZSTD_compressBlock_opt_generic(ctx, src, srcSize, 0); #else - (void)ctx; (void)src; (void)srcSize; - return; + (void)ctx; (void)src; (void)srcSize; + return; #endif } static void ZSTD_compressBlock_btopt2(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { #ifdef ZSTD_OPT_H_91842398743 - ZSTD_compressBlock_opt_generic(ctx, src, srcSize, 1); + ZSTD_compressBlock_opt_generic(ctx, src, srcSize, 1); #else - (void)ctx; (void)src; (void)srcSize; - return; + (void)ctx; (void)src; (void)srcSize; + return; #endif } static void ZSTD_compressBlock_btopt_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { #ifdef ZSTD_OPT_H_91842398743 - ZSTD_compressBlock_opt_extDict_generic(ctx, src, srcSize, 0); + ZSTD_compressBlock_opt_extDict_generic(ctx, src, srcSize, 0); #else - (void)ctx; (void)src; (void)srcSize; - return; + (void)ctx; (void)src; (void)srcSize; + return; #endif } static void ZSTD_compressBlock_btopt2_extDict(ZSTD_CCtx* ctx, const void* src, size_t srcSize) { #ifdef ZSTD_OPT_H_91842398743 - ZSTD_compressBlock_opt_extDict_generic(ctx, src, srcSize, 1); + ZSTD_compressBlock_opt_extDict_generic(ctx, src, srcSize, 1); #else - (void)ctx; (void)src; (void)srcSize; - return; + (void)ctx; (void)src; (void)srcSize; + return; #endif } @@ -2296,27 +2296,27 @@ typedef void (*ZSTD_blockCompressor) (ZSTD_CCtx* ctx, const void* src, size_t sr static ZSTD_blockCompressor ZSTD_selectBlockCompressor(ZSTD_strategy strat, int extDict) { - static const ZSTD_blockCompressor blockCompressor[2][8] = { - { ZSTD_compressBlock_fast, ZSTD_compressBlock_doubleFast, ZSTD_compressBlock_greedy, ZSTD_compressBlock_lazy, ZSTD_compressBlock_lazy2, ZSTD_compressBlock_btlazy2, ZSTD_compressBlock_btopt, ZSTD_compressBlock_btopt2 }, - { ZSTD_compressBlock_fast_extDict, ZSTD_compressBlock_doubleFast_extDict, ZSTD_compressBlock_greedy_extDict, ZSTD_compressBlock_lazy_extDict,ZSTD_compressBlock_lazy2_extDict, ZSTD_compressBlock_btlazy2_extDict, ZSTD_compressBlock_btopt_extDict, ZSTD_compressBlock_btopt2_extDict } - }; + static const ZSTD_blockCompressor blockCompressor[2][8] = { + { ZSTD_compressBlock_fast, ZSTD_compressBlock_doubleFast, ZSTD_compressBlock_greedy, ZSTD_compressBlock_lazy, ZSTD_compressBlock_lazy2, ZSTD_compressBlock_btlazy2, ZSTD_compressBlock_btopt, ZSTD_compressBlock_btopt2 }, + { ZSTD_compressBlock_fast_extDict, ZSTD_compressBlock_doubleFast_extDict, ZSTD_compressBlock_greedy_extDict, ZSTD_compressBlock_lazy_extDict,ZSTD_compressBlock_lazy2_extDict, ZSTD_compressBlock_btlazy2_extDict, ZSTD_compressBlock_btopt_extDict, ZSTD_compressBlock_btopt2_extDict } + }; - return blockCompressor[extDict][(U32)strat]; + return blockCompressor[extDict][(U32)strat]; } static size_t ZSTD_compressBlock_internal(ZSTD_CCtx* zc, void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - ZSTD_blockCompressor const blockCompressor = ZSTD_selectBlockCompressor(zc->params.cParams.strategy, zc->lowLimit < zc->dictLimit); - const BYTE* const base = zc->base; - const BYTE* const istart = (const BYTE*)src; - const U32 current = (U32)(istart-base); - if (srcSize < MIN_CBLOCK_SIZE+ZSTD_blockHeaderSize+1) return 0; /* don't even attempt compression below a certain srcSize */ - ZSTD_resetSeqStore(&(zc->seqStore)); - if (current > zc->nextToUpdate + 384) - zc->nextToUpdate = current - MIN(192, (U32)(current - zc->nextToUpdate - 384)); /* update tree not updated after finding very long rep matches */ - blockCompressor(zc, src, srcSize); - return ZSTD_compressSequences(zc, dst, dstCapacity, srcSize); + ZSTD_blockCompressor const blockCompressor = ZSTD_selectBlockCompressor(zc->params.cParams.strategy, zc->lowLimit < zc->dictLimit); + const BYTE* const base = zc->base; + const BYTE* const istart = (const BYTE*)src; + const U32 current = (U32)(istart-base); + if (srcSize < MIN_CBLOCK_SIZE+ZSTD_blockHeaderSize+1) return 0; /* don't even attempt compression below a certain srcSize */ + ZSTD_resetSeqStore(&(zc->seqStore)); + if (current > zc->nextToUpdate + 384) + zc->nextToUpdate = current - MIN(192, (U32)(current - zc->nextToUpdate - 384)); /* update tree not updated after finding very long rep matches */ + blockCompressor(zc, src, srcSize); + return ZSTD_compressSequences(zc, dst, dstCapacity, srcSize); } @@ -2328,183 +2328,183 @@ static size_t ZSTD_compressBlock_internal(ZSTD_CCtx* zc, void* dst, size_t dstCa * @return : compressed size, or an error code */ static size_t ZSTD_compress_generic (ZSTD_CCtx* cctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - U32 lastFrameChunk) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + U32 lastFrameChunk) { - size_t blockSize = cctx->blockSize; - size_t remaining = srcSize; - const BYTE* ip = (const BYTE*)src; - BYTE* const ostart = (BYTE*)dst; - BYTE* op = ostart; - U32 const maxDist = 1 << cctx->params.cParams.windowLog; + size_t blockSize = cctx->blockSize; + size_t remaining = srcSize; + const BYTE* ip = (const BYTE*)src; + BYTE* const ostart = (BYTE*)dst; + BYTE* op = ostart; + U32 const maxDist = 1 << cctx->params.cParams.windowLog; - if (cctx->params.fParams.checksumFlag && srcSize) - XXH64_update(&cctx->xxhState, src, srcSize); + if (cctx->params.fParams.checksumFlag && srcSize) + XXH64_update(&cctx->xxhState, src, srcSize); - while (remaining) { - U32 const lastBlock = lastFrameChunk & (blockSize >= remaining); - size_t cSize; + while (remaining) { + U32 const lastBlock = lastFrameChunk & (blockSize >= remaining); + size_t cSize; - if (dstCapacity < ZSTD_blockHeaderSize + MIN_CBLOCK_SIZE) return ERROR(dstSize_tooSmall); /* not enough space to store compressed block */ - if (remaining < blockSize) blockSize = remaining; + if (dstCapacity < ZSTD_blockHeaderSize + MIN_CBLOCK_SIZE) return ERROR(dstSize_tooSmall); /* not enough space to store compressed block */ + if (remaining < blockSize) blockSize = remaining; - /* preemptive overflow correction */ - if (cctx->lowLimit > (3U<<29)) { - U32 const cycleMask = (1 << ZSTD_cycleLog(cctx->params.cParams.hashLog, cctx->params.cParams.strategy)) - 1; - U32 const current = (U32)(ip - cctx->base); - U32 const newCurrent = (current & cycleMask) + (1 << cctx->params.cParams.windowLog); - U32 const correction = current - newCurrent; - ZSTD_STATIC_ASSERT(ZSTD_WINDOWLOG_MAX_64 <= 30); - ZSTD_reduceIndex(cctx, correction); - cctx->base += correction; - cctx->dictBase += correction; - cctx->lowLimit -= correction; - cctx->dictLimit -= correction; - if (cctx->nextToUpdate < correction) cctx->nextToUpdate = 0; - else cctx->nextToUpdate -= correction; - } + /* preemptive overflow correction */ + if (cctx->lowLimit > (3U<<29)) { + U32 const cycleMask = (1 << ZSTD_cycleLog(cctx->params.cParams.hashLog, cctx->params.cParams.strategy)) - 1; + U32 const current = (U32)(ip - cctx->base); + U32 const newCurrent = (current & cycleMask) + (1 << cctx->params.cParams.windowLog); + U32 const correction = current - newCurrent; + ZSTD_STATIC_ASSERT(ZSTD_WINDOWLOG_MAX_64 <= 30); + ZSTD_reduceIndex(cctx, correction); + cctx->base += correction; + cctx->dictBase += correction; + cctx->lowLimit -= correction; + cctx->dictLimit -= correction; + if (cctx->nextToUpdate < correction) cctx->nextToUpdate = 0; + else cctx->nextToUpdate -= correction; + } - if ((U32)(ip+blockSize - cctx->base) > cctx->loadedDictEnd + maxDist) { - /* enforce maxDist */ - U32 const newLowLimit = (U32)(ip+blockSize - cctx->base) - maxDist; - if (cctx->lowLimit < newLowLimit) cctx->lowLimit = newLowLimit; - if (cctx->dictLimit < cctx->lowLimit) cctx->dictLimit = cctx->lowLimit; - } + if ((U32)(ip+blockSize - cctx->base) > cctx->loadedDictEnd + maxDist) { + /* enforce maxDist */ + U32 const newLowLimit = (U32)(ip+blockSize - cctx->base) - maxDist; + if (cctx->lowLimit < newLowLimit) cctx->lowLimit = newLowLimit; + if (cctx->dictLimit < cctx->lowLimit) cctx->dictLimit = cctx->lowLimit; + } - cSize = ZSTD_compressBlock_internal(cctx, op+ZSTD_blockHeaderSize, dstCapacity-ZSTD_blockHeaderSize, ip, blockSize); - if (ZSTD_isError(cSize)) return cSize; + cSize = ZSTD_compressBlock_internal(cctx, op+ZSTD_blockHeaderSize, dstCapacity-ZSTD_blockHeaderSize, ip, blockSize); + if (ZSTD_isError(cSize)) return cSize; - if (cSize == 0) { /* block is not compressible */ - U32 const cBlockHeader24 = lastBlock + (((U32)bt_raw)<<1) + (U32)(blockSize << 3); - if (blockSize + ZSTD_blockHeaderSize > dstCapacity) return ERROR(dstSize_tooSmall); - MEM_writeLE32(op, cBlockHeader24); /* no pb, 4th byte will be overwritten */ - memcpy(op + ZSTD_blockHeaderSize, ip, blockSize); - cSize = ZSTD_blockHeaderSize+blockSize; - } else { - U32 const cBlockHeader24 = lastBlock + (((U32)bt_compressed)<<1) + (U32)(cSize << 3); - MEM_writeLE24(op, cBlockHeader24); - cSize += ZSTD_blockHeaderSize; - } + if (cSize == 0) { /* block is not compressible */ + U32 const cBlockHeader24 = lastBlock + (((U32)bt_raw)<<1) + (U32)(blockSize << 3); + if (blockSize + ZSTD_blockHeaderSize > dstCapacity) return ERROR(dstSize_tooSmall); + MEM_writeLE32(op, cBlockHeader24); /* no pb, 4th byte will be overwritten */ + memcpy(op + ZSTD_blockHeaderSize, ip, blockSize); + cSize = ZSTD_blockHeaderSize+blockSize; + } else { + U32 const cBlockHeader24 = lastBlock + (((U32)bt_compressed)<<1) + (U32)(cSize << 3); + MEM_writeLE24(op, cBlockHeader24); + cSize += ZSTD_blockHeaderSize; + } - remaining -= blockSize; - dstCapacity -= cSize; - ip += blockSize; - op += cSize; - } + remaining -= blockSize; + dstCapacity -= cSize; + ip += blockSize; + op += cSize; + } - if (lastFrameChunk && (op>ostart)) cctx->stage = ZSTDcs_ending; - return op-ostart; + if (lastFrameChunk && (op>ostart)) cctx->stage = ZSTDcs_ending; + return op-ostart; } static size_t ZSTD_writeFrameHeader(void* dst, size_t dstCapacity, - ZSTD_parameters params, U64 pledgedSrcSize, U32 dictID) + ZSTD_parameters params, U64 pledgedSrcSize, U32 dictID) { BYTE* const op = (BYTE*)dst; - U32 const dictIDSizeCode = (dictID>0) + (dictID>=256) + (dictID>=65536); /* 0-3 */ - U32 const checksumFlag = params.fParams.checksumFlag>0; - U32 const windowSize = 1U << params.cParams.windowLog; - U32 const singleSegment = params.fParams.contentSizeFlag && (windowSize >= pledgedSrcSize); - BYTE const windowLogByte = (BYTE)((params.cParams.windowLog - ZSTD_WINDOWLOG_ABSOLUTEMIN) << 3); - U32 const fcsCode = params.fParams.contentSizeFlag ? - (pledgedSrcSize>=256) + (pledgedSrcSize>=65536+256) + (pledgedSrcSize>=0xFFFFFFFFU) : /* 0-3 */ - 0; - BYTE const frameHeaderDecriptionByte = (BYTE)(dictIDSizeCode + (checksumFlag<<2) + (singleSegment<<5) + (fcsCode<<6) ); - size_t pos; + U32 const dictIDSizeCode = (dictID>0) + (dictID>=256) + (dictID>=65536); /* 0-3 */ + U32 const checksumFlag = params.fParams.checksumFlag>0; + U32 const windowSize = 1U << params.cParams.windowLog; + U32 const singleSegment = params.fParams.contentSizeFlag && (windowSize >= pledgedSrcSize); + BYTE const windowLogByte = (BYTE)((params.cParams.windowLog - ZSTD_WINDOWLOG_ABSOLUTEMIN) << 3); + U32 const fcsCode = params.fParams.contentSizeFlag ? + (pledgedSrcSize>=256) + (pledgedSrcSize>=65536+256) + (pledgedSrcSize>=0xFFFFFFFFU) : /* 0-3 */ + 0; + BYTE const frameHeaderDecriptionByte = (BYTE)(dictIDSizeCode + (checksumFlag<<2) + (singleSegment<<5) + (fcsCode<<6) ); + size_t pos; - if (dstCapacity < ZSTD_frameHeaderSize_max) return ERROR(dstSize_tooSmall); + if (dstCapacity < ZSTD_frameHeaderSize_max) return ERROR(dstSize_tooSmall); - MEM_writeLE32(dst, ZSTD_MAGICNUMBER); - op[4] = frameHeaderDecriptionByte; pos=5; - if (!singleSegment) op[pos++] = windowLogByte; - switch(dictIDSizeCode) - { - default: /* impossible */ - case 0 : break; - case 1 : op[pos] = (BYTE)(dictID); pos++; break; - case 2 : MEM_writeLE16(op+pos, (U16)dictID); pos+=2; break; - case 3 : MEM_writeLE32(op+pos, dictID); pos+=4; break; - } - switch(fcsCode) - { - default: /* impossible */ - case 0 : if (singleSegment) op[pos++] = (BYTE)(pledgedSrcSize); break; - case 1 : MEM_writeLE16(op+pos, (U16)(pledgedSrcSize-256)); pos+=2; break; - case 2 : MEM_writeLE32(op+pos, (U32)(pledgedSrcSize)); pos+=4; break; - case 3 : MEM_writeLE64(op+pos, (U64)(pledgedSrcSize)); pos+=8; break; - } - return pos; + MEM_writeLE32(dst, ZSTD_MAGICNUMBER); + op[4] = frameHeaderDecriptionByte; pos=5; + if (!singleSegment) op[pos++] = windowLogByte; + switch(dictIDSizeCode) + { + default: /* impossible */ + case 0 : break; + case 1 : op[pos] = (BYTE)(dictID); pos++; break; + case 2 : MEM_writeLE16(op+pos, (U16)dictID); pos+=2; break; + case 3 : MEM_writeLE32(op+pos, dictID); pos+=4; break; + } + switch(fcsCode) + { + default: /* impossible */ + case 0 : if (singleSegment) op[pos++] = (BYTE)(pledgedSrcSize); break; + case 1 : MEM_writeLE16(op+pos, (U16)(pledgedSrcSize-256)); pos+=2; break; + case 2 : MEM_writeLE32(op+pos, (U32)(pledgedSrcSize)); pos+=4; break; + case 3 : MEM_writeLE64(op+pos, (U64)(pledgedSrcSize)); pos+=8; break; + } + return pos; } static size_t ZSTD_compressContinue_internal (ZSTD_CCtx* cctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - U32 frame, U32 lastFrameChunk) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + U32 frame, U32 lastFrameChunk) { - const BYTE* const ip = (const BYTE*) src; - size_t fhSize = 0; + const BYTE* const ip = (const BYTE*) src; + size_t fhSize = 0; - if (cctx->stage==ZSTDcs_created) return ERROR(stage_wrong); /* missing init (ZSTD_compressBegin) */ + if (cctx->stage==ZSTDcs_created) return ERROR(stage_wrong); /* missing init (ZSTD_compressBegin) */ - if (frame && (cctx->stage==ZSTDcs_init)) { - fhSize = ZSTD_writeFrameHeader(dst, dstCapacity, cctx->params, cctx->frameContentSize, cctx->dictID); - if (ZSTD_isError(fhSize)) return fhSize; - dstCapacity -= fhSize; - dst = (char*)dst + fhSize; - cctx->stage = ZSTDcs_ongoing; - } + if (frame && (cctx->stage==ZSTDcs_init)) { + fhSize = ZSTD_writeFrameHeader(dst, dstCapacity, cctx->params, cctx->frameContentSize, cctx->dictID); + if (ZSTD_isError(fhSize)) return fhSize; + dstCapacity -= fhSize; + dst = (char*)dst + fhSize; + cctx->stage = ZSTDcs_ongoing; + } - /* Check if blocks follow each other */ - if (src != cctx->nextSrc) { - /* not contiguous */ - ptrdiff_t const delta = cctx->nextSrc - ip; - cctx->lowLimit = cctx->dictLimit; - cctx->dictLimit = (U32)(cctx->nextSrc - cctx->base); - cctx->dictBase = cctx->base; - cctx->base -= delta; - cctx->nextToUpdate = cctx->dictLimit; - if (cctx->dictLimit - cctx->lowLimit < HASH_READ_SIZE) cctx->lowLimit = cctx->dictLimit; /* too small extDict */ - } + /* Check if blocks follow each other */ + if (src != cctx->nextSrc) { + /* not contiguous */ + ptrdiff_t const delta = cctx->nextSrc - ip; + cctx->lowLimit = cctx->dictLimit; + cctx->dictLimit = (U32)(cctx->nextSrc - cctx->base); + cctx->dictBase = cctx->base; + cctx->base -= delta; + cctx->nextToUpdate = cctx->dictLimit; + if (cctx->dictLimit - cctx->lowLimit < HASH_READ_SIZE) cctx->lowLimit = cctx->dictLimit; /* too small extDict */ + } - /* if input and dictionary overlap : reduce dictionary (area presumed modified by input) */ - if ((ip+srcSize > cctx->dictBase + cctx->lowLimit) & (ip < cctx->dictBase + cctx->dictLimit)) { - ptrdiff_t const highInputIdx = (ip + srcSize) - cctx->dictBase; - U32 const lowLimitMax = (highInputIdx > (ptrdiff_t)cctx->dictLimit) ? cctx->dictLimit : (U32)highInputIdx; - cctx->lowLimit = lowLimitMax; - } + /* if input and dictionary overlap : reduce dictionary (area presumed modified by input) */ + if ((ip+srcSize > cctx->dictBase + cctx->lowLimit) & (ip < cctx->dictBase + cctx->dictLimit)) { + ptrdiff_t const highInputIdx = (ip + srcSize) - cctx->dictBase; + U32 const lowLimitMax = (highInputIdx > (ptrdiff_t)cctx->dictLimit) ? cctx->dictLimit : (U32)highInputIdx; + cctx->lowLimit = lowLimitMax; + } - cctx->nextSrc = ip + srcSize; + cctx->nextSrc = ip + srcSize; - if (srcSize) { - size_t const cSize = frame ? - ZSTD_compress_generic (cctx, dst, dstCapacity, src, srcSize, lastFrameChunk) : - ZSTD_compressBlock_internal (cctx, dst, dstCapacity, src, srcSize); - if (ZSTD_isError(cSize)) return cSize; - return cSize + fhSize; - } else - return fhSize; + if (srcSize) { + size_t const cSize = frame ? + ZSTD_compress_generic (cctx, dst, dstCapacity, src, srcSize, lastFrameChunk) : + ZSTD_compressBlock_internal (cctx, dst, dstCapacity, src, srcSize); + if (ZSTD_isError(cSize)) return cSize; + return cSize + fhSize; + } else + return fhSize; } size_t ZSTD_compressContinue (ZSTD_CCtx* cctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize) { - return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1, 0); + return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1, 0); } size_t ZSTD_getBlockSizeMax(ZSTD_CCtx* cctx) { - return MIN (ZSTD_BLOCKSIZE_ABSOLUTEMAX, 1 << cctx->params.cParams.windowLog); + return MIN (ZSTD_BLOCKSIZE_ABSOLUTEMAX, 1 << cctx->params.cParams.windowLog); } size_t ZSTD_compressBlock(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - size_t const blockSizeMax = ZSTD_getBlockSizeMax(cctx); - if (srcSize > blockSizeMax) return ERROR(srcSize_wrong); - return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 0, 0); + size_t const blockSizeMax = ZSTD_getBlockSizeMax(cctx); + if (srcSize > blockSizeMax) return ERROR(srcSize_wrong); + return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 0, 0); } /*! ZSTD_loadDictionaryContent() : @@ -2512,50 +2512,50 @@ size_t ZSTD_compressBlock(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const */ static size_t ZSTD_loadDictionaryContent(ZSTD_CCtx* zc, const void* src, size_t srcSize) { - const BYTE* const ip = (const BYTE*) src; - const BYTE* const iend = ip + srcSize; + const BYTE* const ip = (const BYTE*) src; + const BYTE* const iend = ip + srcSize; - /* input becomes current prefix */ - zc->lowLimit = zc->dictLimit; - zc->dictLimit = (U32)(zc->nextSrc - zc->base); - zc->dictBase = zc->base; - zc->base += ip - zc->nextSrc; - zc->nextToUpdate = zc->dictLimit; - zc->loadedDictEnd = zc->forceWindow ? 0 : (U32)(iend - zc->base); + /* input becomes current prefix */ + zc->lowLimit = zc->dictLimit; + zc->dictLimit = (U32)(zc->nextSrc - zc->base); + zc->dictBase = zc->base; + zc->base += ip - zc->nextSrc; + zc->nextToUpdate = zc->dictLimit; + zc->loadedDictEnd = zc->forceWindow ? 0 : (U32)(iend - zc->base); - zc->nextSrc = iend; - if (srcSize <= HASH_READ_SIZE) return 0; + zc->nextSrc = iend; + if (srcSize <= HASH_READ_SIZE) return 0; - switch(zc->params.cParams.strategy) - { - case ZSTD_fast: - ZSTD_fillHashTable (zc, iend, zc->params.cParams.searchLength); - break; + switch(zc->params.cParams.strategy) + { + case ZSTD_fast: + ZSTD_fillHashTable (zc, iend, zc->params.cParams.searchLength); + break; - case ZSTD_dfast: - ZSTD_fillDoubleHashTable (zc, iend, zc->params.cParams.searchLength); - break; + case ZSTD_dfast: + ZSTD_fillDoubleHashTable (zc, iend, zc->params.cParams.searchLength); + break; - case ZSTD_greedy: - case ZSTD_lazy: - case ZSTD_lazy2: - if (srcSize >= HASH_READ_SIZE) - ZSTD_insertAndFindFirstIndex(zc, iend-HASH_READ_SIZE, zc->params.cParams.searchLength); - break; + case ZSTD_greedy: + case ZSTD_lazy: + case ZSTD_lazy2: + if (srcSize >= HASH_READ_SIZE) + ZSTD_insertAndFindFirstIndex(zc, iend-HASH_READ_SIZE, zc->params.cParams.searchLength); + break; - case ZSTD_btlazy2: - case ZSTD_btopt: - case ZSTD_btopt2: - if (srcSize >= HASH_READ_SIZE) - ZSTD_updateTree(zc, iend-HASH_READ_SIZE, iend, 1 << zc->params.cParams.searchLog, zc->params.cParams.searchLength); - break; + case ZSTD_btlazy2: + case ZSTD_btopt: + case ZSTD_btopt2: + if (srcSize >= HASH_READ_SIZE) + ZSTD_updateTree(zc, iend-HASH_READ_SIZE, iend, 1 << zc->params.cParams.searchLog, zc->params.cParams.searchLength); + break; - default: - return ERROR(GENERIC); /* strategy doesn't exist; impossible */ - } + default: + return ERROR(GENERIC); /* strategy doesn't exist; impossible */ + } - zc->nextToUpdate = (U32)(iend - zc->base); - return 0; + zc->nextToUpdate = (U32)(iend - zc->base); + return 0; } @@ -2564,12 +2564,12 @@ static size_t ZSTD_loadDictionaryContent(ZSTD_CCtx* zc, const void* src, size_t that we may encounter during compression. NOTE: This behavior is not standard and could be improved in the future. */ static size_t ZSTD_checkDictNCount(short* normalizedCounter, unsigned dictMaxSymbolValue, unsigned maxSymbolValue) { - U32 s; - if (dictMaxSymbolValue < maxSymbolValue) return ERROR(dictionary_corrupted); - for (s = 0; s <= maxSymbolValue; ++s) { - if (normalizedCounter[s] == 0) return ERROR(dictionary_corrupted); - } - return 0; + U32 s; + if (dictMaxSymbolValue < maxSymbolValue) return ERROR(dictionary_corrupted); + for (s = 0; s <= maxSymbolValue; ++s) { + if (normalizedCounter[s] == 0) return ERROR(dictionary_corrupted); + } + return 0; } @@ -2584,127 +2584,127 @@ static size_t ZSTD_checkDictNCount(short* normalizedCounter, unsigned dictMaxSym */ static size_t ZSTD_loadZstdDictionary(ZSTD_CCtx* cctx, const void* dict, size_t dictSize) { - const BYTE* dictPtr = (const BYTE*)dict; - const BYTE* const dictEnd = dictPtr + dictSize; - short offcodeNCount[MaxOff+1]; - unsigned offcodeMaxValue = MaxOff; - BYTE scratchBuffer[1<dictID = cctx->params.fParams.noDictIDFlag ? 0 : MEM_readLE32(dictPtr); - dictPtr += 4; + dictPtr += 4; /* skip magic number */ + cctx->dictID = cctx->params.fParams.noDictIDFlag ? 0 : MEM_readLE32(dictPtr); + dictPtr += 4; - { size_t const hufHeaderSize = HUF_readCTable(cctx->hufTable, 255, dictPtr, dictEnd-dictPtr); - if (HUF_isError(hufHeaderSize)) return ERROR(dictionary_corrupted); - dictPtr += hufHeaderSize; - } + { size_t const hufHeaderSize = HUF_readCTable(cctx->hufTable, 255, dictPtr, dictEnd-dictPtr); + if (HUF_isError(hufHeaderSize)) return ERROR(dictionary_corrupted); + dictPtr += hufHeaderSize; + } - { unsigned offcodeLog; - size_t const offcodeHeaderSize = FSE_readNCount(offcodeNCount, &offcodeMaxValue, &offcodeLog, dictPtr, dictEnd-dictPtr); - if (FSE_isError(offcodeHeaderSize)) return ERROR(dictionary_corrupted); - if (offcodeLog > OffFSELog) return ERROR(dictionary_corrupted); - /* Defer checking offcodeMaxValue because we need to know the size of the dictionary content */ - CHECK_E (FSE_buildCTable_wksp(cctx->offcodeCTable, offcodeNCount, offcodeMaxValue, offcodeLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted); - dictPtr += offcodeHeaderSize; - } + { unsigned offcodeLog; + size_t const offcodeHeaderSize = FSE_readNCount(offcodeNCount, &offcodeMaxValue, &offcodeLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(offcodeHeaderSize)) return ERROR(dictionary_corrupted); + if (offcodeLog > OffFSELog) return ERROR(dictionary_corrupted); + /* Defer checking offcodeMaxValue because we need to know the size of the dictionary content */ + CHECK_E (FSE_buildCTable_wksp(cctx->offcodeCTable, offcodeNCount, offcodeMaxValue, offcodeLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted); + dictPtr += offcodeHeaderSize; + } - { short matchlengthNCount[MaxML+1]; - unsigned matchlengthMaxValue = MaxML, matchlengthLog; - size_t const matchlengthHeaderSize = FSE_readNCount(matchlengthNCount, &matchlengthMaxValue, &matchlengthLog, dictPtr, dictEnd-dictPtr); - if (FSE_isError(matchlengthHeaderSize)) return ERROR(dictionary_corrupted); - if (matchlengthLog > MLFSELog) return ERROR(dictionary_corrupted); - /* Every match length code must have non-zero probability */ - CHECK_F (ZSTD_checkDictNCount(matchlengthNCount, matchlengthMaxValue, MaxML)); - CHECK_E (FSE_buildCTable_wksp(cctx->matchlengthCTable, matchlengthNCount, matchlengthMaxValue, matchlengthLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted); - dictPtr += matchlengthHeaderSize; - } + { short matchlengthNCount[MaxML+1]; + unsigned matchlengthMaxValue = MaxML, matchlengthLog; + size_t const matchlengthHeaderSize = FSE_readNCount(matchlengthNCount, &matchlengthMaxValue, &matchlengthLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(matchlengthHeaderSize)) return ERROR(dictionary_corrupted); + if (matchlengthLog > MLFSELog) return ERROR(dictionary_corrupted); + /* Every match length code must have non-zero probability */ + CHECK_F (ZSTD_checkDictNCount(matchlengthNCount, matchlengthMaxValue, MaxML)); + CHECK_E (FSE_buildCTable_wksp(cctx->matchlengthCTable, matchlengthNCount, matchlengthMaxValue, matchlengthLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted); + dictPtr += matchlengthHeaderSize; + } - { short litlengthNCount[MaxLL+1]; - unsigned litlengthMaxValue = MaxLL, litlengthLog; - size_t const litlengthHeaderSize = FSE_readNCount(litlengthNCount, &litlengthMaxValue, &litlengthLog, dictPtr, dictEnd-dictPtr); - if (FSE_isError(litlengthHeaderSize)) return ERROR(dictionary_corrupted); - if (litlengthLog > LLFSELog) return ERROR(dictionary_corrupted); - /* Every literal length code must have non-zero probability */ - CHECK_F (ZSTD_checkDictNCount(litlengthNCount, litlengthMaxValue, MaxLL)); - CHECK_E(FSE_buildCTable_wksp(cctx->litlengthCTable, litlengthNCount, litlengthMaxValue, litlengthLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted); - dictPtr += litlengthHeaderSize; - } + { short litlengthNCount[MaxLL+1]; + unsigned litlengthMaxValue = MaxLL, litlengthLog; + size_t const litlengthHeaderSize = FSE_readNCount(litlengthNCount, &litlengthMaxValue, &litlengthLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(litlengthHeaderSize)) return ERROR(dictionary_corrupted); + if (litlengthLog > LLFSELog) return ERROR(dictionary_corrupted); + /* Every literal length code must have non-zero probability */ + CHECK_F (ZSTD_checkDictNCount(litlengthNCount, litlengthMaxValue, MaxLL)); + CHECK_E(FSE_buildCTable_wksp(cctx->litlengthCTable, litlengthNCount, litlengthMaxValue, litlengthLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted); + dictPtr += litlengthHeaderSize; + } - if (dictPtr+12 > dictEnd) return ERROR(dictionary_corrupted); - cctx->rep[0] = MEM_readLE32(dictPtr+0); - cctx->rep[1] = MEM_readLE32(dictPtr+4); - cctx->rep[2] = MEM_readLE32(dictPtr+8); - dictPtr += 12; + if (dictPtr+12 > dictEnd) return ERROR(dictionary_corrupted); + cctx->rep[0] = MEM_readLE32(dictPtr+0); + cctx->rep[1] = MEM_readLE32(dictPtr+4); + cctx->rep[2] = MEM_readLE32(dictPtr+8); + dictPtr += 12; - { size_t const dictContentSize = (size_t)(dictEnd - dictPtr); - U32 offcodeMax = MaxOff; - if (dictContentSize <= ((U32)-1) - 128 KB) { - U32 const maxOffset = (U32)dictContentSize + 128 KB; /* The maximum offset that must be supported */ - offcodeMax = ZSTD_highbit32(maxOffset); /* Calculate minimum offset code required to represent maxOffset */ - } - /* All offset values <= dictContentSize + 128 KB must be representable */ - CHECK_F (ZSTD_checkDictNCount(offcodeNCount, offcodeMaxValue, MIN(offcodeMax, MaxOff))); - /* All repCodes must be <= dictContentSize and != 0*/ - { U32 u; - for (u=0; u<3; u++) { - if (cctx->rep[u] == 0) return ERROR(dictionary_corrupted); - if (cctx->rep[u] > dictContentSize) return ERROR(dictionary_corrupted); - } } + { size_t const dictContentSize = (size_t)(dictEnd - dictPtr); + U32 offcodeMax = MaxOff; + if (dictContentSize <= ((U32)-1) - 128 KB) { + U32 const maxOffset = (U32)dictContentSize + 128 KB; /* The maximum offset that must be supported */ + offcodeMax = ZSTD_highbit32(maxOffset); /* Calculate minimum offset code required to represent maxOffset */ + } + /* All offset values <= dictContentSize + 128 KB must be representable */ + CHECK_F (ZSTD_checkDictNCount(offcodeNCount, offcodeMaxValue, MIN(offcodeMax, MaxOff))); + /* All repCodes must be <= dictContentSize and != 0*/ + { U32 u; + for (u=0; u<3; u++) { + if (cctx->rep[u] == 0) return ERROR(dictionary_corrupted); + if (cctx->rep[u] > dictContentSize) return ERROR(dictionary_corrupted); + } } - cctx->flagStaticTables = 1; - cctx->flagStaticHufTable = HUF_repeat_valid; - return ZSTD_loadDictionaryContent(cctx, dictPtr, dictContentSize); - } + cctx->flagStaticTables = 1; + cctx->flagStaticHufTable = HUF_repeat_valid; + return ZSTD_loadDictionaryContent(cctx, dictPtr, dictContentSize); + } } /** ZSTD_compress_insertDictionary() : * @return : 0, or an error code */ static size_t ZSTD_compress_insertDictionary(ZSTD_CCtx* cctx, const void* dict, size_t dictSize) { - if ((dict==NULL) || (dictSize<=8)) return 0; + if ((dict==NULL) || (dictSize<=8)) return 0; - /* dict as pure content */ - if ((MEM_readLE32(dict) != ZSTD_DICT_MAGIC) || (cctx->forceRawDict)) - return ZSTD_loadDictionaryContent(cctx, dict, dictSize); + /* dict as pure content */ + if ((MEM_readLE32(dict) != ZSTD_DICT_MAGIC) || (cctx->forceRawDict)) + return ZSTD_loadDictionaryContent(cctx, dict, dictSize); - /* dict as zstd dictionary */ - return ZSTD_loadZstdDictionary(cctx, dict, dictSize); + /* dict as zstd dictionary */ + return ZSTD_loadZstdDictionary(cctx, dict, dictSize); } /*! ZSTD_compressBegin_internal() : * @return : 0, or an error code */ static size_t ZSTD_compressBegin_internal(ZSTD_CCtx* cctx, - const void* dict, size_t dictSize, - ZSTD_parameters params, U64 pledgedSrcSize) + const void* dict, size_t dictSize, + ZSTD_parameters params, U64 pledgedSrcSize) { - ZSTD_compResetPolicy_e const crp = dictSize ? ZSTDcrp_fullReset : ZSTDcrp_continue; - CHECK_F(ZSTD_resetCCtx_advanced(cctx, params, pledgedSrcSize, crp)); - return ZSTD_compress_insertDictionary(cctx, dict, dictSize); + ZSTD_compResetPolicy_e const crp = dictSize ? ZSTDcrp_fullReset : ZSTDcrp_continue; + CHECK_F(ZSTD_resetCCtx_advanced(cctx, params, pledgedSrcSize, crp)); + return ZSTD_compress_insertDictionary(cctx, dict, dictSize); } /*! ZSTD_compressBegin_advanced() : * @return : 0, or an error code */ size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, - const void* dict, size_t dictSize, - ZSTD_parameters params, unsigned long long pledgedSrcSize) + const void* dict, size_t dictSize, + ZSTD_parameters params, unsigned long long pledgedSrcSize) { - /* compression parameters verification and optimization */ - CHECK_F(ZSTD_checkCParams(params.cParams)); - return ZSTD_compressBegin_internal(cctx, dict, dictSize, params, pledgedSrcSize); + /* compression parameters verification and optimization */ + CHECK_F(ZSTD_checkCParams(params.cParams)); + return ZSTD_compressBegin_internal(cctx, dict, dictSize, params, pledgedSrcSize); } size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel) { - ZSTD_parameters const params = ZSTD_getParams(compressionLevel, 0, dictSize); - return ZSTD_compressBegin_internal(cctx, dict, dictSize, params, 0); + ZSTD_parameters const params = ZSTD_getParams(compressionLevel, 0, dictSize); + return ZSTD_compressBegin_internal(cctx, dict, dictSize, params, 0); } size_t ZSTD_compressBegin(ZSTD_CCtx* cctx, int compressionLevel) { - return ZSTD_compressBegin_usingDict(cctx, NULL, 0, compressionLevel); + return ZSTD_compressBegin_usingDict(cctx, NULL, 0, compressionLevel); } @@ -2713,194 +2713,194 @@ size_t ZSTD_compressBegin(ZSTD_CCtx* cctx, int compressionLevel) * @return : nb of bytes written into dst (or an error code) */ static size_t ZSTD_writeEpilogue(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity) { - BYTE* const ostart = (BYTE*)dst; - BYTE* op = ostart; - size_t fhSize = 0; + BYTE* const ostart = (BYTE*)dst; + BYTE* op = ostart; + size_t fhSize = 0; - if (cctx->stage == ZSTDcs_created) return ERROR(stage_wrong); /* init missing */ + if (cctx->stage == ZSTDcs_created) return ERROR(stage_wrong); /* init missing */ - /* special case : empty frame */ - if (cctx->stage == ZSTDcs_init) { - fhSize = ZSTD_writeFrameHeader(dst, dstCapacity, cctx->params, 0, 0); - if (ZSTD_isError(fhSize)) return fhSize; - dstCapacity -= fhSize; - op += fhSize; - cctx->stage = ZSTDcs_ongoing; - } + /* special case : empty frame */ + if (cctx->stage == ZSTDcs_init) { + fhSize = ZSTD_writeFrameHeader(dst, dstCapacity, cctx->params, 0, 0); + if (ZSTD_isError(fhSize)) return fhSize; + dstCapacity -= fhSize; + op += fhSize; + cctx->stage = ZSTDcs_ongoing; + } - if (cctx->stage != ZSTDcs_ending) { - /* write one last empty block, make it the "last" block */ - U32 const cBlockHeader24 = 1 /* last block */ + (((U32)bt_raw)<<1) + 0; - if (dstCapacity<4) return ERROR(dstSize_tooSmall); - MEM_writeLE32(op, cBlockHeader24); - op += ZSTD_blockHeaderSize; - dstCapacity -= ZSTD_blockHeaderSize; - } + if (cctx->stage != ZSTDcs_ending) { + /* write one last empty block, make it the "last" block */ + U32 const cBlockHeader24 = 1 /* last block */ + (((U32)bt_raw)<<1) + 0; + if (dstCapacity<4) return ERROR(dstSize_tooSmall); + MEM_writeLE32(op, cBlockHeader24); + op += ZSTD_blockHeaderSize; + dstCapacity -= ZSTD_blockHeaderSize; + } - if (cctx->params.fParams.checksumFlag) { - U32 const checksum = (U32) XXH64_digest(&cctx->xxhState); - if (dstCapacity<4) return ERROR(dstSize_tooSmall); - MEM_writeLE32(op, checksum); - op += 4; - } + if (cctx->params.fParams.checksumFlag) { + U32 const checksum = (U32) XXH64_digest(&cctx->xxhState); + if (dstCapacity<4) return ERROR(dstSize_tooSmall); + MEM_writeLE32(op, checksum); + op += 4; + } - cctx->stage = ZSTDcs_created; /* return to "created but no init" status */ - return op-ostart; + cctx->stage = ZSTDcs_created; /* return to "created but no init" status */ + return op-ostart; } size_t ZSTD_compressEnd (ZSTD_CCtx* cctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize) { - size_t endResult; - size_t const cSize = ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1, 1); - if (ZSTD_isError(cSize)) return cSize; - endResult = ZSTD_writeEpilogue(cctx, (char*)dst + cSize, dstCapacity-cSize); - if (ZSTD_isError(endResult)) return endResult; - return cSize + endResult; + size_t endResult; + size_t const cSize = ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1, 1); + if (ZSTD_isError(cSize)) return cSize; + endResult = ZSTD_writeEpilogue(cctx, (char*)dst + cSize, dstCapacity-cSize); + if (ZSTD_isError(endResult)) return endResult; + return cSize + endResult; } static size_t ZSTD_compress_internal (ZSTD_CCtx* cctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict,size_t dictSize, - ZSTD_parameters params) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict,size_t dictSize, + ZSTD_parameters params) { - CHECK_F(ZSTD_compressBegin_internal(cctx, dict, dictSize, params, srcSize)); - return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize); + CHECK_F(ZSTD_compressBegin_internal(cctx, dict, dictSize, params, srcSize)); + return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize); } size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict,size_t dictSize, - ZSTD_parameters params) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict,size_t dictSize, + ZSTD_parameters params) { - CHECK_F(ZSTD_checkCParams(params.cParams)); - return ZSTD_compress_internal(ctx, dst, dstCapacity, src, srcSize, dict, dictSize, params); + CHECK_F(ZSTD_checkCParams(params.cParams)); + return ZSTD_compress_internal(ctx, dst, dstCapacity, src, srcSize, dict, dictSize, params); } size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void* dict, size_t dictSize, int compressionLevel) { - ZSTD_parameters params = ZSTD_getParams(compressionLevel, srcSize, dict ? dictSize : 0); - params.fParams.contentSizeFlag = 1; - return ZSTD_compress_internal(ctx, dst, dstCapacity, src, srcSize, dict, dictSize, params); + ZSTD_parameters params = ZSTD_getParams(compressionLevel, srcSize, dict ? dictSize : 0); + params.fParams.contentSizeFlag = 1; + return ZSTD_compress_internal(ctx, dst, dstCapacity, src, srcSize, dict, dictSize, params); } size_t ZSTD_compressCCtx (ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel) { - return ZSTD_compress_usingDict(ctx, dst, dstCapacity, src, srcSize, NULL, 0, compressionLevel); + return ZSTD_compress_usingDict(ctx, dst, dstCapacity, src, srcSize, NULL, 0, compressionLevel); } size_t ZSTD_compress(void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel) { - size_t result; - ZSTD_CCtx ctxBody; - memset(&ctxBody, 0, sizeof(ctxBody)); - memcpy(&ctxBody.customMem, &defaultCustomMem, sizeof(ZSTD_customMem)); - result = ZSTD_compressCCtx(&ctxBody, dst, dstCapacity, src, srcSize, compressionLevel); - ZSTD_free(ctxBody.workSpace, defaultCustomMem); /* can't free ctxBody itself, as it's on stack; free only heap content */ - return result; + size_t result; + ZSTD_CCtx ctxBody; + memset(&ctxBody, 0, sizeof(ctxBody)); + memcpy(&ctxBody.customMem, &defaultCustomMem, sizeof(ZSTD_customMem)); + result = ZSTD_compressCCtx(&ctxBody, dst, dstCapacity, src, srcSize, compressionLevel); + ZSTD_free(ctxBody.workSpace, defaultCustomMem); /* can't free ctxBody itself, as it's on stack; free only heap content */ + return result; } /* ===== Dictionary API ===== */ struct ZSTD_CDict_s { - void* dictBuffer; - const void* dictContent; - size_t dictContentSize; - ZSTD_CCtx* refContext; + void* dictBuffer; + const void* dictContent; + size_t dictContentSize; + ZSTD_CCtx* refContext; }; /* typedef'd tp ZSTD_CDict within "zstd.h" */ size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict) { - if (cdict==NULL) return 0; /* support sizeof on NULL */ - return ZSTD_sizeof_CCtx(cdict->refContext) + (cdict->dictBuffer ? cdict->dictContentSize : 0) + sizeof(*cdict); + if (cdict==NULL) return 0; /* support sizeof on NULL */ + return ZSTD_sizeof_CCtx(cdict->refContext) + (cdict->dictBuffer ? cdict->dictContentSize : 0) + sizeof(*cdict); } ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, unsigned byReference, - ZSTD_parameters params, ZSTD_customMem customMem) + ZSTD_parameters params, ZSTD_customMem customMem) { - if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; - if (!customMem.customAlloc || !customMem.customFree) return NULL; + if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; + if (!customMem.customAlloc || !customMem.customFree) return NULL; - { ZSTD_CDict* const cdict = (ZSTD_CDict*) ZSTD_malloc(sizeof(ZSTD_CDict), customMem); - ZSTD_CCtx* const cctx = ZSTD_createCCtx_advanced(customMem); + { ZSTD_CDict* const cdict = (ZSTD_CDict*) ZSTD_malloc(sizeof(ZSTD_CDict), customMem); + ZSTD_CCtx* const cctx = ZSTD_createCCtx_advanced(customMem); - if (!cdict || !cctx) { - ZSTD_free(cdict, customMem); - ZSTD_freeCCtx(cctx); - return NULL; - } + if (!cdict || !cctx) { + ZSTD_free(cdict, customMem); + ZSTD_freeCCtx(cctx); + return NULL; + } - if ((byReference) || (!dictBuffer) || (!dictSize)) { - cdict->dictBuffer = NULL; - cdict->dictContent = dictBuffer; - } else { - void* const internalBuffer = ZSTD_malloc(dictSize, customMem); - if (!internalBuffer) { ZSTD_free(cctx, customMem); ZSTD_free(cdict, customMem); return NULL; } - memcpy(internalBuffer, dictBuffer, dictSize); - cdict->dictBuffer = internalBuffer; - cdict->dictContent = internalBuffer; - } + if ((byReference) || (!dictBuffer) || (!dictSize)) { + cdict->dictBuffer = NULL; + cdict->dictContent = dictBuffer; + } else { + void* const internalBuffer = ZSTD_malloc(dictSize, customMem); + if (!internalBuffer) { ZSTD_free(cctx, customMem); ZSTD_free(cdict, customMem); return NULL; } + memcpy(internalBuffer, dictBuffer, dictSize); + cdict->dictBuffer = internalBuffer; + cdict->dictContent = internalBuffer; + } - { size_t const errorCode = ZSTD_compressBegin_advanced(cctx, cdict->dictContent, dictSize, params, 0); - if (ZSTD_isError(errorCode)) { - ZSTD_free(cdict->dictBuffer, customMem); - ZSTD_free(cdict, customMem); - ZSTD_freeCCtx(cctx); - return NULL; - } } + { size_t const errorCode = ZSTD_compressBegin_advanced(cctx, cdict->dictContent, dictSize, params, 0); + if (ZSTD_isError(errorCode)) { + ZSTD_free(cdict->dictBuffer, customMem); + ZSTD_free(cdict, customMem); + ZSTD_freeCCtx(cctx); + return NULL; + } } - cdict->refContext = cctx; - cdict->dictContentSize = dictSize; - return cdict; - } + cdict->refContext = cctx; + cdict->dictContentSize = dictSize; + return cdict; + } } ZSTD_CDict* ZSTD_createCDict(const void* dict, size_t dictSize, int compressionLevel) { - ZSTD_customMem const allocator = { NULL, NULL, NULL }; - ZSTD_parameters params = ZSTD_getParams(compressionLevel, 0, dictSize); - params.fParams.contentSizeFlag = 1; - return ZSTD_createCDict_advanced(dict, dictSize, 0, params, allocator); + ZSTD_customMem const allocator = { NULL, NULL, NULL }; + ZSTD_parameters params = ZSTD_getParams(compressionLevel, 0, dictSize); + params.fParams.contentSizeFlag = 1; + return ZSTD_createCDict_advanced(dict, dictSize, 0, params, allocator); } ZSTD_CDict* ZSTD_createCDict_byReference(const void* dict, size_t dictSize, int compressionLevel) { - ZSTD_customMem const allocator = { NULL, NULL, NULL }; - ZSTD_parameters params = ZSTD_getParams(compressionLevel, 0, dictSize); - params.fParams.contentSizeFlag = 1; - return ZSTD_createCDict_advanced(dict, dictSize, 1, params, allocator); + ZSTD_customMem const allocator = { NULL, NULL, NULL }; + ZSTD_parameters params = ZSTD_getParams(compressionLevel, 0, dictSize); + params.fParams.contentSizeFlag = 1; + return ZSTD_createCDict_advanced(dict, dictSize, 1, params, allocator); } size_t ZSTD_freeCDict(ZSTD_CDict* cdict) { - if (cdict==NULL) return 0; /* support free on NULL */ - { ZSTD_customMem const cMem = cdict->refContext->customMem; - ZSTD_freeCCtx(cdict->refContext); - ZSTD_free(cdict->dictBuffer, cMem); - ZSTD_free(cdict, cMem); - return 0; - } + if (cdict==NULL) return 0; /* support free on NULL */ + { ZSTD_customMem const cMem = cdict->refContext->customMem; + ZSTD_freeCCtx(cdict->refContext); + ZSTD_free(cdict->dictBuffer, cMem); + ZSTD_free(cdict, cMem); + return 0; + } } static ZSTD_parameters ZSTD_getParamsFromCDict(const ZSTD_CDict* cdict) { - return ZSTD_getParamsFromCCtx(cdict->refContext); + return ZSTD_getParamsFromCCtx(cdict->refContext); } size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize) { - if (cdict->dictContentSize) CHECK_F(ZSTD_copyCCtx(cctx, cdict->refContext, pledgedSrcSize)) - else { - ZSTD_parameters params = cdict->refContext->params; - params.fParams.contentSizeFlag = (pledgedSrcSize > 0); - CHECK_F(ZSTD_compressBegin_advanced(cctx, NULL, 0, params, pledgedSrcSize)); - } - return 0; + if (cdict->dictContentSize) CHECK_F(ZSTD_copyCCtx(cctx, cdict->refContext, pledgedSrcSize)) + else { + ZSTD_parameters params = cdict->refContext->params; + params.fParams.contentSizeFlag = (pledgedSrcSize > 0); + CHECK_F(ZSTD_compressBegin_advanced(cctx, NULL, 0, params, pledgedSrcSize)); + } + return 0; } /*! ZSTD_compress_usingCDict() : @@ -2908,18 +2908,18 @@ size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, u * Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. * Note that compression level is decided during dictionary creation */ size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const ZSTD_CDict* cdict) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const ZSTD_CDict* cdict) { - CHECK_F(ZSTD_compressBegin_usingCDict(cctx, cdict, srcSize)); + CHECK_F(ZSTD_compressBegin_usingCDict(cctx, cdict, srcSize)); - if (cdict->refContext->params.fParams.contentSizeFlag==1) { - cctx->params.fParams.contentSizeFlag = 1; - cctx->frameContentSize = srcSize; - } + if (cdict->refContext->params.fParams.contentSizeFlag==1) { + cctx->params.fParams.contentSizeFlag = 1; + cctx->frameContentSize = srcSize; + } - return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize); + return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize); } @@ -2931,64 +2931,64 @@ size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, typedef enum { zcss_init, zcss_load, zcss_flush, zcss_final } ZSTD_cStreamStage; struct ZSTD_CStream_s { - ZSTD_CCtx* cctx; - ZSTD_CDict* cdictLocal; - const ZSTD_CDict* cdict; - char* inBuff; - size_t inBuffSize; - size_t inToCompress; - size_t inBuffPos; - size_t inBuffTarget; - size_t blockSize; - char* outBuff; - size_t outBuffSize; - size_t outBuffContentSize; - size_t outBuffFlushedSize; - ZSTD_cStreamStage stage; - U32 checksum; - U32 frameEnded; - U64 pledgedSrcSize; - U64 inputProcessed; - ZSTD_parameters params; - ZSTD_customMem customMem; + ZSTD_CCtx* cctx; + ZSTD_CDict* cdictLocal; + const ZSTD_CDict* cdict; + char* inBuff; + size_t inBuffSize; + size_t inToCompress; + size_t inBuffPos; + size_t inBuffTarget; + size_t blockSize; + char* outBuff; + size_t outBuffSize; + size_t outBuffContentSize; + size_t outBuffFlushedSize; + ZSTD_cStreamStage stage; + U32 checksum; + U32 frameEnded; + U64 pledgedSrcSize; + U64 inputProcessed; + ZSTD_parameters params; + ZSTD_customMem customMem; }; /* typedef'd to ZSTD_CStream within "zstd.h" */ ZSTD_CStream* ZSTD_createCStream(void) { - return ZSTD_createCStream_advanced(defaultCustomMem); + return ZSTD_createCStream_advanced(defaultCustomMem); } ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem) { - ZSTD_CStream* zcs; + ZSTD_CStream* zcs; - if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; - if (!customMem.customAlloc || !customMem.customFree) return NULL; + if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; + if (!customMem.customAlloc || !customMem.customFree) return NULL; - zcs = (ZSTD_CStream*)ZSTD_malloc(sizeof(ZSTD_CStream), customMem); - if (zcs==NULL) return NULL; - memset(zcs, 0, sizeof(ZSTD_CStream)); - memcpy(&zcs->customMem, &customMem, sizeof(ZSTD_customMem)); - zcs->cctx = ZSTD_createCCtx_advanced(customMem); - if (zcs->cctx == NULL) { ZSTD_freeCStream(zcs); return NULL; } - return zcs; + zcs = (ZSTD_CStream*)ZSTD_malloc(sizeof(ZSTD_CStream), customMem); + if (zcs==NULL) return NULL; + memset(zcs, 0, sizeof(ZSTD_CStream)); + memcpy(&zcs->customMem, &customMem, sizeof(ZSTD_customMem)); + zcs->cctx = ZSTD_createCCtx_advanced(customMem); + if (zcs->cctx == NULL) { ZSTD_freeCStream(zcs); return NULL; } + return zcs; } size_t ZSTD_freeCStream(ZSTD_CStream* zcs) { - if (zcs==NULL) return 0; /* support free on NULL */ - { ZSTD_customMem const cMem = zcs->customMem; - ZSTD_freeCCtx(zcs->cctx); - zcs->cctx = NULL; - ZSTD_freeCDict(zcs->cdictLocal); - zcs->cdictLocal = NULL; - ZSTD_free(zcs->inBuff, cMem); - zcs->inBuff = NULL; - ZSTD_free(zcs->outBuff, cMem); - zcs->outBuff = NULL; - ZSTD_free(zcs, cMem); - return 0; - } + if (zcs==NULL) return 0; /* support free on NULL */ + { ZSTD_customMem const cMem = zcs->customMem; + ZSTD_freeCCtx(zcs->cctx); + zcs->cctx = NULL; + ZSTD_freeCDict(zcs->cdictLocal); + zcs->cdictLocal = NULL; + ZSTD_free(zcs->inBuff, cMem); + zcs->inBuff = NULL; + ZSTD_free(zcs->outBuff, cMem); + zcs->outBuff = NULL; + ZSTD_free(zcs, cMem); + return 0; + } } @@ -2999,96 +2999,96 @@ size_t ZSTD_CStreamOutSize(void) { return ZSTD_compressBound(ZSTD_BLOCKSIZE_ABSO static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize) { - if (zcs->inBuffSize==0) return ERROR(stage_wrong); /* zcs has not been init at least once => can't reset */ + if (zcs->inBuffSize==0) return ERROR(stage_wrong); /* zcs has not been init at least once => can't reset */ - if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict(zcs->cctx, zcs->cdict, pledgedSrcSize)) - else CHECK_F(ZSTD_compressBegin_advanced(zcs->cctx, NULL, 0, zcs->params, pledgedSrcSize)); + if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict(zcs->cctx, zcs->cdict, pledgedSrcSize)) + else CHECK_F(ZSTD_compressBegin_advanced(zcs->cctx, NULL, 0, zcs->params, pledgedSrcSize)); - zcs->inToCompress = 0; - zcs->inBuffPos = 0; - zcs->inBuffTarget = zcs->blockSize; - zcs->outBuffContentSize = zcs->outBuffFlushedSize = 0; - zcs->stage = zcss_load; - zcs->frameEnded = 0; - zcs->pledgedSrcSize = pledgedSrcSize; - zcs->inputProcessed = 0; - return 0; /* ready to go */ + zcs->inToCompress = 0; + zcs->inBuffPos = 0; + zcs->inBuffTarget = zcs->blockSize; + zcs->outBuffContentSize = zcs->outBuffFlushedSize = 0; + zcs->stage = zcss_load; + zcs->frameEnded = 0; + zcs->pledgedSrcSize = pledgedSrcSize; + zcs->inputProcessed = 0; + return 0; /* ready to go */ } size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize) { - zcs->params.fParams.contentSizeFlag = (pledgedSrcSize > 0); + zcs->params.fParams.contentSizeFlag = (pledgedSrcSize > 0); - return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); + return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, - const void* dict, size_t dictSize, - ZSTD_parameters params, unsigned long long pledgedSrcSize) + const void* dict, size_t dictSize, + ZSTD_parameters params, unsigned long long pledgedSrcSize) { - /* allocate buffers */ - { size_t const neededInBuffSize = (size_t)1 << params.cParams.windowLog; - if (zcs->inBuffSize < neededInBuffSize) { - zcs->inBuffSize = neededInBuffSize; - ZSTD_free(zcs->inBuff, zcs->customMem); - zcs->inBuff = (char*) ZSTD_malloc(neededInBuffSize, zcs->customMem); - if (zcs->inBuff == NULL) return ERROR(memory_allocation); - } - zcs->blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, neededInBuffSize); - } - if (zcs->outBuffSize < ZSTD_compressBound(zcs->blockSize)+1) { - zcs->outBuffSize = ZSTD_compressBound(zcs->blockSize)+1; - ZSTD_free(zcs->outBuff, zcs->customMem); - zcs->outBuff = (char*) ZSTD_malloc(zcs->outBuffSize, zcs->customMem); - if (zcs->outBuff == NULL) return ERROR(memory_allocation); - } + /* allocate buffers */ + { size_t const neededInBuffSize = (size_t)1 << params.cParams.windowLog; + if (zcs->inBuffSize < neededInBuffSize) { + zcs->inBuffSize = neededInBuffSize; + ZSTD_free(zcs->inBuff, zcs->customMem); + zcs->inBuff = (char*) ZSTD_malloc(neededInBuffSize, zcs->customMem); + if (zcs->inBuff == NULL) return ERROR(memory_allocation); + } + zcs->blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, neededInBuffSize); + } + if (zcs->outBuffSize < ZSTD_compressBound(zcs->blockSize)+1) { + zcs->outBuffSize = ZSTD_compressBound(zcs->blockSize)+1; + ZSTD_free(zcs->outBuff, zcs->customMem); + zcs->outBuff = (char*) ZSTD_malloc(zcs->outBuffSize, zcs->customMem); + if (zcs->outBuff == NULL) return ERROR(memory_allocation); + } - if (dict && dictSize >= 8) { - ZSTD_freeCDict(zcs->cdictLocal); - zcs->cdictLocal = ZSTD_createCDict_advanced(dict, dictSize, 0, params, zcs->customMem); - if (zcs->cdictLocal == NULL) return ERROR(memory_allocation); - zcs->cdict = zcs->cdictLocal; - } else zcs->cdict = NULL; + if (dict && dictSize >= 8) { + ZSTD_freeCDict(zcs->cdictLocal); + zcs->cdictLocal = ZSTD_createCDict_advanced(dict, dictSize, 0, params, zcs->customMem); + if (zcs->cdictLocal == NULL) return ERROR(memory_allocation); + zcs->cdict = zcs->cdictLocal; + } else zcs->cdict = NULL; - zcs->checksum = params.fParams.checksumFlag > 0; - zcs->params = params; + zcs->checksum = params.fParams.checksumFlag > 0; + zcs->params = params; - return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); + return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } /* note : cdict must outlive compression session */ size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) { - ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict); - size_t const initError = ZSTD_initCStream_advanced(zcs, NULL, 0, params, 0); - zcs->cdict = cdict; - zcs->cctx->dictID = params.fParams.noDictIDFlag ? 0 : cdict->refContext->dictID; - return initError; + ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict); + size_t const initError = ZSTD_initCStream_advanced(zcs, NULL, 0, params, 0); + zcs->cdict = cdict; + zcs->cctx->dictID = params.fParams.noDictIDFlag ? 0 : cdict->refContext->dictID; + return initError; } size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel) { - ZSTD_parameters const params = ZSTD_getParams(compressionLevel, 0, dictSize); - return ZSTD_initCStream_advanced(zcs, dict, dictSize, params, 0); + ZSTD_parameters const params = ZSTD_getParams(compressionLevel, 0, dictSize); + return ZSTD_initCStream_advanced(zcs, dict, dictSize, params, 0); } size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize) { - ZSTD_parameters params = ZSTD_getParams(compressionLevel, pledgedSrcSize, 0); - if (pledgedSrcSize) params.fParams.contentSizeFlag = 1; - return ZSTD_initCStream_advanced(zcs, NULL, 0, params, pledgedSrcSize); + ZSTD_parameters params = ZSTD_getParams(compressionLevel, pledgedSrcSize, 0); + if (pledgedSrcSize) params.fParams.contentSizeFlag = 1; + return ZSTD_initCStream_advanced(zcs, NULL, 0, params, pledgedSrcSize); } size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel) { - return ZSTD_initCStream_usingDict(zcs, NULL, 0, compressionLevel); + return ZSTD_initCStream_usingDict(zcs, NULL, 0, compressionLevel); } size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs) { - if (zcs==NULL) return 0; /* support sizeof on NULL */ - return sizeof(*zcs) + ZSTD_sizeof_CCtx(zcs->cctx) + ZSTD_sizeof_CDict(zcs->cdictLocal) + zcs->outBuffSize + zcs->inBuffSize; + if (zcs==NULL) return 0; /* support sizeof on NULL */ + return sizeof(*zcs) + ZSTD_sizeof_CCtx(zcs->cctx) + ZSTD_sizeof_CDict(zcs->cdictLocal) + zcs->outBuffSize + zcs->inBuffSize; } /*====== Compression ======*/ @@ -3097,103 +3097,103 @@ typedef enum { zsf_gather, zsf_flush, zsf_end } ZSTD_flush_e; MEM_STATIC size_t ZSTD_limitCopy(void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - size_t const length = MIN(dstCapacity, srcSize); - memcpy(dst, src, length); - return length; + size_t const length = MIN(dstCapacity, srcSize); + memcpy(dst, src, length); + return length; } static size_t ZSTD_compressStream_generic(ZSTD_CStream* zcs, - void* dst, size_t* dstCapacityPtr, - const void* src, size_t* srcSizePtr, - ZSTD_flush_e const flush) + void* dst, size_t* dstCapacityPtr, + const void* src, size_t* srcSizePtr, + ZSTD_flush_e const flush) { - U32 someMoreWork = 1; - const char* const istart = (const char*)src; - const char* const iend = istart + *srcSizePtr; - const char* ip = istart; - char* const ostart = (char*)dst; - char* const oend = ostart + *dstCapacityPtr; - char* op = ostart; + U32 someMoreWork = 1; + const char* const istart = (const char*)src; + const char* const iend = istart + *srcSizePtr; + const char* ip = istart; + char* const ostart = (char*)dst; + char* const oend = ostart + *dstCapacityPtr; + char* op = ostart; - while (someMoreWork) { - switch(zcs->stage) - { - case zcss_init: return ERROR(init_missing); /* call ZBUFF_compressInit() first ! */ + while (someMoreWork) { + switch(zcs->stage) + { + case zcss_init: return ERROR(init_missing); /* call ZBUFF_compressInit() first ! */ - case zcss_load: - /* complete inBuffer */ - { size_t const toLoad = zcs->inBuffTarget - zcs->inBuffPos; - size_t const loaded = ZSTD_limitCopy(zcs->inBuff + zcs->inBuffPos, toLoad, ip, iend-ip); - zcs->inBuffPos += loaded; - ip += loaded; - if ( (zcs->inBuffPos==zcs->inToCompress) || (!flush && (toLoad != loaded)) ) { - someMoreWork = 0; break; /* not enough input to get a full block : stop there, wait for more */ - } } - /* compress current block (note : this stage cannot be stopped in the middle) */ - { void* cDst; - size_t cSize; - size_t const iSize = zcs->inBuffPos - zcs->inToCompress; - size_t oSize = oend-op; - if (oSize >= ZSTD_compressBound(iSize)) - cDst = op; /* compress directly into output buffer (avoid flush stage) */ - else - cDst = zcs->outBuff, oSize = zcs->outBuffSize; - cSize = (flush == zsf_end) ? - ZSTD_compressEnd(zcs->cctx, cDst, oSize, zcs->inBuff + zcs->inToCompress, iSize) : - ZSTD_compressContinue(zcs->cctx, cDst, oSize, zcs->inBuff + zcs->inToCompress, iSize); - if (ZSTD_isError(cSize)) return cSize; - if (flush == zsf_end) zcs->frameEnded = 1; - /* prepare next block */ - zcs->inBuffTarget = zcs->inBuffPos + zcs->blockSize; - if (zcs->inBuffTarget > zcs->inBuffSize) - zcs->inBuffPos = 0, zcs->inBuffTarget = zcs->blockSize; /* note : inBuffSize >= blockSize */ - zcs->inToCompress = zcs->inBuffPos; - if (cDst == op) { op += cSize; break; } /* no need to flush */ - zcs->outBuffContentSize = cSize; - zcs->outBuffFlushedSize = 0; - zcs->stage = zcss_flush; /* pass-through to flush stage */ - } + case zcss_load: + /* complete inBuffer */ + { size_t const toLoad = zcs->inBuffTarget - zcs->inBuffPos; + size_t const loaded = ZSTD_limitCopy(zcs->inBuff + zcs->inBuffPos, toLoad, ip, iend-ip); + zcs->inBuffPos += loaded; + ip += loaded; + if ( (zcs->inBuffPos==zcs->inToCompress) || (!flush && (toLoad != loaded)) ) { + someMoreWork = 0; break; /* not enough input to get a full block : stop there, wait for more */ + } } + /* compress current block (note : this stage cannot be stopped in the middle) */ + { void* cDst; + size_t cSize; + size_t const iSize = zcs->inBuffPos - zcs->inToCompress; + size_t oSize = oend-op; + if (oSize >= ZSTD_compressBound(iSize)) + cDst = op; /* compress directly into output buffer (avoid flush stage) */ + else + cDst = zcs->outBuff, oSize = zcs->outBuffSize; + cSize = (flush == zsf_end) ? + ZSTD_compressEnd(zcs->cctx, cDst, oSize, zcs->inBuff + zcs->inToCompress, iSize) : + ZSTD_compressContinue(zcs->cctx, cDst, oSize, zcs->inBuff + zcs->inToCompress, iSize); + if (ZSTD_isError(cSize)) return cSize; + if (flush == zsf_end) zcs->frameEnded = 1; + /* prepare next block */ + zcs->inBuffTarget = zcs->inBuffPos + zcs->blockSize; + if (zcs->inBuffTarget > zcs->inBuffSize) + zcs->inBuffPos = 0, zcs->inBuffTarget = zcs->blockSize; /* note : inBuffSize >= blockSize */ + zcs->inToCompress = zcs->inBuffPos; + if (cDst == op) { op += cSize; break; } /* no need to flush */ + zcs->outBuffContentSize = cSize; + zcs->outBuffFlushedSize = 0; + zcs->stage = zcss_flush; /* pass-through to flush stage */ + } - case zcss_flush: - { size_t const toFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize; - size_t const flushed = ZSTD_limitCopy(op, oend-op, zcs->outBuff + zcs->outBuffFlushedSize, toFlush); - op += flushed; - zcs->outBuffFlushedSize += flushed; - if (toFlush!=flushed) { someMoreWork = 0; break; } /* dst too small to store flushed data : stop there */ - zcs->outBuffContentSize = zcs->outBuffFlushedSize = 0; - zcs->stage = zcss_load; - break; - } + case zcss_flush: + { size_t const toFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize; + size_t const flushed = ZSTD_limitCopy(op, oend-op, zcs->outBuff + zcs->outBuffFlushedSize, toFlush); + op += flushed; + zcs->outBuffFlushedSize += flushed; + if (toFlush!=flushed) { someMoreWork = 0; break; } /* dst too small to store flushed data : stop there */ + zcs->outBuffContentSize = zcs->outBuffFlushedSize = 0; + zcs->stage = zcss_load; + break; + } - case zcss_final: - someMoreWork = 0; /* do nothing */ - break; + case zcss_final: + someMoreWork = 0; /* do nothing */ + break; - default: - return ERROR(GENERIC); /* impossible */ - } - } + default: + return ERROR(GENERIC); /* impossible */ + } + } - *srcSizePtr = ip - istart; - *dstCapacityPtr = op - ostart; - zcs->inputProcessed += *srcSizePtr; - if (zcs->frameEnded) return 0; - { size_t hintInSize = zcs->inBuffTarget - zcs->inBuffPos; - if (hintInSize==0) hintInSize = zcs->blockSize; - return hintInSize; - } + *srcSizePtr = ip - istart; + *dstCapacityPtr = op - ostart; + zcs->inputProcessed += *srcSizePtr; + if (zcs->frameEnded) return 0; + { size_t hintInSize = zcs->inBuffTarget - zcs->inBuffPos; + if (hintInSize==0) hintInSize = zcs->blockSize; + return hintInSize; + } } size_t ZSTD_compressStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input) { - size_t sizeRead = input->size - input->pos; - size_t sizeWritten = output->size - output->pos; - size_t const result = ZSTD_compressStream_generic(zcs, - (char*)(output->dst) + output->pos, &sizeWritten, - (const char*)(input->src) + input->pos, &sizeRead, zsf_gather); - input->pos += sizeRead; - output->pos += sizeWritten; - return result; + size_t sizeRead = input->size - input->pos; + size_t sizeWritten = output->size - output->pos; + size_t const result = ZSTD_compressStream_generic(zcs, + (char*)(output->dst) + output->pos, &sizeWritten, + (const char*)(input->src) + input->pos, &sizeRead, zsf_gather); + input->pos += sizeRead; + output->pos += sizeWritten; + return result; } @@ -3203,53 +3203,53 @@ size_t ZSTD_compressStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuf * @return : amount of data remaining to flush */ size_t ZSTD_flushStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output) { - size_t srcSize = 0; - size_t sizeWritten = output->size - output->pos; - size_t const result = ZSTD_compressStream_generic(zcs, - (char*)(output->dst) + output->pos, &sizeWritten, - &srcSize, &srcSize, /* use a valid src address instead of NULL */ - zsf_flush); - output->pos += sizeWritten; - if (ZSTD_isError(result)) return result; - return zcs->outBuffContentSize - zcs->outBuffFlushedSize; /* remaining to flush */ + size_t srcSize = 0; + size_t sizeWritten = output->size - output->pos; + size_t const result = ZSTD_compressStream_generic(zcs, + (char*)(output->dst) + output->pos, &sizeWritten, + &srcSize, &srcSize, /* use a valid src address instead of NULL */ + zsf_flush); + output->pos += sizeWritten; + if (ZSTD_isError(result)) return result; + return zcs->outBuffContentSize - zcs->outBuffFlushedSize; /* remaining to flush */ } size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output) { - BYTE* const ostart = (BYTE*)(output->dst) + output->pos; - BYTE* const oend = (BYTE*)(output->dst) + output->size; - BYTE* op = ostart; + BYTE* const ostart = (BYTE*)(output->dst) + output->pos; + BYTE* const oend = (BYTE*)(output->dst) + output->size; + BYTE* op = ostart; - if ((zcs->pledgedSrcSize) && (zcs->inputProcessed != zcs->pledgedSrcSize)) - return ERROR(srcSize_wrong); /* pledgedSrcSize not respected */ + if ((zcs->pledgedSrcSize) && (zcs->inputProcessed != zcs->pledgedSrcSize)) + return ERROR(srcSize_wrong); /* pledgedSrcSize not respected */ - if (zcs->stage != zcss_final) { - /* flush whatever remains */ - size_t srcSize = 0; - size_t sizeWritten = output->size - output->pos; - size_t const notEnded = ZSTD_compressStream_generic(zcs, ostart, &sizeWritten, &srcSize, &srcSize, zsf_end); /* use a valid src address instead of NULL */ - size_t const remainingToFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize; - op += sizeWritten; - if (remainingToFlush) { - output->pos += sizeWritten; - return remainingToFlush + ZSTD_BLOCKHEADERSIZE /* final empty block */ + (zcs->checksum * 4); - } - /* create epilogue */ - zcs->stage = zcss_final; - zcs->outBuffContentSize = !notEnded ? 0 : - ZSTD_compressEnd(zcs->cctx, zcs->outBuff, zcs->outBuffSize, NULL, 0); /* write epilogue, including final empty block, into outBuff */ - } + if (zcs->stage != zcss_final) { + /* flush whatever remains */ + size_t srcSize = 0; + size_t sizeWritten = output->size - output->pos; + size_t const notEnded = ZSTD_compressStream_generic(zcs, ostart, &sizeWritten, &srcSize, &srcSize, zsf_end); /* use a valid src address instead of NULL */ + size_t const remainingToFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize; + op += sizeWritten; + if (remainingToFlush) { + output->pos += sizeWritten; + return remainingToFlush + ZSTD_BLOCKHEADERSIZE /* final empty block */ + (zcs->checksum * 4); + } + /* create epilogue */ + zcs->stage = zcss_final; + zcs->outBuffContentSize = !notEnded ? 0 : + ZSTD_compressEnd(zcs->cctx, zcs->outBuff, zcs->outBuffSize, NULL, 0); /* write epilogue, including final empty block, into outBuff */ + } - /* flush epilogue */ - { size_t const toFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize; - size_t const flushed = ZSTD_limitCopy(op, oend-op, zcs->outBuff + zcs->outBuffFlushedSize, toFlush); - op += flushed; - zcs->outBuffFlushedSize += flushed; - output->pos += op-ostart; - if (toFlush==flushed) zcs->stage = zcss_init; /* end reached */ - return toFlush - flushed; - } + /* flush epilogue */ + { size_t const toFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize; + size_t const flushed = ZSTD_limitCopy(op, oend-op, zcs->outBuff + zcs->outBuffFlushedSize, toFlush); + op += flushed; + zcs->outBuffFlushedSize += flushed; + output->pos += op-ostart; + if (toFlush==flushed) zcs->stage = zcss_init; /* end reached */ + return toFlush - flushed; + } } @@ -3262,108 +3262,108 @@ int ZSTD_maxCLevel(void) { return ZSTD_MAX_CLEVEL; } static const ZSTD_compressionParameters ZSTD_defaultCParameters[4][ZSTD_MAX_CLEVEL+1] = { { /* "default" */ - /* W, C, H, S, L, TL, strat */ - { 18, 12, 12, 1, 7, 16, ZSTD_fast }, /* level 0 - never used */ - { 19, 13, 14, 1, 7, 16, ZSTD_fast }, /* level 1 */ - { 19, 15, 16, 1, 6, 16, ZSTD_fast }, /* level 2 */ - { 20, 16, 17, 1, 5, 16, ZSTD_dfast }, /* level 3.*/ - { 20, 18, 18, 1, 5, 16, ZSTD_dfast }, /* level 4.*/ - { 20, 15, 18, 3, 5, 16, ZSTD_greedy }, /* level 5 */ - { 21, 16, 19, 2, 5, 16, ZSTD_lazy }, /* level 6 */ - { 21, 17, 20, 3, 5, 16, ZSTD_lazy }, /* level 7 */ - { 21, 18, 20, 3, 5, 16, ZSTD_lazy2 }, /* level 8 */ - { 21, 20, 20, 3, 5, 16, ZSTD_lazy2 }, /* level 9 */ - { 21, 19, 21, 4, 5, 16, ZSTD_lazy2 }, /* level 10 */ - { 22, 20, 22, 4, 5, 16, ZSTD_lazy2 }, /* level 11 */ - { 22, 20, 22, 5, 5, 16, ZSTD_lazy2 }, /* level 12 */ - { 22, 21, 22, 5, 5, 16, ZSTD_lazy2 }, /* level 13 */ - { 22, 21, 22, 6, 5, 16, ZSTD_lazy2 }, /* level 14 */ - { 22, 21, 21, 5, 5, 16, ZSTD_btlazy2 }, /* level 15 */ - { 23, 22, 22, 5, 5, 16, ZSTD_btlazy2 }, /* level 16 */ - { 23, 21, 22, 4, 5, 24, ZSTD_btopt }, /* level 17 */ - { 23, 23, 22, 6, 5, 32, ZSTD_btopt }, /* level 18 */ - { 23, 23, 22, 6, 3, 48, ZSTD_btopt }, /* level 19 */ - { 25, 25, 23, 7, 3, 64, ZSTD_btopt2 }, /* level 20 */ - { 26, 26, 23, 7, 3,256, ZSTD_btopt2 }, /* level 21 */ - { 27, 27, 25, 9, 3,512, ZSTD_btopt2 }, /* level 22 */ + /* W, C, H, S, L, TL, strat */ + { 18, 12, 12, 1, 7, 16, ZSTD_fast }, /* level 0 - never used */ + { 19, 13, 14, 1, 7, 16, ZSTD_fast }, /* level 1 */ + { 19, 15, 16, 1, 6, 16, ZSTD_fast }, /* level 2 */ + { 20, 16, 17, 1, 5, 16, ZSTD_dfast }, /* level 3.*/ + { 20, 18, 18, 1, 5, 16, ZSTD_dfast }, /* level 4.*/ + { 20, 15, 18, 3, 5, 16, ZSTD_greedy }, /* level 5 */ + { 21, 16, 19, 2, 5, 16, ZSTD_lazy }, /* level 6 */ + { 21, 17, 20, 3, 5, 16, ZSTD_lazy }, /* level 7 */ + { 21, 18, 20, 3, 5, 16, ZSTD_lazy2 }, /* level 8 */ + { 21, 20, 20, 3, 5, 16, ZSTD_lazy2 }, /* level 9 */ + { 21, 19, 21, 4, 5, 16, ZSTD_lazy2 }, /* level 10 */ + { 22, 20, 22, 4, 5, 16, ZSTD_lazy2 }, /* level 11 */ + { 22, 20, 22, 5, 5, 16, ZSTD_lazy2 }, /* level 12 */ + { 22, 21, 22, 5, 5, 16, ZSTD_lazy2 }, /* level 13 */ + { 22, 21, 22, 6, 5, 16, ZSTD_lazy2 }, /* level 14 */ + { 22, 21, 21, 5, 5, 16, ZSTD_btlazy2 }, /* level 15 */ + { 23, 22, 22, 5, 5, 16, ZSTD_btlazy2 }, /* level 16 */ + { 23, 21, 22, 4, 5, 24, ZSTD_btopt }, /* level 17 */ + { 23, 23, 22, 6, 5, 32, ZSTD_btopt }, /* level 18 */ + { 23, 23, 22, 6, 3, 48, ZSTD_btopt }, /* level 19 */ + { 25, 25, 23, 7, 3, 64, ZSTD_btopt2 }, /* level 20 */ + { 26, 26, 23, 7, 3,256, ZSTD_btopt2 }, /* level 21 */ + { 27, 27, 25, 9, 3,512, ZSTD_btopt2 }, /* level 22 */ }, { /* for srcSize <= 256 KB */ - /* W, C, H, S, L, T, strat */ - { 0, 0, 0, 0, 0, 0, ZSTD_fast }, /* level 0 - not used */ - { 18, 13, 14, 1, 6, 8, ZSTD_fast }, /* level 1 */ - { 18, 14, 13, 1, 5, 8, ZSTD_dfast }, /* level 2 */ - { 18, 16, 15, 1, 5, 8, ZSTD_dfast }, /* level 3 */ - { 18, 15, 17, 1, 5, 8, ZSTD_greedy }, /* level 4.*/ - { 18, 16, 17, 4, 5, 8, ZSTD_greedy }, /* level 5.*/ - { 18, 16, 17, 3, 5, 8, ZSTD_lazy }, /* level 6.*/ - { 18, 17, 17, 4, 4, 8, ZSTD_lazy }, /* level 7 */ - { 18, 17, 17, 4, 4, 8, ZSTD_lazy2 }, /* level 8 */ - { 18, 17, 17, 5, 4, 8, ZSTD_lazy2 }, /* level 9 */ - { 18, 17, 17, 6, 4, 8, ZSTD_lazy2 }, /* level 10 */ - { 18, 18, 17, 6, 4, 8, ZSTD_lazy2 }, /* level 11.*/ - { 18, 18, 17, 7, 4, 8, ZSTD_lazy2 }, /* level 12.*/ - { 18, 19, 17, 6, 4, 8, ZSTD_btlazy2 }, /* level 13 */ - { 18, 18, 18, 4, 4, 16, ZSTD_btopt }, /* level 14.*/ - { 18, 18, 18, 4, 3, 16, ZSTD_btopt }, /* level 15.*/ - { 18, 19, 18, 6, 3, 32, ZSTD_btopt }, /* level 16.*/ - { 18, 19, 18, 8, 3, 64, ZSTD_btopt }, /* level 17.*/ - { 18, 19, 18, 9, 3,128, ZSTD_btopt }, /* level 18.*/ - { 18, 19, 18, 10, 3,256, ZSTD_btopt }, /* level 19.*/ - { 18, 19, 18, 11, 3,512, ZSTD_btopt2 }, /* level 20.*/ - { 18, 19, 18, 12, 3,512, ZSTD_btopt2 }, /* level 21.*/ - { 18, 19, 18, 13, 3,512, ZSTD_btopt2 }, /* level 22.*/ + /* W, C, H, S, L, T, strat */ + { 0, 0, 0, 0, 0, 0, ZSTD_fast }, /* level 0 - not used */ + { 18, 13, 14, 1, 6, 8, ZSTD_fast }, /* level 1 */ + { 18, 14, 13, 1, 5, 8, ZSTD_dfast }, /* level 2 */ + { 18, 16, 15, 1, 5, 8, ZSTD_dfast }, /* level 3 */ + { 18, 15, 17, 1, 5, 8, ZSTD_greedy }, /* level 4.*/ + { 18, 16, 17, 4, 5, 8, ZSTD_greedy }, /* level 5.*/ + { 18, 16, 17, 3, 5, 8, ZSTD_lazy }, /* level 6.*/ + { 18, 17, 17, 4, 4, 8, ZSTD_lazy }, /* level 7 */ + { 18, 17, 17, 4, 4, 8, ZSTD_lazy2 }, /* level 8 */ + { 18, 17, 17, 5, 4, 8, ZSTD_lazy2 }, /* level 9 */ + { 18, 17, 17, 6, 4, 8, ZSTD_lazy2 }, /* level 10 */ + { 18, 18, 17, 6, 4, 8, ZSTD_lazy2 }, /* level 11.*/ + { 18, 18, 17, 7, 4, 8, ZSTD_lazy2 }, /* level 12.*/ + { 18, 19, 17, 6, 4, 8, ZSTD_btlazy2 }, /* level 13 */ + { 18, 18, 18, 4, 4, 16, ZSTD_btopt }, /* level 14.*/ + { 18, 18, 18, 4, 3, 16, ZSTD_btopt }, /* level 15.*/ + { 18, 19, 18, 6, 3, 32, ZSTD_btopt }, /* level 16.*/ + { 18, 19, 18, 8, 3, 64, ZSTD_btopt }, /* level 17.*/ + { 18, 19, 18, 9, 3,128, ZSTD_btopt }, /* level 18.*/ + { 18, 19, 18, 10, 3,256, ZSTD_btopt }, /* level 19.*/ + { 18, 19, 18, 11, 3,512, ZSTD_btopt2 }, /* level 20.*/ + { 18, 19, 18, 12, 3,512, ZSTD_btopt2 }, /* level 21.*/ + { 18, 19, 18, 13, 3,512, ZSTD_btopt2 }, /* level 22.*/ }, { /* for srcSize <= 128 KB */ - /* W, C, H, S, L, T, strat */ - { 17, 12, 12, 1, 7, 8, ZSTD_fast }, /* level 0 - not used */ - { 17, 12, 13, 1, 6, 8, ZSTD_fast }, /* level 1 */ - { 17, 13, 16, 1, 5, 8, ZSTD_fast }, /* level 2 */ - { 17, 16, 16, 2, 5, 8, ZSTD_dfast }, /* level 3 */ - { 17, 13, 15, 3, 4, 8, ZSTD_greedy }, /* level 4 */ - { 17, 15, 17, 4, 4, 8, ZSTD_greedy }, /* level 5 */ - { 17, 16, 17, 3, 4, 8, ZSTD_lazy }, /* level 6 */ - { 17, 15, 17, 4, 4, 8, ZSTD_lazy2 }, /* level 7 */ - { 17, 17, 17, 4, 4, 8, ZSTD_lazy2 }, /* level 8 */ - { 17, 17, 17, 5, 4, 8, ZSTD_lazy2 }, /* level 9 */ - { 17, 17, 17, 6, 4, 8, ZSTD_lazy2 }, /* level 10 */ - { 17, 17, 17, 7, 4, 8, ZSTD_lazy2 }, /* level 11 */ - { 17, 17, 17, 8, 4, 8, ZSTD_lazy2 }, /* level 12 */ - { 17, 18, 17, 6, 4, 8, ZSTD_btlazy2 }, /* level 13.*/ - { 17, 17, 17, 7, 3, 8, ZSTD_btopt }, /* level 14.*/ - { 17, 17, 17, 7, 3, 16, ZSTD_btopt }, /* level 15.*/ - { 17, 18, 17, 7, 3, 32, ZSTD_btopt }, /* level 16.*/ - { 17, 18, 17, 7, 3, 64, ZSTD_btopt }, /* level 17.*/ - { 17, 18, 17, 7, 3,256, ZSTD_btopt }, /* level 18.*/ - { 17, 18, 17, 8, 3,256, ZSTD_btopt }, /* level 19.*/ - { 17, 18, 17, 9, 3,256, ZSTD_btopt2 }, /* level 20.*/ - { 17, 18, 17, 10, 3,256, ZSTD_btopt2 }, /* level 21.*/ - { 17, 18, 17, 11, 3,512, ZSTD_btopt2 }, /* level 22.*/ + /* W, C, H, S, L, T, strat */ + { 17, 12, 12, 1, 7, 8, ZSTD_fast }, /* level 0 - not used */ + { 17, 12, 13, 1, 6, 8, ZSTD_fast }, /* level 1 */ + { 17, 13, 16, 1, 5, 8, ZSTD_fast }, /* level 2 */ + { 17, 16, 16, 2, 5, 8, ZSTD_dfast }, /* level 3 */ + { 17, 13, 15, 3, 4, 8, ZSTD_greedy }, /* level 4 */ + { 17, 15, 17, 4, 4, 8, ZSTD_greedy }, /* level 5 */ + { 17, 16, 17, 3, 4, 8, ZSTD_lazy }, /* level 6 */ + { 17, 15, 17, 4, 4, 8, ZSTD_lazy2 }, /* level 7 */ + { 17, 17, 17, 4, 4, 8, ZSTD_lazy2 }, /* level 8 */ + { 17, 17, 17, 5, 4, 8, ZSTD_lazy2 }, /* level 9 */ + { 17, 17, 17, 6, 4, 8, ZSTD_lazy2 }, /* level 10 */ + { 17, 17, 17, 7, 4, 8, ZSTD_lazy2 }, /* level 11 */ + { 17, 17, 17, 8, 4, 8, ZSTD_lazy2 }, /* level 12 */ + { 17, 18, 17, 6, 4, 8, ZSTD_btlazy2 }, /* level 13.*/ + { 17, 17, 17, 7, 3, 8, ZSTD_btopt }, /* level 14.*/ + { 17, 17, 17, 7, 3, 16, ZSTD_btopt }, /* level 15.*/ + { 17, 18, 17, 7, 3, 32, ZSTD_btopt }, /* level 16.*/ + { 17, 18, 17, 7, 3, 64, ZSTD_btopt }, /* level 17.*/ + { 17, 18, 17, 7, 3,256, ZSTD_btopt }, /* level 18.*/ + { 17, 18, 17, 8, 3,256, ZSTD_btopt }, /* level 19.*/ + { 17, 18, 17, 9, 3,256, ZSTD_btopt2 }, /* level 20.*/ + { 17, 18, 17, 10, 3,256, ZSTD_btopt2 }, /* level 21.*/ + { 17, 18, 17, 11, 3,512, ZSTD_btopt2 }, /* level 22.*/ }, { /* for srcSize <= 16 KB */ - /* W, C, H, S, L, T, strat */ - { 14, 12, 12, 1, 7, 6, ZSTD_fast }, /* level 0 - not used */ - { 14, 14, 14, 1, 6, 6, ZSTD_fast }, /* level 1 */ - { 14, 14, 14, 1, 4, 6, ZSTD_fast }, /* level 2 */ - { 14, 14, 14, 1, 4, 6, ZSTD_dfast }, /* level 3.*/ - { 14, 14, 14, 4, 4, 6, ZSTD_greedy }, /* level 4.*/ - { 14, 14, 14, 3, 4, 6, ZSTD_lazy }, /* level 5.*/ - { 14, 14, 14, 4, 4, 6, ZSTD_lazy2 }, /* level 6 */ - { 14, 14, 14, 5, 4, 6, ZSTD_lazy2 }, /* level 7 */ - { 14, 14, 14, 6, 4, 6, ZSTD_lazy2 }, /* level 8.*/ - { 14, 15, 14, 6, 4, 6, ZSTD_btlazy2 }, /* level 9.*/ - { 14, 15, 14, 3, 3, 6, ZSTD_btopt }, /* level 10.*/ - { 14, 15, 14, 6, 3, 8, ZSTD_btopt }, /* level 11.*/ - { 14, 15, 14, 6, 3, 16, ZSTD_btopt }, /* level 12.*/ - { 14, 15, 14, 6, 3, 24, ZSTD_btopt }, /* level 13.*/ - { 14, 15, 15, 6, 3, 48, ZSTD_btopt }, /* level 14.*/ - { 14, 15, 15, 6, 3, 64, ZSTD_btopt }, /* level 15.*/ - { 14, 15, 15, 6, 3, 96, ZSTD_btopt }, /* level 16.*/ - { 14, 15, 15, 6, 3,128, ZSTD_btopt }, /* level 17.*/ - { 14, 15, 15, 6, 3,256, ZSTD_btopt }, /* level 18.*/ - { 14, 15, 15, 7, 3,256, ZSTD_btopt }, /* level 19.*/ - { 14, 15, 15, 8, 3,256, ZSTD_btopt2 }, /* level 20.*/ - { 14, 15, 15, 9, 3,256, ZSTD_btopt2 }, /* level 21.*/ - { 14, 15, 15, 10, 3,256, ZSTD_btopt2 }, /* level 22.*/ + /* W, C, H, S, L, T, strat */ + { 14, 12, 12, 1, 7, 6, ZSTD_fast }, /* level 0 - not used */ + { 14, 14, 14, 1, 6, 6, ZSTD_fast }, /* level 1 */ + { 14, 14, 14, 1, 4, 6, ZSTD_fast }, /* level 2 */ + { 14, 14, 14, 1, 4, 6, ZSTD_dfast }, /* level 3.*/ + { 14, 14, 14, 4, 4, 6, ZSTD_greedy }, /* level 4.*/ + { 14, 14, 14, 3, 4, 6, ZSTD_lazy }, /* level 5.*/ + { 14, 14, 14, 4, 4, 6, ZSTD_lazy2 }, /* level 6 */ + { 14, 14, 14, 5, 4, 6, ZSTD_lazy2 }, /* level 7 */ + { 14, 14, 14, 6, 4, 6, ZSTD_lazy2 }, /* level 8.*/ + { 14, 15, 14, 6, 4, 6, ZSTD_btlazy2 }, /* level 9.*/ + { 14, 15, 14, 3, 3, 6, ZSTD_btopt }, /* level 10.*/ + { 14, 15, 14, 6, 3, 8, ZSTD_btopt }, /* level 11.*/ + { 14, 15, 14, 6, 3, 16, ZSTD_btopt }, /* level 12.*/ + { 14, 15, 14, 6, 3, 24, ZSTD_btopt }, /* level 13.*/ + { 14, 15, 15, 6, 3, 48, ZSTD_btopt }, /* level 14.*/ + { 14, 15, 15, 6, 3, 64, ZSTD_btopt }, /* level 15.*/ + { 14, 15, 15, 6, 3, 96, ZSTD_btopt }, /* level 16.*/ + { 14, 15, 15, 6, 3,128, ZSTD_btopt }, /* level 17.*/ + { 14, 15, 15, 6, 3,256, ZSTD_btopt }, /* level 18.*/ + { 14, 15, 15, 7, 3,256, ZSTD_btopt }, /* level 19.*/ + { 14, 15, 15, 8, 3,256, ZSTD_btopt2 }, /* level 20.*/ + { 14, 15, 15, 9, 3,256, ZSTD_btopt2 }, /* level 21.*/ + { 14, 15, 15, 10, 3,256, ZSTD_btopt2 }, /* level 22.*/ }, }; @@ -3372,29 +3372,29 @@ static const ZSTD_compressionParameters ZSTD_defaultCParameters[4][ZSTD_MAX_CLEV * Size values are optional, provide 0 if not known or unused */ ZSTD_compressionParameters ZSTD_getCParams(int compressionLevel, unsigned long long srcSize, size_t dictSize) { - ZSTD_compressionParameters cp; - size_t const addedSize = srcSize ? 0 : 500; - U64 const rSize = srcSize+dictSize ? srcSize+dictSize+addedSize : (U64)-1; - U32 const tableID = (rSize <= 256 KB) + (rSize <= 128 KB) + (rSize <= 16 KB); /* intentional underflow for srcSizeHint == 0 */ - if (compressionLevel <= 0) compressionLevel = ZSTD_DEFAULT_CLEVEL; /* 0 == default; no negative compressionLevel yet */ - if (compressionLevel > ZSTD_MAX_CLEVEL) compressionLevel = ZSTD_MAX_CLEVEL; - cp = ZSTD_defaultCParameters[tableID][compressionLevel]; - if (MEM_32bits()) { /* auto-correction, for 32-bits mode */ - if (cp.windowLog > ZSTD_WINDOWLOG_MAX) cp.windowLog = ZSTD_WINDOWLOG_MAX; - if (cp.chainLog > ZSTD_CHAINLOG_MAX) cp.chainLog = ZSTD_CHAINLOG_MAX; - if (cp.hashLog > ZSTD_HASHLOG_MAX) cp.hashLog = ZSTD_HASHLOG_MAX; - } - cp = ZSTD_adjustCParams(cp, srcSize, dictSize); - return cp; + ZSTD_compressionParameters cp; + size_t const addedSize = srcSize ? 0 : 500; + U64 const rSize = srcSize+dictSize ? srcSize+dictSize+addedSize : (U64)-1; + U32 const tableID = (rSize <= 256 KB) + (rSize <= 128 KB) + (rSize <= 16 KB); /* intentional underflow for srcSizeHint == 0 */ + if (compressionLevel <= 0) compressionLevel = ZSTD_DEFAULT_CLEVEL; /* 0 == default; no negative compressionLevel yet */ + if (compressionLevel > ZSTD_MAX_CLEVEL) compressionLevel = ZSTD_MAX_CLEVEL; + cp = ZSTD_defaultCParameters[tableID][compressionLevel]; + if (MEM_32bits()) { /* auto-correction, for 32-bits mode */ + if (cp.windowLog > ZSTD_WINDOWLOG_MAX) cp.windowLog = ZSTD_WINDOWLOG_MAX; + if (cp.chainLog > ZSTD_CHAINLOG_MAX) cp.chainLog = ZSTD_CHAINLOG_MAX; + if (cp.hashLog > ZSTD_HASHLOG_MAX) cp.hashLog = ZSTD_HASHLOG_MAX; + } + cp = ZSTD_adjustCParams(cp, srcSize, dictSize); + return cp; } /*! ZSTD_getParams() : * same as ZSTD_getCParams(), but @return a `ZSTD_parameters` object (instead of `ZSTD_compressionParameters`). * All fields of `ZSTD_frameParameters` are set to default (0) */ ZSTD_parameters ZSTD_getParams(int compressionLevel, unsigned long long srcSize, size_t dictSize) { - ZSTD_parameters params; - ZSTD_compressionParameters const cParams = ZSTD_getCParams(compressionLevel, srcSize, dictSize); - memset(¶ms, 0, sizeof(params)); - params.cParams = cParams; - return params; + ZSTD_parameters params; + ZSTD_compressionParameters const cParams = ZSTD_getCParams(compressionLevel, srcSize, dictSize); + memset(¶ms, 0, sizeof(params)); + params.cParams = cParams; + return params; } diff --git a/contrib/linux-kernel/lib/zstd_decompress.c b/contrib/linux-kernel/lib/zstd_decompress.c index 06337dbd5..0c0d34686 100644 --- a/contrib/linux-kernel/lib/zstd_decompress.c +++ b/contrib/linux-kernel/lib/zstd_decompress.c @@ -81,44 +81,44 @@ static void ZSTD_copy4(void* dst, const void* src) { memcpy(dst, src, 4); } * Context management ***************************************************************/ typedef enum { ZSTDds_getFrameHeaderSize, ZSTDds_decodeFrameHeader, - ZSTDds_decodeBlockHeader, ZSTDds_decompressBlock, - ZSTDds_decompressLastBlock, ZSTDds_checkChecksum, - ZSTDds_decodeSkippableHeader, ZSTDds_skipFrame } ZSTD_dStage; + ZSTDds_decodeBlockHeader, ZSTDds_decompressBlock, + ZSTDds_decompressLastBlock, ZSTDds_checkChecksum, + ZSTDds_decodeSkippableHeader, ZSTDds_skipFrame } ZSTD_dStage; typedef struct { - FSE_DTable LLTable[FSE_DTABLE_SIZE_U32(LLFSELog)]; - FSE_DTable OFTable[FSE_DTABLE_SIZE_U32(OffFSELog)]; - FSE_DTable MLTable[FSE_DTABLE_SIZE_U32(MLFSELog)]; - HUF_DTable hufTable[HUF_DTABLE_SIZE(HufLog)]; /* can accommodate HUF_decompress4X */ - U32 rep[ZSTD_REP_NUM]; + FSE_DTable LLTable[FSE_DTABLE_SIZE_U32(LLFSELog)]; + FSE_DTable OFTable[FSE_DTABLE_SIZE_U32(OffFSELog)]; + FSE_DTable MLTable[FSE_DTABLE_SIZE_U32(MLFSELog)]; + HUF_DTable hufTable[HUF_DTABLE_SIZE(HufLog)]; /* can accommodate HUF_decompress4X */ + U32 rep[ZSTD_REP_NUM]; } ZSTD_entropyTables_t; struct ZSTD_DCtx_s { - const FSE_DTable* LLTptr; - const FSE_DTable* MLTptr; - const FSE_DTable* OFTptr; - const HUF_DTable* HUFptr; - ZSTD_entropyTables_t entropy; - const void* previousDstEnd; /* detect continuity */ - const void* base; /* start of current segment */ - const void* vBase; /* virtual start of previous segment if it was just before current one */ - const void* dictEnd; /* end of previous segment */ - size_t expected; - ZSTD_frameParams fParams; - blockType_e bType; /* used in ZSTD_decompressContinue(), to transfer blockType between header decoding and block decoding stages */ - ZSTD_dStage stage; - U32 litEntropy; - U32 fseEntropy; - XXH64_state_t xxhState; - size_t headerSize; - U32 dictID; - const BYTE* litPtr; - ZSTD_customMem customMem; - size_t litSize; - size_t rleSize; - BYTE litBuffer[ZSTD_BLOCKSIZE_ABSOLUTEMAX + WILDCOPY_OVERLENGTH]; - BYTE headerBuffer[ZSTD_FRAMEHEADERSIZE_MAX]; + const FSE_DTable* LLTptr; + const FSE_DTable* MLTptr; + const FSE_DTable* OFTptr; + const HUF_DTable* HUFptr; + ZSTD_entropyTables_t entropy; + const void* previousDstEnd; /* detect continuity */ + const void* base; /* start of current segment */ + const void* vBase; /* virtual start of previous segment if it was just before current one */ + const void* dictEnd; /* end of previous segment */ + size_t expected; + ZSTD_frameParams fParams; + blockType_e bType; /* used in ZSTD_decompressContinue(), to transfer blockType between header decoding and block decoding stages */ + ZSTD_dStage stage; + U32 litEntropy; + U32 fseEntropy; + XXH64_state_t xxhState; + size_t headerSize; + U32 dictID; + const BYTE* litPtr; + ZSTD_customMem customMem; + size_t litSize; + size_t rleSize; + BYTE litBuffer[ZSTD_BLOCKSIZE_ABSOLUTEMAX + WILDCOPY_OVERLENGTH]; + BYTE headerBuffer[ZSTD_FRAMEHEADERSIZE_MAX]; }; /* typedef'd to ZSTD_DCtx within "zstd.h" */ size_t ZSTD_sizeof_DCtx (const ZSTD_DCtx* dctx) { return (dctx==NULL) ? 0 : sizeof(ZSTD_DCtx); } @@ -127,77 +127,77 @@ size_t ZSTD_estimateDCtxSize(void) { return sizeof(ZSTD_DCtx); } size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx) { - dctx->expected = ZSTD_frameHeaderSize_prefix; - dctx->stage = ZSTDds_getFrameHeaderSize; - dctx->previousDstEnd = NULL; - dctx->base = NULL; - dctx->vBase = NULL; - dctx->dictEnd = NULL; - dctx->entropy.hufTable[0] = (HUF_DTable)((HufLog)*0x1000001); /* cover both little and big endian */ - dctx->litEntropy = dctx->fseEntropy = 0; - dctx->dictID = 0; - MEM_STATIC_ASSERT(sizeof(dctx->entropy.rep) == sizeof(repStartValue)); - memcpy(dctx->entropy.rep, repStartValue, sizeof(repStartValue)); /* initial repcodes */ - dctx->LLTptr = dctx->entropy.LLTable; - dctx->MLTptr = dctx->entropy.MLTable; - dctx->OFTptr = dctx->entropy.OFTable; - dctx->HUFptr = dctx->entropy.hufTable; - return 0; + dctx->expected = ZSTD_frameHeaderSize_prefix; + dctx->stage = ZSTDds_getFrameHeaderSize; + dctx->previousDstEnd = NULL; + dctx->base = NULL; + dctx->vBase = NULL; + dctx->dictEnd = NULL; + dctx->entropy.hufTable[0] = (HUF_DTable)((HufLog)*0x1000001); /* cover both little and big endian */ + dctx->litEntropy = dctx->fseEntropy = 0; + dctx->dictID = 0; + MEM_STATIC_ASSERT(sizeof(dctx->entropy.rep) == sizeof(repStartValue)); + memcpy(dctx->entropy.rep, repStartValue, sizeof(repStartValue)); /* initial repcodes */ + dctx->LLTptr = dctx->entropy.LLTable; + dctx->MLTptr = dctx->entropy.MLTable; + dctx->OFTptr = dctx->entropy.OFTable; + dctx->HUFptr = dctx->entropy.hufTable; + return 0; } ZSTD_DCtx* ZSTD_createDCtx_advanced(ZSTD_customMem customMem) { - ZSTD_DCtx* dctx; + ZSTD_DCtx* dctx; - if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; - if (!customMem.customAlloc || !customMem.customFree) return NULL; + if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; + if (!customMem.customAlloc || !customMem.customFree) return NULL; - dctx = (ZSTD_DCtx*)ZSTD_malloc(sizeof(ZSTD_DCtx), customMem); - if (!dctx) return NULL; - memcpy(&dctx->customMem, &customMem, sizeof(customMem)); - ZSTD_decompressBegin(dctx); - return dctx; + dctx = (ZSTD_DCtx*)ZSTD_malloc(sizeof(ZSTD_DCtx), customMem); + if (!dctx) return NULL; + memcpy(&dctx->customMem, &customMem, sizeof(customMem)); + ZSTD_decompressBegin(dctx); + return dctx; } ZSTD_DCtx* ZSTD_createDCtx(void) { - return ZSTD_createDCtx_advanced(defaultCustomMem); + return ZSTD_createDCtx_advanced(defaultCustomMem); } size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx) { - if (dctx==NULL) return 0; /* support free on NULL */ - ZSTD_free(dctx, dctx->customMem); - return 0; /* reserved as a potential error code in the future */ + if (dctx==NULL) return 0; /* support free on NULL */ + ZSTD_free(dctx, dctx->customMem); + return 0; /* reserved as a potential error code in the future */ } void ZSTD_copyDCtx(ZSTD_DCtx* dstDCtx, const ZSTD_DCtx* srcDCtx) { - size_t const workSpaceSize = (ZSTD_BLOCKSIZE_ABSOLUTEMAX+WILDCOPY_OVERLENGTH) + ZSTD_frameHeaderSize_max; - memcpy(dstDCtx, srcDCtx, sizeof(ZSTD_DCtx) - workSpaceSize); /* no need to copy workspace */ + size_t const workSpaceSize = (ZSTD_BLOCKSIZE_ABSOLUTEMAX+WILDCOPY_OVERLENGTH) + ZSTD_frameHeaderSize_max; + memcpy(dstDCtx, srcDCtx, sizeof(ZSTD_DCtx) - workSpaceSize); /* no need to copy workspace */ } #if 0 /* deprecated */ static void ZSTD_refDCtx(ZSTD_DCtx* dstDCtx, const ZSTD_DCtx* srcDCtx) { - ZSTD_decompressBegin(dstDCtx); /* init */ - if (srcDCtx) { /* support refDCtx on NULL */ - dstDCtx->dictEnd = srcDCtx->dictEnd; - dstDCtx->vBase = srcDCtx->vBase; - dstDCtx->base = srcDCtx->base; - dstDCtx->previousDstEnd = srcDCtx->previousDstEnd; - dstDCtx->dictID = srcDCtx->dictID; - dstDCtx->litEntropy = srcDCtx->litEntropy; - dstDCtx->fseEntropy = srcDCtx->fseEntropy; - dstDCtx->LLTptr = srcDCtx->entropy.LLTable; - dstDCtx->MLTptr = srcDCtx->entropy.MLTable; - dstDCtx->OFTptr = srcDCtx->entropy.OFTable; - dstDCtx->HUFptr = srcDCtx->entropy.hufTable; - dstDCtx->entropy.rep[0] = srcDCtx->entropy.rep[0]; - dstDCtx->entropy.rep[1] = srcDCtx->entropy.rep[1]; - dstDCtx->entropy.rep[2] = srcDCtx->entropy.rep[2]; - } + ZSTD_decompressBegin(dstDCtx); /* init */ + if (srcDCtx) { /* support refDCtx on NULL */ + dstDCtx->dictEnd = srcDCtx->dictEnd; + dstDCtx->vBase = srcDCtx->vBase; + dstDCtx->base = srcDCtx->base; + dstDCtx->previousDstEnd = srcDCtx->previousDstEnd; + dstDCtx->dictID = srcDCtx->dictID; + dstDCtx->litEntropy = srcDCtx->litEntropy; + dstDCtx->fseEntropy = srcDCtx->fseEntropy; + dstDCtx->LLTptr = srcDCtx->entropy.LLTable; + dstDCtx->MLTptr = srcDCtx->entropy.MLTable; + dstDCtx->OFTptr = srcDCtx->entropy.OFTable; + dstDCtx->HUFptr = srcDCtx->entropy.hufTable; + dstDCtx->entropy.rep[0] = srcDCtx->entropy.rep[0]; + dstDCtx->entropy.rep[1] = srcDCtx->entropy.rep[1]; + dstDCtx->entropy.rep[2] = srcDCtx->entropy.rep[2]; + } } #endif @@ -215,15 +215,15 @@ static void ZSTD_refDDict(ZSTD_DCtx* dstDCtx, const ZSTD_DDict* ddict); * Note 3 : Skippable Frame Identifiers are considered valid. */ unsigned ZSTD_isFrame(const void* buffer, size_t size) { - if (size < 4) return 0; - { U32 const magic = MEM_readLE32(buffer); - if (magic == ZSTD_MAGICNUMBER) return 1; - if ((magic & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) return 1; - } + if (size < 4) return 0; + { U32 const magic = MEM_readLE32(buffer); + if (magic == ZSTD_MAGICNUMBER) return 1; + if ((magic & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) return 1; + } #if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) - if (ZSTD_isLegacy(buffer, size)) return 1; + if (ZSTD_isLegacy(buffer, size)) return 1; #endif - return 0; + return 0; } @@ -232,14 +232,14 @@ unsigned ZSTD_isFrame(const void* buffer, size_t size) * @return : size of the Frame Header */ static size_t ZSTD_frameHeaderSize(const void* src, size_t srcSize) { - if (srcSize < ZSTD_frameHeaderSize_prefix) return ERROR(srcSize_wrong); - { BYTE const fhd = ((const BYTE*)src)[4]; - U32 const dictID= fhd & 3; - U32 const singleSegment = (fhd >> 5) & 1; - U32 const fcsId = fhd >> 6; - return ZSTD_frameHeaderSize_prefix + !singleSegment + ZSTD_did_fieldSize[dictID] + ZSTD_fcs_fieldSize[fcsId] - + (singleSegment && !fcsId); - } + if (srcSize < ZSTD_frameHeaderSize_prefix) return ERROR(srcSize_wrong); + { BYTE const fhd = ((const BYTE*)src)[4]; + U32 const dictID= fhd & 3; + U32 const singleSegment = (fhd >> 5) & 1; + U32 const fcsId = fhd >> 6; + return ZSTD_frameHeaderSize_prefix + !singleSegment + ZSTD_did_fieldSize[dictID] + ZSTD_fcs_fieldSize[fcsId] + + (singleSegment && !fcsId); + } } @@ -250,67 +250,67 @@ static size_t ZSTD_frameHeaderSize(const void* src, size_t srcSize) * or an error code, which can be tested using ZSTD_isError() */ size_t ZSTD_getFrameParams(ZSTD_frameParams* fparamsPtr, const void* src, size_t srcSize) { - const BYTE* ip = (const BYTE*)src; + const BYTE* ip = (const BYTE*)src; - if (srcSize < ZSTD_frameHeaderSize_prefix) return ZSTD_frameHeaderSize_prefix; - if (MEM_readLE32(src) != ZSTD_MAGICNUMBER) { - if ((MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { - if (srcSize < ZSTD_skippableHeaderSize) return ZSTD_skippableHeaderSize; /* magic number + skippable frame length */ - memset(fparamsPtr, 0, sizeof(*fparamsPtr)); - fparamsPtr->frameContentSize = MEM_readLE32((const char *)src + 4); - fparamsPtr->windowSize = 0; /* windowSize==0 means a frame is skippable */ - return 0; - } - return ERROR(prefix_unknown); - } + if (srcSize < ZSTD_frameHeaderSize_prefix) return ZSTD_frameHeaderSize_prefix; + if (MEM_readLE32(src) != ZSTD_MAGICNUMBER) { + if ((MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { + if (srcSize < ZSTD_skippableHeaderSize) return ZSTD_skippableHeaderSize; /* magic number + skippable frame length */ + memset(fparamsPtr, 0, sizeof(*fparamsPtr)); + fparamsPtr->frameContentSize = MEM_readLE32((const char *)src + 4); + fparamsPtr->windowSize = 0; /* windowSize==0 means a frame is skippable */ + return 0; + } + return ERROR(prefix_unknown); + } - /* ensure there is enough `srcSize` to fully read/decode frame header */ - { size_t const fhsize = ZSTD_frameHeaderSize(src, srcSize); - if (srcSize < fhsize) return fhsize; } + /* ensure there is enough `srcSize` to fully read/decode frame header */ + { size_t const fhsize = ZSTD_frameHeaderSize(src, srcSize); + if (srcSize < fhsize) return fhsize; } - { BYTE const fhdByte = ip[4]; - size_t pos = 5; - U32 const dictIDSizeCode = fhdByte&3; - U32 const checksumFlag = (fhdByte>>2)&1; - U32 const singleSegment = (fhdByte>>5)&1; - U32 const fcsID = fhdByte>>6; - U32 const windowSizeMax = 1U << ZSTD_WINDOWLOG_MAX; - U32 windowSize = 0; - U32 dictID = 0; - U64 frameContentSize = 0; - if ((fhdByte & 0x08) != 0) return ERROR(frameParameter_unsupported); /* reserved bits, which must be zero */ - if (!singleSegment) { - BYTE const wlByte = ip[pos++]; - U32 const windowLog = (wlByte >> 3) + ZSTD_WINDOWLOG_ABSOLUTEMIN; - if (windowLog > ZSTD_WINDOWLOG_MAX) return ERROR(frameParameter_windowTooLarge); /* avoids issue with 1 << windowLog */ - windowSize = (1U << windowLog); - windowSize += (windowSize >> 3) * (wlByte&7); - } + { BYTE const fhdByte = ip[4]; + size_t pos = 5; + U32 const dictIDSizeCode = fhdByte&3; + U32 const checksumFlag = (fhdByte>>2)&1; + U32 const singleSegment = (fhdByte>>5)&1; + U32 const fcsID = fhdByte>>6; + U32 const windowSizeMax = 1U << ZSTD_WINDOWLOG_MAX; + U32 windowSize = 0; + U32 dictID = 0; + U64 frameContentSize = 0; + if ((fhdByte & 0x08) != 0) return ERROR(frameParameter_unsupported); /* reserved bits, which must be zero */ + if (!singleSegment) { + BYTE const wlByte = ip[pos++]; + U32 const windowLog = (wlByte >> 3) + ZSTD_WINDOWLOG_ABSOLUTEMIN; + if (windowLog > ZSTD_WINDOWLOG_MAX) return ERROR(frameParameter_windowTooLarge); /* avoids issue with 1 << windowLog */ + windowSize = (1U << windowLog); + windowSize += (windowSize >> 3) * (wlByte&7); + } - switch(dictIDSizeCode) - { - default: /* impossible */ - case 0 : break; - case 1 : dictID = ip[pos]; pos++; break; - case 2 : dictID = MEM_readLE16(ip+pos); pos+=2; break; - case 3 : dictID = MEM_readLE32(ip+pos); pos+=4; break; - } - switch(fcsID) - { - default: /* impossible */ - case 0 : if (singleSegment) frameContentSize = ip[pos]; break; - case 1 : frameContentSize = MEM_readLE16(ip+pos)+256; break; - case 2 : frameContentSize = MEM_readLE32(ip+pos); break; - case 3 : frameContentSize = MEM_readLE64(ip+pos); break; - } - if (!windowSize) windowSize = (U32)frameContentSize; - if (windowSize > windowSizeMax) return ERROR(frameParameter_windowTooLarge); - fparamsPtr->frameContentSize = frameContentSize; - fparamsPtr->windowSize = windowSize; - fparamsPtr->dictID = dictID; - fparamsPtr->checksumFlag = checksumFlag; - } - return 0; + switch(dictIDSizeCode) + { + default: /* impossible */ + case 0 : break; + case 1 : dictID = ip[pos]; pos++; break; + case 2 : dictID = MEM_readLE16(ip+pos); pos+=2; break; + case 3 : dictID = MEM_readLE32(ip+pos); pos+=4; break; + } + switch(fcsID) + { + default: /* impossible */ + case 0 : if (singleSegment) frameContentSize = ip[pos]; break; + case 1 : frameContentSize = MEM_readLE16(ip+pos)+256; break; + case 2 : frameContentSize = MEM_readLE32(ip+pos); break; + case 3 : frameContentSize = MEM_readLE64(ip+pos); break; + } + if (!windowSize) windowSize = (U32)frameContentSize; + if (windowSize > windowSizeMax) return ERROR(frameParameter_windowTooLarge); + fparamsPtr->frameContentSize = frameContentSize; + fparamsPtr->windowSize = windowSize; + fparamsPtr->dictID = dictID; + fparamsPtr->checksumFlag = checksumFlag; + } + return 0; } /** ZSTD_getFrameContentSize() : @@ -321,23 +321,23 @@ size_t ZSTD_getFrameParams(ZSTD_frameParams* fparamsPtr, const void* src, size_t unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize) { #if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) - if (ZSTD_isLegacy(src, srcSize)) { - unsigned long long const ret = ZSTD_getDecompressedSize_legacy(src, srcSize); - return ret == 0 ? ZSTD_CONTENTSIZE_UNKNOWN : ret; - } + if (ZSTD_isLegacy(src, srcSize)) { + unsigned long long const ret = ZSTD_getDecompressedSize_legacy(src, srcSize); + return ret == 0 ? ZSTD_CONTENTSIZE_UNKNOWN : ret; + } #endif - { - ZSTD_frameParams fParams; - if (ZSTD_getFrameParams(&fParams, src, srcSize) != 0) return ZSTD_CONTENTSIZE_ERROR; - if (fParams.windowSize == 0) { - /* Either skippable or empty frame, size == 0 either way */ - return 0; - } else if (fParams.frameContentSize != 0) { - return fParams.frameContentSize; - } else { - return ZSTD_CONTENTSIZE_UNKNOWN; - } - } + { + ZSTD_frameParams fParams; + if (ZSTD_getFrameParams(&fParams, src, srcSize) != 0) return ZSTD_CONTENTSIZE_ERROR; + if (fParams.windowSize == 0) { + /* Either skippable or empty frame, size == 0 either way */ + return 0; + } else if (fParams.frameContentSize != 0) { + return fParams.frameContentSize; + } else { + return ZSTD_CONTENTSIZE_UNKNOWN; + } + } } /** ZSTD_findDecompressedSize() : @@ -347,64 +347,64 @@ unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize) * @return : decompressed size of the frames contained */ unsigned long long ZSTD_findDecompressedSize(const void* src, size_t srcSize) { - { - unsigned long long totalDstSize = 0; - while (srcSize >= ZSTD_frameHeaderSize_prefix) { - const U32 magicNumber = MEM_readLE32(src); + { + unsigned long long totalDstSize = 0; + while (srcSize >= ZSTD_frameHeaderSize_prefix) { + const U32 magicNumber = MEM_readLE32(src); - if ((magicNumber & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { - size_t skippableSize; - if (srcSize < ZSTD_skippableHeaderSize) - return ERROR(srcSize_wrong); - skippableSize = MEM_readLE32((const BYTE *)src + 4) + - ZSTD_skippableHeaderSize; - if (srcSize < skippableSize) { - return ZSTD_CONTENTSIZE_ERROR; - } + if ((magicNumber & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { + size_t skippableSize; + if (srcSize < ZSTD_skippableHeaderSize) + return ERROR(srcSize_wrong); + skippableSize = MEM_readLE32((const BYTE *)src + 4) + + ZSTD_skippableHeaderSize; + if (srcSize < skippableSize) { + return ZSTD_CONTENTSIZE_ERROR; + } - src = (const BYTE *)src + skippableSize; - srcSize -= skippableSize; - continue; - } + src = (const BYTE *)src + skippableSize; + srcSize -= skippableSize; + continue; + } - { - unsigned long long const ret = ZSTD_getFrameContentSize(src, srcSize); - if (ret >= ZSTD_CONTENTSIZE_ERROR) return ret; + { + unsigned long long const ret = ZSTD_getFrameContentSize(src, srcSize); + if (ret >= ZSTD_CONTENTSIZE_ERROR) return ret; - /* check for overflow */ - if (totalDstSize + ret < totalDstSize) return ZSTD_CONTENTSIZE_ERROR; - totalDstSize += ret; - } - { - size_t const frameSrcSize = ZSTD_findFrameCompressedSize(src, srcSize); - if (ZSTD_isError(frameSrcSize)) { - return ZSTD_CONTENTSIZE_ERROR; - } + /* check for overflow */ + if (totalDstSize + ret < totalDstSize) return ZSTD_CONTENTSIZE_ERROR; + totalDstSize += ret; + } + { + size_t const frameSrcSize = ZSTD_findFrameCompressedSize(src, srcSize); + if (ZSTD_isError(frameSrcSize)) { + return ZSTD_CONTENTSIZE_ERROR; + } - src = (const BYTE *)src + frameSrcSize; - srcSize -= frameSrcSize; - } - } + src = (const BYTE *)src + frameSrcSize; + srcSize -= frameSrcSize; + } + } - if (srcSize) { - return ZSTD_CONTENTSIZE_ERROR; - } + if (srcSize) { + return ZSTD_CONTENTSIZE_ERROR; + } - return totalDstSize; - } + return totalDstSize; + } } /** ZSTD_getDecompressedSize() : * compatible with legacy mode * @return : decompressed size if known, 0 otherwise - note : 0 can mean any of the following : - - decompressed size is not present within frame header - - frame header unknown / not supported - - frame header not complete (`srcSize` too small) */ + note : 0 can mean any of the following : + - decompressed size is not present within frame header + - frame header unknown / not supported + - frame header not complete (`srcSize` too small) */ unsigned long long ZSTD_getDecompressedSize(const void* src, size_t srcSize) { - unsigned long long const ret = ZSTD_getFrameContentSize(src, srcSize); - return ret >= ZSTD_CONTENTSIZE_ERROR ? 0 : ret; + unsigned long long const ret = ZSTD_getFrameContentSize(src, srcSize); + return ret >= ZSTD_CONTENTSIZE_ERROR ? 0 : ret; } @@ -413,517 +413,517 @@ unsigned long long ZSTD_getDecompressedSize(const void* src, size_t srcSize) * @return : 0 if success, or an error code, which can be tested using ZSTD_isError() */ static size_t ZSTD_decodeFrameHeader(ZSTD_DCtx* dctx, const void* src, size_t headerSize) { - size_t const result = ZSTD_getFrameParams(&(dctx->fParams), src, headerSize); - if (ZSTD_isError(result)) return result; /* invalid header */ - if (result>0) return ERROR(srcSize_wrong); /* headerSize too small */ - if (dctx->fParams.dictID && (dctx->dictID != dctx->fParams.dictID)) return ERROR(dictionary_wrong); - if (dctx->fParams.checksumFlag) XXH64_reset(&dctx->xxhState, 0); - return 0; + size_t const result = ZSTD_getFrameParams(&(dctx->fParams), src, headerSize); + if (ZSTD_isError(result)) return result; /* invalid header */ + if (result>0) return ERROR(srcSize_wrong); /* headerSize too small */ + if (dctx->fParams.dictID && (dctx->dictID != dctx->fParams.dictID)) return ERROR(dictionary_wrong); + if (dctx->fParams.checksumFlag) XXH64_reset(&dctx->xxhState, 0); + return 0; } typedef struct { - blockType_e blockType; - U32 lastBlock; - U32 origSize; + blockType_e blockType; + U32 lastBlock; + U32 origSize; } blockProperties_t; /*! ZSTD_getcBlockSize() : * Provides the size of compressed block from block header `src` */ size_t ZSTD_getcBlockSize(const void* src, size_t srcSize, blockProperties_t* bpPtr) { - if (srcSize < ZSTD_blockHeaderSize) return ERROR(srcSize_wrong); - { U32 const cBlockHeader = MEM_readLE24(src); - U32 const cSize = cBlockHeader >> 3; - bpPtr->lastBlock = cBlockHeader & 1; - bpPtr->blockType = (blockType_e)((cBlockHeader >> 1) & 3); - bpPtr->origSize = cSize; /* only useful for RLE */ - if (bpPtr->blockType == bt_rle) return 1; - if (bpPtr->blockType == bt_reserved) return ERROR(corruption_detected); - return cSize; - } + if (srcSize < ZSTD_blockHeaderSize) return ERROR(srcSize_wrong); + { U32 const cBlockHeader = MEM_readLE24(src); + U32 const cSize = cBlockHeader >> 3; + bpPtr->lastBlock = cBlockHeader & 1; + bpPtr->blockType = (blockType_e)((cBlockHeader >> 1) & 3); + bpPtr->origSize = cSize; /* only useful for RLE */ + if (bpPtr->blockType == bt_rle) return 1; + if (bpPtr->blockType == bt_reserved) return ERROR(corruption_detected); + return cSize; + } } static size_t ZSTD_copyRawBlock(void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - if (srcSize > dstCapacity) return ERROR(dstSize_tooSmall); - memcpy(dst, src, srcSize); - return srcSize; + if (srcSize > dstCapacity) return ERROR(dstSize_tooSmall); + memcpy(dst, src, srcSize); + return srcSize; } static size_t ZSTD_setRleBlock(void* dst, size_t dstCapacity, const void* src, size_t srcSize, size_t regenSize) { - if (srcSize != 1) return ERROR(srcSize_wrong); - if (regenSize > dstCapacity) return ERROR(dstSize_tooSmall); - memset(dst, *(const BYTE*)src, regenSize); - return regenSize; + if (srcSize != 1) return ERROR(srcSize_wrong); + if (regenSize > dstCapacity) return ERROR(dstSize_tooSmall); + memset(dst, *(const BYTE*)src, regenSize); + return regenSize; } /*! ZSTD_decodeLiteralsBlock() : - @return : nb of bytes read from src (< srcSize ) */ + @return : nb of bytes read from src (< srcSize ) */ size_t ZSTD_decodeLiteralsBlock(ZSTD_DCtx* dctx, - const void* src, size_t srcSize) /* note : srcSize < BLOCKSIZE */ + const void* src, size_t srcSize) /* note : srcSize < BLOCKSIZE */ { - if (srcSize < MIN_CBLOCK_SIZE) return ERROR(corruption_detected); + if (srcSize < MIN_CBLOCK_SIZE) return ERROR(corruption_detected); - { const BYTE* const istart = (const BYTE*) src; - symbolEncodingType_e const litEncType = (symbolEncodingType_e)(istart[0] & 3); + { const BYTE* const istart = (const BYTE*) src; + symbolEncodingType_e const litEncType = (symbolEncodingType_e)(istart[0] & 3); - switch(litEncType) - { - case set_repeat: - if (dctx->litEntropy==0) return ERROR(dictionary_corrupted); - /* fall-through */ - case set_compressed: - if (srcSize < 5) return ERROR(corruption_detected); /* srcSize >= MIN_CBLOCK_SIZE == 3; here we need up to 5 for case 3 */ - { size_t lhSize, litSize, litCSize; - U32 singleStream=0; - U32 const lhlCode = (istart[0] >> 2) & 3; - U32 const lhc = MEM_readLE32(istart); - switch(lhlCode) - { - case 0: case 1: default: /* note : default is impossible, since lhlCode into [0..3] */ - /* 2 - 2 - 10 - 10 */ - singleStream = !lhlCode; - lhSize = 3; - litSize = (lhc >> 4) & 0x3FF; - litCSize = (lhc >> 14) & 0x3FF; - break; - case 2: - /* 2 - 2 - 14 - 14 */ - lhSize = 4; - litSize = (lhc >> 4) & 0x3FFF; - litCSize = lhc >> 18; - break; - case 3: - /* 2 - 2 - 18 - 18 */ - lhSize = 5; - litSize = (lhc >> 4) & 0x3FFFF; - litCSize = (lhc >> 22) + (istart[4] << 10); - break; - } - if (litSize > ZSTD_BLOCKSIZE_ABSOLUTEMAX) return ERROR(corruption_detected); - if (litCSize + lhSize > srcSize) return ERROR(corruption_detected); + switch(litEncType) + { + case set_repeat: + if (dctx->litEntropy==0) return ERROR(dictionary_corrupted); + /* fall-through */ + case set_compressed: + if (srcSize < 5) return ERROR(corruption_detected); /* srcSize >= MIN_CBLOCK_SIZE == 3; here we need up to 5 for case 3 */ + { size_t lhSize, litSize, litCSize; + U32 singleStream=0; + U32 const lhlCode = (istart[0] >> 2) & 3; + U32 const lhc = MEM_readLE32(istart); + switch(lhlCode) + { + case 0: case 1: default: /* note : default is impossible, since lhlCode into [0..3] */ + /* 2 - 2 - 10 - 10 */ + singleStream = !lhlCode; + lhSize = 3; + litSize = (lhc >> 4) & 0x3FF; + litCSize = (lhc >> 14) & 0x3FF; + break; + case 2: + /* 2 - 2 - 14 - 14 */ + lhSize = 4; + litSize = (lhc >> 4) & 0x3FFF; + litCSize = lhc >> 18; + break; + case 3: + /* 2 - 2 - 18 - 18 */ + lhSize = 5; + litSize = (lhc >> 4) & 0x3FFFF; + litCSize = (lhc >> 22) + (istart[4] << 10); + break; + } + if (litSize > ZSTD_BLOCKSIZE_ABSOLUTEMAX) return ERROR(corruption_detected); + if (litCSize + lhSize > srcSize) return ERROR(corruption_detected); - if (HUF_isError((litEncType==set_repeat) ? - ( singleStream ? - HUF_decompress1X_usingDTable(dctx->litBuffer, litSize, istart+lhSize, litCSize, dctx->HUFptr) : - HUF_decompress4X_usingDTable(dctx->litBuffer, litSize, istart+lhSize, litCSize, dctx->HUFptr) ) : - ( singleStream ? - HUF_decompress1X2_DCtx(dctx->entropy.hufTable, dctx->litBuffer, litSize, istart+lhSize, litCSize) : - HUF_decompress4X_hufOnly (dctx->entropy.hufTable, dctx->litBuffer, litSize, istart+lhSize, litCSize)) )) - return ERROR(corruption_detected); + if (HUF_isError((litEncType==set_repeat) ? + ( singleStream ? + HUF_decompress1X_usingDTable(dctx->litBuffer, litSize, istart+lhSize, litCSize, dctx->HUFptr) : + HUF_decompress4X_usingDTable(dctx->litBuffer, litSize, istart+lhSize, litCSize, dctx->HUFptr) ) : + ( singleStream ? + HUF_decompress1X2_DCtx(dctx->entropy.hufTable, dctx->litBuffer, litSize, istart+lhSize, litCSize) : + HUF_decompress4X_hufOnly (dctx->entropy.hufTable, dctx->litBuffer, litSize, istart+lhSize, litCSize)) )) + return ERROR(corruption_detected); - dctx->litPtr = dctx->litBuffer; - dctx->litSize = litSize; - dctx->litEntropy = 1; - if (litEncType==set_compressed) dctx->HUFptr = dctx->entropy.hufTable; - memset(dctx->litBuffer + dctx->litSize, 0, WILDCOPY_OVERLENGTH); - return litCSize + lhSize; - } + dctx->litPtr = dctx->litBuffer; + dctx->litSize = litSize; + dctx->litEntropy = 1; + if (litEncType==set_compressed) dctx->HUFptr = dctx->entropy.hufTable; + memset(dctx->litBuffer + dctx->litSize, 0, WILDCOPY_OVERLENGTH); + return litCSize + lhSize; + } - case set_basic: - { size_t litSize, lhSize; - U32 const lhlCode = ((istart[0]) >> 2) & 3; - switch(lhlCode) - { - case 0: case 2: default: /* note : default is impossible, since lhlCode into [0..3] */ - lhSize = 1; - litSize = istart[0] >> 3; - break; - case 1: - lhSize = 2; - litSize = MEM_readLE16(istart) >> 4; - break; - case 3: - lhSize = 3; - litSize = MEM_readLE24(istart) >> 4; - break; - } + case set_basic: + { size_t litSize, lhSize; + U32 const lhlCode = ((istart[0]) >> 2) & 3; + switch(lhlCode) + { + case 0: case 2: default: /* note : default is impossible, since lhlCode into [0..3] */ + lhSize = 1; + litSize = istart[0] >> 3; + break; + case 1: + lhSize = 2; + litSize = MEM_readLE16(istart) >> 4; + break; + case 3: + lhSize = 3; + litSize = MEM_readLE24(istart) >> 4; + break; + } - if (lhSize+litSize+WILDCOPY_OVERLENGTH > srcSize) { /* risk reading beyond src buffer with wildcopy */ - if (litSize+lhSize > srcSize) return ERROR(corruption_detected); - memcpy(dctx->litBuffer, istart+lhSize, litSize); - dctx->litPtr = dctx->litBuffer; - dctx->litSize = litSize; - memset(dctx->litBuffer + dctx->litSize, 0, WILDCOPY_OVERLENGTH); - return lhSize+litSize; - } - /* direct reference into compressed stream */ - dctx->litPtr = istart+lhSize; - dctx->litSize = litSize; - return lhSize+litSize; - } + if (lhSize+litSize+WILDCOPY_OVERLENGTH > srcSize) { /* risk reading beyond src buffer with wildcopy */ + if (litSize+lhSize > srcSize) return ERROR(corruption_detected); + memcpy(dctx->litBuffer, istart+lhSize, litSize); + dctx->litPtr = dctx->litBuffer; + dctx->litSize = litSize; + memset(dctx->litBuffer + dctx->litSize, 0, WILDCOPY_OVERLENGTH); + return lhSize+litSize; + } + /* direct reference into compressed stream */ + dctx->litPtr = istart+lhSize; + dctx->litSize = litSize; + return lhSize+litSize; + } - case set_rle: - { U32 const lhlCode = ((istart[0]) >> 2) & 3; - size_t litSize, lhSize; - switch(lhlCode) - { - case 0: case 2: default: /* note : default is impossible, since lhlCode into [0..3] */ - lhSize = 1; - litSize = istart[0] >> 3; - break; - case 1: - lhSize = 2; - litSize = MEM_readLE16(istart) >> 4; - break; - case 3: - lhSize = 3; - litSize = MEM_readLE24(istart) >> 4; - if (srcSize<4) return ERROR(corruption_detected); /* srcSize >= MIN_CBLOCK_SIZE == 3; here we need lhSize+1 = 4 */ - break; - } - if (litSize > ZSTD_BLOCKSIZE_ABSOLUTEMAX) return ERROR(corruption_detected); - memset(dctx->litBuffer, istart[lhSize], litSize + WILDCOPY_OVERLENGTH); - dctx->litPtr = dctx->litBuffer; - dctx->litSize = litSize; - return lhSize+1; - } - default: - return ERROR(corruption_detected); /* impossible */ - } - } + case set_rle: + { U32 const lhlCode = ((istart[0]) >> 2) & 3; + size_t litSize, lhSize; + switch(lhlCode) + { + case 0: case 2: default: /* note : default is impossible, since lhlCode into [0..3] */ + lhSize = 1; + litSize = istart[0] >> 3; + break; + case 1: + lhSize = 2; + litSize = MEM_readLE16(istart) >> 4; + break; + case 3: + lhSize = 3; + litSize = MEM_readLE24(istart) >> 4; + if (srcSize<4) return ERROR(corruption_detected); /* srcSize >= MIN_CBLOCK_SIZE == 3; here we need lhSize+1 = 4 */ + break; + } + if (litSize > ZSTD_BLOCKSIZE_ABSOLUTEMAX) return ERROR(corruption_detected); + memset(dctx->litBuffer, istart[lhSize], litSize + WILDCOPY_OVERLENGTH); + dctx->litPtr = dctx->litBuffer; + dctx->litSize = litSize; + return lhSize+1; + } + default: + return ERROR(corruption_detected); /* impossible */ + } + } } typedef union { - FSE_decode_t realData; - U32 alignedBy4; + FSE_decode_t realData; + U32 alignedBy4; } FSE_decode_t4; static const FSE_decode_t4 LL_defaultDTable[(1< max) return ERROR(corruption_detected); - FSE_buildDTable_rle(DTableSpace, *(const BYTE*)src); - *DTablePtr = DTableSpace; - return 1; - case set_basic : - *DTablePtr = (const FSE_DTable*)tmpPtr; - return 0; - case set_repeat: - if (!flagRepeatTable) return ERROR(corruption_detected); - return 0; - default : /* impossible */ - case set_compressed : - { U32 tableLog; - S16 norm[MaxSeq+1]; - size_t const headerSize = FSE_readNCount(norm, &max, &tableLog, src, srcSize); - if (FSE_isError(headerSize)) return ERROR(corruption_detected); - if (tableLog > maxLog) return ERROR(corruption_detected); - FSE_buildDTable(DTableSpace, norm, max, tableLog); - *DTablePtr = DTableSpace; - return headerSize; - } } + const void* const tmpPtr = defaultTable; /* bypass strict aliasing */ + switch(type) + { + case set_rle : + if (!srcSize) return ERROR(srcSize_wrong); + if ( (*(const BYTE*)src) > max) return ERROR(corruption_detected); + FSE_buildDTable_rle(DTableSpace, *(const BYTE*)src); + *DTablePtr = DTableSpace; + return 1; + case set_basic : + *DTablePtr = (const FSE_DTable*)tmpPtr; + return 0; + case set_repeat: + if (!flagRepeatTable) return ERROR(corruption_detected); + return 0; + default : /* impossible */ + case set_compressed : + { U32 tableLog; + S16 norm[MaxSeq+1]; + size_t const headerSize = FSE_readNCount(norm, &max, &tableLog, src, srcSize); + if (FSE_isError(headerSize)) return ERROR(corruption_detected); + if (tableLog > maxLog) return ERROR(corruption_detected); + FSE_buildDTable(DTableSpace, norm, max, tableLog); + *DTablePtr = DTableSpace; + return headerSize; + } } } size_t ZSTD_decodeSeqHeaders(ZSTD_DCtx* dctx, int* nbSeqPtr, - const void* src, size_t srcSize) + const void* src, size_t srcSize) { - const BYTE* const istart = (const BYTE* const)src; - const BYTE* const iend = istart + srcSize; - const BYTE* ip = istart; + const BYTE* const istart = (const BYTE* const)src; + const BYTE* const iend = istart + srcSize; + const BYTE* ip = istart; - /* check */ - if (srcSize < MIN_SEQUENCES_SIZE) return ERROR(srcSize_wrong); + /* check */ + if (srcSize < MIN_SEQUENCES_SIZE) return ERROR(srcSize_wrong); - /* SeqHead */ - { int nbSeq = *ip++; - if (!nbSeq) { *nbSeqPtr=0; return 1; } - if (nbSeq > 0x7F) { - if (nbSeq == 0xFF) { - if (ip+2 > iend) return ERROR(srcSize_wrong); - nbSeq = MEM_readLE16(ip) + LONGNBSEQ, ip+=2; - } else { - if (ip >= iend) return ERROR(srcSize_wrong); - nbSeq = ((nbSeq-0x80)<<8) + *ip++; - } - } - *nbSeqPtr = nbSeq; - } + /* SeqHead */ + { int nbSeq = *ip++; + if (!nbSeq) { *nbSeqPtr=0; return 1; } + if (nbSeq > 0x7F) { + if (nbSeq == 0xFF) { + if (ip+2 > iend) return ERROR(srcSize_wrong); + nbSeq = MEM_readLE16(ip) + LONGNBSEQ, ip+=2; + } else { + if (ip >= iend) return ERROR(srcSize_wrong); + nbSeq = ((nbSeq-0x80)<<8) + *ip++; + } + } + *nbSeqPtr = nbSeq; + } - /* FSE table descriptors */ - if (ip+4 > iend) return ERROR(srcSize_wrong); /* minimum possible size */ - { symbolEncodingType_e const LLtype = (symbolEncodingType_e)(*ip >> 6); - symbolEncodingType_e const OFtype = (symbolEncodingType_e)((*ip >> 4) & 3); - symbolEncodingType_e const MLtype = (symbolEncodingType_e)((*ip >> 2) & 3); - ip++; + /* FSE table descriptors */ + if (ip+4 > iend) return ERROR(srcSize_wrong); /* minimum possible size */ + { symbolEncodingType_e const LLtype = (symbolEncodingType_e)(*ip >> 6); + symbolEncodingType_e const OFtype = (symbolEncodingType_e)((*ip >> 4) & 3); + symbolEncodingType_e const MLtype = (symbolEncodingType_e)((*ip >> 2) & 3); + ip++; - /* Build DTables */ - { size_t const llhSize = ZSTD_buildSeqTable(dctx->entropy.LLTable, &dctx->LLTptr, - LLtype, MaxLL, LLFSELog, - ip, iend-ip, LL_defaultDTable, dctx->fseEntropy); - if (ZSTD_isError(llhSize)) return ERROR(corruption_detected); - ip += llhSize; - } - { size_t const ofhSize = ZSTD_buildSeqTable(dctx->entropy.OFTable, &dctx->OFTptr, - OFtype, MaxOff, OffFSELog, - ip, iend-ip, OF_defaultDTable, dctx->fseEntropy); - if (ZSTD_isError(ofhSize)) return ERROR(corruption_detected); - ip += ofhSize; - } - { size_t const mlhSize = ZSTD_buildSeqTable(dctx->entropy.MLTable, &dctx->MLTptr, - MLtype, MaxML, MLFSELog, - ip, iend-ip, ML_defaultDTable, dctx->fseEntropy); - if (ZSTD_isError(mlhSize)) return ERROR(corruption_detected); - ip += mlhSize; - } - } + /* Build DTables */ + { size_t const llhSize = ZSTD_buildSeqTable(dctx->entropy.LLTable, &dctx->LLTptr, + LLtype, MaxLL, LLFSELog, + ip, iend-ip, LL_defaultDTable, dctx->fseEntropy); + if (ZSTD_isError(llhSize)) return ERROR(corruption_detected); + ip += llhSize; + } + { size_t const ofhSize = ZSTD_buildSeqTable(dctx->entropy.OFTable, &dctx->OFTptr, + OFtype, MaxOff, OffFSELog, + ip, iend-ip, OF_defaultDTable, dctx->fseEntropy); + if (ZSTD_isError(ofhSize)) return ERROR(corruption_detected); + ip += ofhSize; + } + { size_t const mlhSize = ZSTD_buildSeqTable(dctx->entropy.MLTable, &dctx->MLTptr, + MLtype, MaxML, MLFSELog, + ip, iend-ip, ML_defaultDTable, dctx->fseEntropy); + if (ZSTD_isError(mlhSize)) return ERROR(corruption_detected); + ip += mlhSize; + } + } - return ip-istart; + return ip-istart; } typedef struct { - size_t litLength; - size_t matchLength; - size_t offset; - const BYTE* match; + size_t litLength; + size_t matchLength; + size_t offset; + const BYTE* match; } seq_t; typedef struct { - BIT_DStream_t DStream; - FSE_DState_t stateLL; - FSE_DState_t stateOffb; - FSE_DState_t stateML; - size_t prevOffset[ZSTD_REP_NUM]; - const BYTE* base; - size_t pos; - uPtrDiff gotoDict; + BIT_DStream_t DStream; + FSE_DState_t stateLL; + FSE_DState_t stateOffb; + FSE_DState_t stateML; + size_t prevOffset[ZSTD_REP_NUM]; + const BYTE* base; + size_t pos; + uPtrDiff gotoDict; } seqState_t; FORCE_NOINLINE size_t ZSTD_execSequenceLast7(BYTE* op, - BYTE* const oend, seq_t sequence, - const BYTE** litPtr, const BYTE* const litLimit, - const BYTE* const base, const BYTE* const vBase, const BYTE* const dictEnd) + BYTE* const oend, seq_t sequence, + const BYTE** litPtr, const BYTE* const litLimit, + const BYTE* const base, const BYTE* const vBase, const BYTE* const dictEnd) { - BYTE* const oLitEnd = op + sequence.litLength; - size_t const sequenceLength = sequence.litLength + sequence.matchLength; - BYTE* const oMatchEnd = op + sequenceLength; /* risk : address space overflow (32-bits) */ - BYTE* const oend_w = oend - WILDCOPY_OVERLENGTH; - const BYTE* const iLitEnd = *litPtr + sequence.litLength; - const BYTE* match = oLitEnd - sequence.offset; + BYTE* const oLitEnd = op + sequence.litLength; + size_t const sequenceLength = sequence.litLength + sequence.matchLength; + BYTE* const oMatchEnd = op + sequenceLength; /* risk : address space overflow (32-bits) */ + BYTE* const oend_w = oend - WILDCOPY_OVERLENGTH; + const BYTE* const iLitEnd = *litPtr + sequence.litLength; + const BYTE* match = oLitEnd - sequence.offset; - /* check */ - if (oMatchEnd>oend) return ERROR(dstSize_tooSmall); /* last match must start at a minimum distance of WILDCOPY_OVERLENGTH from oend */ - if (iLitEnd > litLimit) return ERROR(corruption_detected); /* over-read beyond lit buffer */ - if (oLitEnd <= oend_w) return ERROR(GENERIC); /* Precondition */ + /* check */ + if (oMatchEnd>oend) return ERROR(dstSize_tooSmall); /* last match must start at a minimum distance of WILDCOPY_OVERLENGTH from oend */ + if (iLitEnd > litLimit) return ERROR(corruption_detected); /* over-read beyond lit buffer */ + if (oLitEnd <= oend_w) return ERROR(GENERIC); /* Precondition */ - /* copy literals */ - if (op < oend_w) { - ZSTD_wildcopy(op, *litPtr, oend_w - op); - *litPtr += oend_w - op; - op = oend_w; - } - while (op < oLitEnd) *op++ = *(*litPtr)++; + /* copy literals */ + if (op < oend_w) { + ZSTD_wildcopy(op, *litPtr, oend_w - op); + *litPtr += oend_w - op; + op = oend_w; + } + while (op < oLitEnd) *op++ = *(*litPtr)++; - /* copy Match */ - if (sequence.offset > (size_t)(oLitEnd - base)) { - /* offset beyond prefix */ - if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); - match = dictEnd - (base-match); - if (match + sequence.matchLength <= dictEnd) { - memmove(oLitEnd, match, sequence.matchLength); - return sequenceLength; - } - /* span extDict & currentPrefixSegment */ - { size_t const length1 = dictEnd - match; - memmove(oLitEnd, match, length1); - op = oLitEnd + length1; - sequence.matchLength -= length1; - match = base; - } } - while (op < oMatchEnd) *op++ = *match++; - return sequenceLength; + /* copy Match */ + if (sequence.offset > (size_t)(oLitEnd - base)) { + /* offset beyond prefix */ + if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); + match = dictEnd - (base-match); + if (match + sequence.matchLength <= dictEnd) { + memmove(oLitEnd, match, sequence.matchLength); + return sequenceLength; + } + /* span extDict & currentPrefixSegment */ + { size_t const length1 = dictEnd - match; + memmove(oLitEnd, match, length1); + op = oLitEnd + length1; + sequence.matchLength -= length1; + match = base; + } } + while (op < oMatchEnd) *op++ = *match++; + return sequenceLength; } @@ -931,538 +931,538 @@ size_t ZSTD_execSequenceLast7(BYTE* op, static seq_t ZSTD_decodeSequence(seqState_t* seqState) { - seq_t seq; + seq_t seq; - U32 const llCode = FSE_peekSymbol(&seqState->stateLL); - U32 const mlCode = FSE_peekSymbol(&seqState->stateML); - U32 const ofCode = FSE_peekSymbol(&seqState->stateOffb); /* <= maxOff, by table construction */ + U32 const llCode = FSE_peekSymbol(&seqState->stateLL); + U32 const mlCode = FSE_peekSymbol(&seqState->stateML); + U32 const ofCode = FSE_peekSymbol(&seqState->stateOffb); /* <= maxOff, by table construction */ - U32 const llBits = LL_bits[llCode]; - U32 const mlBits = ML_bits[mlCode]; - U32 const ofBits = ofCode; - U32 const totalBits = llBits+mlBits+ofBits; + U32 const llBits = LL_bits[llCode]; + U32 const mlBits = ML_bits[mlCode]; + U32 const ofBits = ofCode; + U32 const totalBits = llBits+mlBits+ofBits; - static const U32 LL_base[MaxLL+1] = { - 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, - 16, 18, 20, 22, 24, 28, 32, 40, 48, 64, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000, - 0x2000, 0x4000, 0x8000, 0x10000 }; + static const U32 LL_base[MaxLL+1] = { + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, + 16, 18, 20, 22, 24, 28, 32, 40, 48, 64, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000, + 0x2000, 0x4000, 0x8000, 0x10000 }; - static const U32 ML_base[MaxML+1] = { - 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, 37, 39, 41, 43, 47, 51, 59, 67, 83, 99, 0x83, 0x103, 0x203, 0x403, 0x803, - 0x1003, 0x2003, 0x4003, 0x8003, 0x10003 }; + static const U32 ML_base[MaxML+1] = { + 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, 37, 39, 41, 43, 47, 51, 59, 67, 83, 99, 0x83, 0x103, 0x203, 0x403, 0x803, + 0x1003, 0x2003, 0x4003, 0x8003, 0x10003 }; - static const U32 OF_base[MaxOff+1] = { - 0, 1, 1, 5, 0xD, 0x1D, 0x3D, 0x7D, - 0xFD, 0x1FD, 0x3FD, 0x7FD, 0xFFD, 0x1FFD, 0x3FFD, 0x7FFD, - 0xFFFD, 0x1FFFD, 0x3FFFD, 0x7FFFD, 0xFFFFD, 0x1FFFFD, 0x3FFFFD, 0x7FFFFD, - 0xFFFFFD, 0x1FFFFFD, 0x3FFFFFD, 0x7FFFFFD, 0xFFFFFFD }; + static const U32 OF_base[MaxOff+1] = { + 0, 1, 1, 5, 0xD, 0x1D, 0x3D, 0x7D, + 0xFD, 0x1FD, 0x3FD, 0x7FD, 0xFFD, 0x1FFD, 0x3FFD, 0x7FFD, + 0xFFFD, 0x1FFFD, 0x3FFFD, 0x7FFFD, 0xFFFFD, 0x1FFFFD, 0x3FFFFD, 0x7FFFFD, + 0xFFFFFD, 0x1FFFFFD, 0x3FFFFFD, 0x7FFFFFD, 0xFFFFFFD }; - /* sequence */ - { size_t offset; - if (!ofCode) - offset = 0; - else { - offset = OF_base[ofCode] + BIT_readBitsFast(&seqState->DStream, ofBits); /* <= (ZSTD_WINDOWLOG_MAX-1) bits */ - if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); - } + /* sequence */ + { size_t offset; + if (!ofCode) + offset = 0; + else { + offset = OF_base[ofCode] + BIT_readBitsFast(&seqState->DStream, ofBits); /* <= (ZSTD_WINDOWLOG_MAX-1) bits */ + if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); + } - if (ofCode <= 1) { - offset += (llCode==0); - if (offset) { - size_t temp = (offset==3) ? seqState->prevOffset[0] - 1 : seqState->prevOffset[offset]; - temp += !temp; /* 0 is not valid; input is corrupted; force offset to 1 */ - if (offset != 1) seqState->prevOffset[2] = seqState->prevOffset[1]; - seqState->prevOffset[1] = seqState->prevOffset[0]; - seqState->prevOffset[0] = offset = temp; - } else { - offset = seqState->prevOffset[0]; - } - } else { - seqState->prevOffset[2] = seqState->prevOffset[1]; - seqState->prevOffset[1] = seqState->prevOffset[0]; - seqState->prevOffset[0] = offset; - } - seq.offset = offset; - } + if (ofCode <= 1) { + offset += (llCode==0); + if (offset) { + size_t temp = (offset==3) ? seqState->prevOffset[0] - 1 : seqState->prevOffset[offset]; + temp += !temp; /* 0 is not valid; input is corrupted; force offset to 1 */ + if (offset != 1) seqState->prevOffset[2] = seqState->prevOffset[1]; + seqState->prevOffset[1] = seqState->prevOffset[0]; + seqState->prevOffset[0] = offset = temp; + } else { + offset = seqState->prevOffset[0]; + } + } else { + seqState->prevOffset[2] = seqState->prevOffset[1]; + seqState->prevOffset[1] = seqState->prevOffset[0]; + seqState->prevOffset[0] = offset; + } + seq.offset = offset; + } - seq.matchLength = ML_base[mlCode] + ((mlCode>31) ? BIT_readBitsFast(&seqState->DStream, mlBits) : 0); /* <= 16 bits */ - if (MEM_32bits() && (mlBits+llBits>24)) BIT_reloadDStream(&seqState->DStream); + seq.matchLength = ML_base[mlCode] + ((mlCode>31) ? BIT_readBitsFast(&seqState->DStream, mlBits) : 0); /* <= 16 bits */ + if (MEM_32bits() && (mlBits+llBits>24)) BIT_reloadDStream(&seqState->DStream); - seq.litLength = LL_base[llCode] + ((llCode>15) ? BIT_readBitsFast(&seqState->DStream, llBits) : 0); /* <= 16 bits */ - if (MEM_32bits() || - (totalBits > 64 - 7 - (LLFSELog+MLFSELog+OffFSELog)) ) BIT_reloadDStream(&seqState->DStream); + seq.litLength = LL_base[llCode] + ((llCode>15) ? BIT_readBitsFast(&seqState->DStream, llBits) : 0); /* <= 16 bits */ + if (MEM_32bits() || + (totalBits > 64 - 7 - (LLFSELog+MLFSELog+OffFSELog)) ) BIT_reloadDStream(&seqState->DStream); - /* ANS state update */ - FSE_updateState(&seqState->stateLL, &seqState->DStream); /* <= 9 bits */ - FSE_updateState(&seqState->stateML, &seqState->DStream); /* <= 9 bits */ - if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); /* <= 18 bits */ - FSE_updateState(&seqState->stateOffb, &seqState->DStream); /* <= 8 bits */ + /* ANS state update */ + FSE_updateState(&seqState->stateLL, &seqState->DStream); /* <= 9 bits */ + FSE_updateState(&seqState->stateML, &seqState->DStream); /* <= 9 bits */ + if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); /* <= 18 bits */ + FSE_updateState(&seqState->stateOffb, &seqState->DStream); /* <= 8 bits */ - return seq; + return seq; } FORCE_INLINE size_t ZSTD_execSequence(BYTE* op, - BYTE* const oend, seq_t sequence, - const BYTE** litPtr, const BYTE* const litLimit, - const BYTE* const base, const BYTE* const vBase, const BYTE* const dictEnd) + BYTE* const oend, seq_t sequence, + const BYTE** litPtr, const BYTE* const litLimit, + const BYTE* const base, const BYTE* const vBase, const BYTE* const dictEnd) { - BYTE* const oLitEnd = op + sequence.litLength; - size_t const sequenceLength = sequence.litLength + sequence.matchLength; - BYTE* const oMatchEnd = op + sequenceLength; /* risk : address space overflow (32-bits) */ - BYTE* const oend_w = oend - WILDCOPY_OVERLENGTH; - const BYTE* const iLitEnd = *litPtr + sequence.litLength; - const BYTE* match = oLitEnd - sequence.offset; + BYTE* const oLitEnd = op + sequence.litLength; + size_t const sequenceLength = sequence.litLength + sequence.matchLength; + BYTE* const oMatchEnd = op + sequenceLength; /* risk : address space overflow (32-bits) */ + BYTE* const oend_w = oend - WILDCOPY_OVERLENGTH; + const BYTE* const iLitEnd = *litPtr + sequence.litLength; + const BYTE* match = oLitEnd - sequence.offset; - /* check */ - if (oMatchEnd>oend) return ERROR(dstSize_tooSmall); /* last match must start at a minimum distance of WILDCOPY_OVERLENGTH from oend */ - if (iLitEnd > litLimit) return ERROR(corruption_detected); /* over-read beyond lit buffer */ - if (oLitEnd>oend_w) return ZSTD_execSequenceLast7(op, oend, sequence, litPtr, litLimit, base, vBase, dictEnd); + /* check */ + if (oMatchEnd>oend) return ERROR(dstSize_tooSmall); /* last match must start at a minimum distance of WILDCOPY_OVERLENGTH from oend */ + if (iLitEnd > litLimit) return ERROR(corruption_detected); /* over-read beyond lit buffer */ + if (oLitEnd>oend_w) return ZSTD_execSequenceLast7(op, oend, sequence, litPtr, litLimit, base, vBase, dictEnd); - /* copy Literals */ - ZSTD_copy8(op, *litPtr); - if (sequence.litLength > 8) - ZSTD_wildcopy(op+8, (*litPtr)+8, sequence.litLength - 8); /* note : since oLitEnd <= oend-WILDCOPY_OVERLENGTH, no risk of overwrite beyond oend */ - op = oLitEnd; - *litPtr = iLitEnd; /* update for next sequence */ + /* copy Literals */ + ZSTD_copy8(op, *litPtr); + if (sequence.litLength > 8) + ZSTD_wildcopy(op+8, (*litPtr)+8, sequence.litLength - 8); /* note : since oLitEnd <= oend-WILDCOPY_OVERLENGTH, no risk of overwrite beyond oend */ + op = oLitEnd; + *litPtr = iLitEnd; /* update for next sequence */ - /* copy Match */ - if (sequence.offset > (size_t)(oLitEnd - base)) { - /* offset beyond prefix */ - if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); - match = dictEnd + (match - base); - if (match + sequence.matchLength <= dictEnd) { - memmove(oLitEnd, match, sequence.matchLength); - return sequenceLength; - } - /* span extDict & currentPrefixSegment */ - { size_t const length1 = dictEnd - match; - memmove(oLitEnd, match, length1); - op = oLitEnd + length1; - sequence.matchLength -= length1; - match = base; - if (op > oend_w || sequence.matchLength < MINMATCH) { - U32 i; - for (i = 0; i < sequence.matchLength; ++i) op[i] = match[i]; - return sequenceLength; - } - } } - /* Requirement: op <= oend_w && sequence.matchLength >= MINMATCH */ + /* copy Match */ + if (sequence.offset > (size_t)(oLitEnd - base)) { + /* offset beyond prefix */ + if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); + match = dictEnd + (match - base); + if (match + sequence.matchLength <= dictEnd) { + memmove(oLitEnd, match, sequence.matchLength); + return sequenceLength; + } + /* span extDict & currentPrefixSegment */ + { size_t const length1 = dictEnd - match; + memmove(oLitEnd, match, length1); + op = oLitEnd + length1; + sequence.matchLength -= length1; + match = base; + if (op > oend_w || sequence.matchLength < MINMATCH) { + U32 i; + for (i = 0; i < sequence.matchLength; ++i) op[i] = match[i]; + return sequenceLength; + } + } } + /* Requirement: op <= oend_w && sequence.matchLength >= MINMATCH */ - /* match within prefix */ - if (sequence.offset < 8) { - /* close range match, overlap */ - static const U32 dec32table[] = { 0, 1, 2, 1, 4, 4, 4, 4 }; /* added */ - static const int dec64table[] = { 8, 8, 8, 7, 8, 9,10,11 }; /* subtracted */ - int const sub2 = dec64table[sequence.offset]; - op[0] = match[0]; - op[1] = match[1]; - op[2] = match[2]; - op[3] = match[3]; - match += dec32table[sequence.offset]; - ZSTD_copy4(op+4, match); - match -= sub2; - } else { - ZSTD_copy8(op, match); - } - op += 8; match += 8; + /* match within prefix */ + if (sequence.offset < 8) { + /* close range match, overlap */ + static const U32 dec32table[] = { 0, 1, 2, 1, 4, 4, 4, 4 }; /* added */ + static const int dec64table[] = { 8, 8, 8, 7, 8, 9,10,11 }; /* subtracted */ + int const sub2 = dec64table[sequence.offset]; + op[0] = match[0]; + op[1] = match[1]; + op[2] = match[2]; + op[3] = match[3]; + match += dec32table[sequence.offset]; + ZSTD_copy4(op+4, match); + match -= sub2; + } else { + ZSTD_copy8(op, match); + } + op += 8; match += 8; - if (oMatchEnd > oend-(16-MINMATCH)) { - if (op < oend_w) { - ZSTD_wildcopy(op, match, oend_w - op); - match += oend_w - op; - op = oend_w; - } - while (op < oMatchEnd) *op++ = *match++; - } else { - ZSTD_wildcopy(op, match, (ptrdiff_t)sequence.matchLength-8); /* works even if matchLength < 8 */ - } - return sequenceLength; + if (oMatchEnd > oend-(16-MINMATCH)) { + if (op < oend_w) { + ZSTD_wildcopy(op, match, oend_w - op); + match += oend_w - op; + op = oend_w; + } + while (op < oMatchEnd) *op++ = *match++; + } else { + ZSTD_wildcopy(op, match, (ptrdiff_t)sequence.matchLength-8); /* works even if matchLength < 8 */ + } + return sequenceLength; } static size_t ZSTD_decompressSequences( - ZSTD_DCtx* dctx, - void* dst, size_t maxDstSize, - const void* seqStart, size_t seqSize) + ZSTD_DCtx* dctx, + void* dst, size_t maxDstSize, + const void* seqStart, size_t seqSize) { - const BYTE* ip = (const BYTE*)seqStart; - const BYTE* const iend = ip + seqSize; - BYTE* const ostart = (BYTE* const)dst; - BYTE* const oend = ostart + maxDstSize; - BYTE* op = ostart; - const BYTE* litPtr = dctx->litPtr; - const BYTE* const litEnd = litPtr + dctx->litSize; - const BYTE* const base = (const BYTE*) (dctx->base); - const BYTE* const vBase = (const BYTE*) (dctx->vBase); - const BYTE* const dictEnd = (const BYTE*) (dctx->dictEnd); - int nbSeq; + const BYTE* ip = (const BYTE*)seqStart; + const BYTE* const iend = ip + seqSize; + BYTE* const ostart = (BYTE* const)dst; + BYTE* const oend = ostart + maxDstSize; + BYTE* op = ostart; + const BYTE* litPtr = dctx->litPtr; + const BYTE* const litEnd = litPtr + dctx->litSize; + const BYTE* const base = (const BYTE*) (dctx->base); + const BYTE* const vBase = (const BYTE*) (dctx->vBase); + const BYTE* const dictEnd = (const BYTE*) (dctx->dictEnd); + int nbSeq; - /* Build Decoding Tables */ - { size_t const seqHSize = ZSTD_decodeSeqHeaders(dctx, &nbSeq, ip, seqSize); - if (ZSTD_isError(seqHSize)) return seqHSize; - ip += seqHSize; - } + /* Build Decoding Tables */ + { size_t const seqHSize = ZSTD_decodeSeqHeaders(dctx, &nbSeq, ip, seqSize); + if (ZSTD_isError(seqHSize)) return seqHSize; + ip += seqHSize; + } - /* Regen sequences */ - if (nbSeq) { - seqState_t seqState; - dctx->fseEntropy = 1; - { U32 i; for (i=0; ientropy.rep[i]; } - CHECK_E(BIT_initDStream(&seqState.DStream, ip, iend-ip), corruption_detected); - FSE_initDState(&seqState.stateLL, &seqState.DStream, dctx->LLTptr); - FSE_initDState(&seqState.stateOffb, &seqState.DStream, dctx->OFTptr); - FSE_initDState(&seqState.stateML, &seqState.DStream, dctx->MLTptr); + /* Regen sequences */ + if (nbSeq) { + seqState_t seqState; + dctx->fseEntropy = 1; + { U32 i; for (i=0; ientropy.rep[i]; } + CHECK_E(BIT_initDStream(&seqState.DStream, ip, iend-ip), corruption_detected); + FSE_initDState(&seqState.stateLL, &seqState.DStream, dctx->LLTptr); + FSE_initDState(&seqState.stateOffb, &seqState.DStream, dctx->OFTptr); + FSE_initDState(&seqState.stateML, &seqState.DStream, dctx->MLTptr); - for ( ; (BIT_reloadDStream(&(seqState.DStream)) <= BIT_DStream_completed) && nbSeq ; ) { - nbSeq--; - { seq_t const sequence = ZSTD_decodeSequence(&seqState); - size_t const oneSeqSize = ZSTD_execSequence(op, oend, sequence, &litPtr, litEnd, base, vBase, dictEnd); - if (ZSTD_isError(oneSeqSize)) return oneSeqSize; - op += oneSeqSize; - } } + for ( ; (BIT_reloadDStream(&(seqState.DStream)) <= BIT_DStream_completed) && nbSeq ; ) { + nbSeq--; + { seq_t const sequence = ZSTD_decodeSequence(&seqState); + size_t const oneSeqSize = ZSTD_execSequence(op, oend, sequence, &litPtr, litEnd, base, vBase, dictEnd); + if (ZSTD_isError(oneSeqSize)) return oneSeqSize; + op += oneSeqSize; + } } - /* check if reached exact end */ - if (nbSeq) return ERROR(corruption_detected); - /* save reps for next block */ - { U32 i; for (i=0; ientropy.rep[i] = (U32)(seqState.prevOffset[i]); } - } + /* check if reached exact end */ + if (nbSeq) return ERROR(corruption_detected); + /* save reps for next block */ + { U32 i; for (i=0; ientropy.rep[i] = (U32)(seqState.prevOffset[i]); } + } - /* last literal segment */ - { size_t const lastLLSize = litEnd - litPtr; - if (lastLLSize > (size_t)(oend-op)) return ERROR(dstSize_tooSmall); - memcpy(op, litPtr, lastLLSize); - op += lastLLSize; - } + /* last literal segment */ + { size_t const lastLLSize = litEnd - litPtr; + if (lastLLSize > (size_t)(oend-op)) return ERROR(dstSize_tooSmall); + memcpy(op, litPtr, lastLLSize); + op += lastLLSize; + } - return op-ostart; + return op-ostart; } FORCE_INLINE seq_t ZSTD_decodeSequenceLong_generic(seqState_t* seqState, int const longOffsets) { - seq_t seq; + seq_t seq; - U32 const llCode = FSE_peekSymbol(&seqState->stateLL); - U32 const mlCode = FSE_peekSymbol(&seqState->stateML); - U32 const ofCode = FSE_peekSymbol(&seqState->stateOffb); /* <= maxOff, by table construction */ + U32 const llCode = FSE_peekSymbol(&seqState->stateLL); + U32 const mlCode = FSE_peekSymbol(&seqState->stateML); + U32 const ofCode = FSE_peekSymbol(&seqState->stateOffb); /* <= maxOff, by table construction */ - U32 const llBits = LL_bits[llCode]; - U32 const mlBits = ML_bits[mlCode]; - U32 const ofBits = ofCode; - U32 const totalBits = llBits+mlBits+ofBits; + U32 const llBits = LL_bits[llCode]; + U32 const mlBits = ML_bits[mlCode]; + U32 const ofBits = ofCode; + U32 const totalBits = llBits+mlBits+ofBits; - static const U32 LL_base[MaxLL+1] = { - 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, - 16, 18, 20, 22, 24, 28, 32, 40, 48, 64, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000, - 0x2000, 0x4000, 0x8000, 0x10000 }; + static const U32 LL_base[MaxLL+1] = { + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, + 16, 18, 20, 22, 24, 28, 32, 40, 48, 64, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000, + 0x2000, 0x4000, 0x8000, 0x10000 }; - static const U32 ML_base[MaxML+1] = { - 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, 37, 39, 41, 43, 47, 51, 59, 67, 83, 99, 0x83, 0x103, 0x203, 0x403, 0x803, - 0x1003, 0x2003, 0x4003, 0x8003, 0x10003 }; + static const U32 ML_base[MaxML+1] = { + 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, 37, 39, 41, 43, 47, 51, 59, 67, 83, 99, 0x83, 0x103, 0x203, 0x403, 0x803, + 0x1003, 0x2003, 0x4003, 0x8003, 0x10003 }; - static const U32 OF_base[MaxOff+1] = { - 0, 1, 1, 5, 0xD, 0x1D, 0x3D, 0x7D, - 0xFD, 0x1FD, 0x3FD, 0x7FD, 0xFFD, 0x1FFD, 0x3FFD, 0x7FFD, - 0xFFFD, 0x1FFFD, 0x3FFFD, 0x7FFFD, 0xFFFFD, 0x1FFFFD, 0x3FFFFD, 0x7FFFFD, - 0xFFFFFD, 0x1FFFFFD, 0x3FFFFFD, 0x7FFFFFD, 0xFFFFFFD }; + static const U32 OF_base[MaxOff+1] = { + 0, 1, 1, 5, 0xD, 0x1D, 0x3D, 0x7D, + 0xFD, 0x1FD, 0x3FD, 0x7FD, 0xFFD, 0x1FFD, 0x3FFD, 0x7FFD, + 0xFFFD, 0x1FFFD, 0x3FFFD, 0x7FFFD, 0xFFFFD, 0x1FFFFD, 0x3FFFFD, 0x7FFFFD, + 0xFFFFFD, 0x1FFFFFD, 0x3FFFFFD, 0x7FFFFFD, 0xFFFFFFD }; - /* sequence */ - { size_t offset; - if (!ofCode) - offset = 0; - else { - if (longOffsets) { - int const extraBits = ofBits - MIN(ofBits, STREAM_ACCUMULATOR_MIN); - offset = OF_base[ofCode] + (BIT_readBitsFast(&seqState->DStream, ofBits - extraBits) << extraBits); - if (MEM_32bits() || extraBits) BIT_reloadDStream(&seqState->DStream); - if (extraBits) offset += BIT_readBitsFast(&seqState->DStream, extraBits); - } else { - offset = OF_base[ofCode] + BIT_readBitsFast(&seqState->DStream, ofBits); /* <= (ZSTD_WINDOWLOG_MAX-1) bits */ - if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); - } - } + /* sequence */ + { size_t offset; + if (!ofCode) + offset = 0; + else { + if (longOffsets) { + int const extraBits = ofBits - MIN(ofBits, STREAM_ACCUMULATOR_MIN); + offset = OF_base[ofCode] + (BIT_readBitsFast(&seqState->DStream, ofBits - extraBits) << extraBits); + if (MEM_32bits() || extraBits) BIT_reloadDStream(&seqState->DStream); + if (extraBits) offset += BIT_readBitsFast(&seqState->DStream, extraBits); + } else { + offset = OF_base[ofCode] + BIT_readBitsFast(&seqState->DStream, ofBits); /* <= (ZSTD_WINDOWLOG_MAX-1) bits */ + if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); + } + } - if (ofCode <= 1) { - offset += (llCode==0); - if (offset) { - size_t temp = (offset==3) ? seqState->prevOffset[0] - 1 : seqState->prevOffset[offset]; - temp += !temp; /* 0 is not valid; input is corrupted; force offset to 1 */ - if (offset != 1) seqState->prevOffset[2] = seqState->prevOffset[1]; - seqState->prevOffset[1] = seqState->prevOffset[0]; - seqState->prevOffset[0] = offset = temp; - } else { - offset = seqState->prevOffset[0]; - } - } else { - seqState->prevOffset[2] = seqState->prevOffset[1]; - seqState->prevOffset[1] = seqState->prevOffset[0]; - seqState->prevOffset[0] = offset; - } - seq.offset = offset; - } + if (ofCode <= 1) { + offset += (llCode==0); + if (offset) { + size_t temp = (offset==3) ? seqState->prevOffset[0] - 1 : seqState->prevOffset[offset]; + temp += !temp; /* 0 is not valid; input is corrupted; force offset to 1 */ + if (offset != 1) seqState->prevOffset[2] = seqState->prevOffset[1]; + seqState->prevOffset[1] = seqState->prevOffset[0]; + seqState->prevOffset[0] = offset = temp; + } else { + offset = seqState->prevOffset[0]; + } + } else { + seqState->prevOffset[2] = seqState->prevOffset[1]; + seqState->prevOffset[1] = seqState->prevOffset[0]; + seqState->prevOffset[0] = offset; + } + seq.offset = offset; + } - seq.matchLength = ML_base[mlCode] + ((mlCode>31) ? BIT_readBitsFast(&seqState->DStream, mlBits) : 0); /* <= 16 bits */ - if (MEM_32bits() && (mlBits+llBits>24)) BIT_reloadDStream(&seqState->DStream); + seq.matchLength = ML_base[mlCode] + ((mlCode>31) ? BIT_readBitsFast(&seqState->DStream, mlBits) : 0); /* <= 16 bits */ + if (MEM_32bits() && (mlBits+llBits>24)) BIT_reloadDStream(&seqState->DStream); - seq.litLength = LL_base[llCode] + ((llCode>15) ? BIT_readBitsFast(&seqState->DStream, llBits) : 0); /* <= 16 bits */ - if (MEM_32bits() || - (totalBits > 64 - 7 - (LLFSELog+MLFSELog+OffFSELog)) ) BIT_reloadDStream(&seqState->DStream); + seq.litLength = LL_base[llCode] + ((llCode>15) ? BIT_readBitsFast(&seqState->DStream, llBits) : 0); /* <= 16 bits */ + if (MEM_32bits() || + (totalBits > 64 - 7 - (LLFSELog+MLFSELog+OffFSELog)) ) BIT_reloadDStream(&seqState->DStream); - { size_t const pos = seqState->pos + seq.litLength; - seq.match = seqState->base + pos - seq.offset; /* single memory segment */ - if (seq.offset > pos) seq.match += seqState->gotoDict; /* separate memory segment */ - seqState->pos = pos + seq.matchLength; - } + { size_t const pos = seqState->pos + seq.litLength; + seq.match = seqState->base + pos - seq.offset; /* single memory segment */ + if (seq.offset > pos) seq.match += seqState->gotoDict; /* separate memory segment */ + seqState->pos = pos + seq.matchLength; + } - /* ANS state update */ - FSE_updateState(&seqState->stateLL, &seqState->DStream); /* <= 9 bits */ - FSE_updateState(&seqState->stateML, &seqState->DStream); /* <= 9 bits */ - if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); /* <= 18 bits */ - FSE_updateState(&seqState->stateOffb, &seqState->DStream); /* <= 8 bits */ + /* ANS state update */ + FSE_updateState(&seqState->stateLL, &seqState->DStream); /* <= 9 bits */ + FSE_updateState(&seqState->stateML, &seqState->DStream); /* <= 9 bits */ + if (MEM_32bits()) BIT_reloadDStream(&seqState->DStream); /* <= 18 bits */ + FSE_updateState(&seqState->stateOffb, &seqState->DStream); /* <= 8 bits */ - return seq; + return seq; } static seq_t ZSTD_decodeSequenceLong(seqState_t* seqState, unsigned const windowSize) { - if (ZSTD_highbit32(windowSize) > STREAM_ACCUMULATOR_MIN) { - return ZSTD_decodeSequenceLong_generic(seqState, 1); - } else { - return ZSTD_decodeSequenceLong_generic(seqState, 0); - } + if (ZSTD_highbit32(windowSize) > STREAM_ACCUMULATOR_MIN) { + return ZSTD_decodeSequenceLong_generic(seqState, 1); + } else { + return ZSTD_decodeSequenceLong_generic(seqState, 0); + } } FORCE_INLINE size_t ZSTD_execSequenceLong(BYTE* op, - BYTE* const oend, seq_t sequence, - const BYTE** litPtr, const BYTE* const litLimit, - const BYTE* const base, const BYTE* const vBase, const BYTE* const dictEnd) + BYTE* const oend, seq_t sequence, + const BYTE** litPtr, const BYTE* const litLimit, + const BYTE* const base, const BYTE* const vBase, const BYTE* const dictEnd) { - BYTE* const oLitEnd = op + sequence.litLength; - size_t const sequenceLength = sequence.litLength + sequence.matchLength; - BYTE* const oMatchEnd = op + sequenceLength; /* risk : address space overflow (32-bits) */ - BYTE* const oend_w = oend - WILDCOPY_OVERLENGTH; - const BYTE* const iLitEnd = *litPtr + sequence.litLength; - const BYTE* match = sequence.match; + BYTE* const oLitEnd = op + sequence.litLength; + size_t const sequenceLength = sequence.litLength + sequence.matchLength; + BYTE* const oMatchEnd = op + sequenceLength; /* risk : address space overflow (32-bits) */ + BYTE* const oend_w = oend - WILDCOPY_OVERLENGTH; + const BYTE* const iLitEnd = *litPtr + sequence.litLength; + const BYTE* match = sequence.match; - /* check */ + /* check */ #if 1 - if (oMatchEnd>oend) return ERROR(dstSize_tooSmall); /* last match must start at a minimum distance of WILDCOPY_OVERLENGTH from oend */ - if (iLitEnd > litLimit) return ERROR(corruption_detected); /* over-read beyond lit buffer */ - if (oLitEnd>oend_w) return ZSTD_execSequenceLast7(op, oend, sequence, litPtr, litLimit, base, vBase, dictEnd); + if (oMatchEnd>oend) return ERROR(dstSize_tooSmall); /* last match must start at a minimum distance of WILDCOPY_OVERLENGTH from oend */ + if (iLitEnd > litLimit) return ERROR(corruption_detected); /* over-read beyond lit buffer */ + if (oLitEnd>oend_w) return ZSTD_execSequenceLast7(op, oend, sequence, litPtr, litLimit, base, vBase, dictEnd); #endif - /* copy Literals */ - ZSTD_copy8(op, *litPtr); - if (sequence.litLength > 8) - ZSTD_wildcopy(op+8, (*litPtr)+8, sequence.litLength - 8); /* note : since oLitEnd <= oend-WILDCOPY_OVERLENGTH, no risk of overwrite beyond oend */ - op = oLitEnd; - *litPtr = iLitEnd; /* update for next sequence */ + /* copy Literals */ + ZSTD_copy8(op, *litPtr); + if (sequence.litLength > 8) + ZSTD_wildcopy(op+8, (*litPtr)+8, sequence.litLength - 8); /* note : since oLitEnd <= oend-WILDCOPY_OVERLENGTH, no risk of overwrite beyond oend */ + op = oLitEnd; + *litPtr = iLitEnd; /* update for next sequence */ - /* copy Match */ + /* copy Match */ #if 1 - if (sequence.offset > (size_t)(oLitEnd - base)) { - /* offset beyond prefix */ - if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); - if (match + sequence.matchLength <= dictEnd) { - memmove(oLitEnd, match, sequence.matchLength); - return sequenceLength; - } - /* span extDict & currentPrefixSegment */ - { size_t const length1 = dictEnd - match; - memmove(oLitEnd, match, length1); - op = oLitEnd + length1; - sequence.matchLength -= length1; - match = base; - if (op > oend_w || sequence.matchLength < MINMATCH) { - U32 i; - for (i = 0; i < sequence.matchLength; ++i) op[i] = match[i]; - return sequenceLength; - } - } } - /* Requirement: op <= oend_w && sequence.matchLength >= MINMATCH */ + if (sequence.offset > (size_t)(oLitEnd - base)) { + /* offset beyond prefix */ + if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); + if (match + sequence.matchLength <= dictEnd) { + memmove(oLitEnd, match, sequence.matchLength); + return sequenceLength; + } + /* span extDict & currentPrefixSegment */ + { size_t const length1 = dictEnd - match; + memmove(oLitEnd, match, length1); + op = oLitEnd + length1; + sequence.matchLength -= length1; + match = base; + if (op > oend_w || sequence.matchLength < MINMATCH) { + U32 i; + for (i = 0; i < sequence.matchLength; ++i) op[i] = match[i]; + return sequenceLength; + } + } } + /* Requirement: op <= oend_w && sequence.matchLength >= MINMATCH */ #endif - /* match within prefix */ - if (sequence.offset < 8) { - /* close range match, overlap */ - static const U32 dec32table[] = { 0, 1, 2, 1, 4, 4, 4, 4 }; /* added */ - static const int dec64table[] = { 8, 8, 8, 7, 8, 9,10,11 }; /* subtracted */ - int const sub2 = dec64table[sequence.offset]; - op[0] = match[0]; - op[1] = match[1]; - op[2] = match[2]; - op[3] = match[3]; - match += dec32table[sequence.offset]; - ZSTD_copy4(op+4, match); - match -= sub2; - } else { - ZSTD_copy8(op, match); - } - op += 8; match += 8; + /* match within prefix */ + if (sequence.offset < 8) { + /* close range match, overlap */ + static const U32 dec32table[] = { 0, 1, 2, 1, 4, 4, 4, 4 }; /* added */ + static const int dec64table[] = { 8, 8, 8, 7, 8, 9,10,11 }; /* subtracted */ + int const sub2 = dec64table[sequence.offset]; + op[0] = match[0]; + op[1] = match[1]; + op[2] = match[2]; + op[3] = match[3]; + match += dec32table[sequence.offset]; + ZSTD_copy4(op+4, match); + match -= sub2; + } else { + ZSTD_copy8(op, match); + } + op += 8; match += 8; - if (oMatchEnd > oend-(16-MINMATCH)) { - if (op < oend_w) { - ZSTD_wildcopy(op, match, oend_w - op); - match += oend_w - op; - op = oend_w; - } - while (op < oMatchEnd) *op++ = *match++; - } else { - ZSTD_wildcopy(op, match, (ptrdiff_t)sequence.matchLength-8); /* works even if matchLength < 8 */ - } - return sequenceLength; + if (oMatchEnd > oend-(16-MINMATCH)) { + if (op < oend_w) { + ZSTD_wildcopy(op, match, oend_w - op); + match += oend_w - op; + op = oend_w; + } + while (op < oMatchEnd) *op++ = *match++; + } else { + ZSTD_wildcopy(op, match, (ptrdiff_t)sequence.matchLength-8); /* works even if matchLength < 8 */ + } + return sequenceLength; } static size_t ZSTD_decompressSequencesLong( - ZSTD_DCtx* dctx, - void* dst, size_t maxDstSize, - const void* seqStart, size_t seqSize) + ZSTD_DCtx* dctx, + void* dst, size_t maxDstSize, + const void* seqStart, size_t seqSize) { - const BYTE* ip = (const BYTE*)seqStart; - const BYTE* const iend = ip + seqSize; - BYTE* const ostart = (BYTE* const)dst; - BYTE* const oend = ostart + maxDstSize; - BYTE* op = ostart; - const BYTE* litPtr = dctx->litPtr; - const BYTE* const litEnd = litPtr + dctx->litSize; - const BYTE* const base = (const BYTE*) (dctx->base); - const BYTE* const vBase = (const BYTE*) (dctx->vBase); - const BYTE* const dictEnd = (const BYTE*) (dctx->dictEnd); - unsigned const windowSize = dctx->fParams.windowSize; - int nbSeq; + const BYTE* ip = (const BYTE*)seqStart; + const BYTE* const iend = ip + seqSize; + BYTE* const ostart = (BYTE* const)dst; + BYTE* const oend = ostart + maxDstSize; + BYTE* op = ostart; + const BYTE* litPtr = dctx->litPtr; + const BYTE* const litEnd = litPtr + dctx->litSize; + const BYTE* const base = (const BYTE*) (dctx->base); + const BYTE* const vBase = (const BYTE*) (dctx->vBase); + const BYTE* const dictEnd = (const BYTE*) (dctx->dictEnd); + unsigned const windowSize = dctx->fParams.windowSize; + int nbSeq; - /* Build Decoding Tables */ - { size_t const seqHSize = ZSTD_decodeSeqHeaders(dctx, &nbSeq, ip, seqSize); - if (ZSTD_isError(seqHSize)) return seqHSize; - ip += seqHSize; - } + /* Build Decoding Tables */ + { size_t const seqHSize = ZSTD_decodeSeqHeaders(dctx, &nbSeq, ip, seqSize); + if (ZSTD_isError(seqHSize)) return seqHSize; + ip += seqHSize; + } - /* Regen sequences */ - if (nbSeq) { + /* Regen sequences */ + if (nbSeq) { #define STORED_SEQS 4 #define STOSEQ_MASK (STORED_SEQS-1) #define ADVANCED_SEQS 4 - seq_t sequences[STORED_SEQS]; - int const seqAdvance = MIN(nbSeq, ADVANCED_SEQS); - seqState_t seqState; - int seqNb; - dctx->fseEntropy = 1; - { U32 i; for (i=0; ientropy.rep[i]; } - seqState.base = base; - seqState.pos = (size_t)(op-base); - seqState.gotoDict = (uPtrDiff)dictEnd - (uPtrDiff)base; /* cast to avoid undefined behaviour */ - CHECK_E(BIT_initDStream(&seqState.DStream, ip, iend-ip), corruption_detected); - FSE_initDState(&seqState.stateLL, &seqState.DStream, dctx->LLTptr); - FSE_initDState(&seqState.stateOffb, &seqState.DStream, dctx->OFTptr); - FSE_initDState(&seqState.stateML, &seqState.DStream, dctx->MLTptr); + seq_t sequences[STORED_SEQS]; + int const seqAdvance = MIN(nbSeq, ADVANCED_SEQS); + seqState_t seqState; + int seqNb; + dctx->fseEntropy = 1; + { U32 i; for (i=0; ientropy.rep[i]; } + seqState.base = base; + seqState.pos = (size_t)(op-base); + seqState.gotoDict = (uPtrDiff)dictEnd - (uPtrDiff)base; /* cast to avoid undefined behaviour */ + CHECK_E(BIT_initDStream(&seqState.DStream, ip, iend-ip), corruption_detected); + FSE_initDState(&seqState.stateLL, &seqState.DStream, dctx->LLTptr); + FSE_initDState(&seqState.stateOffb, &seqState.DStream, dctx->OFTptr); + FSE_initDState(&seqState.stateML, &seqState.DStream, dctx->MLTptr); - /* prepare in advance */ - for (seqNb=0; (BIT_reloadDStream(&seqState.DStream) <= BIT_DStream_completed) && seqNbentropy.rep[i] = (U32)(seqState.prevOffset[i]); } - } + /* save reps for next block */ + { U32 i; for (i=0; ientropy.rep[i] = (U32)(seqState.prevOffset[i]); } + } - /* last literal segment */ - { size_t const lastLLSize = litEnd - litPtr; - if (lastLLSize > (size_t)(oend-op)) return ERROR(dstSize_tooSmall); - memcpy(op, litPtr, lastLLSize); - op += lastLLSize; - } + /* last literal segment */ + { size_t const lastLLSize = litEnd - litPtr; + if (lastLLSize > (size_t)(oend-op)) return ERROR(dstSize_tooSmall); + memcpy(op, litPtr, lastLLSize); + op += lastLLSize; + } - return op-ostart; + return op-ostart; } static size_t ZSTD_decompressBlock_internal(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize) { /* blockType == blockCompressed */ - const BYTE* ip = (const BYTE*)src; + const BYTE* ip = (const BYTE*)src; - if (srcSize >= ZSTD_BLOCKSIZE_ABSOLUTEMAX) return ERROR(srcSize_wrong); + if (srcSize >= ZSTD_BLOCKSIZE_ABSOLUTEMAX) return ERROR(srcSize_wrong); - /* Decode literals section */ - { size_t const litCSize = ZSTD_decodeLiteralsBlock(dctx, src, srcSize); - if (ZSTD_isError(litCSize)) return litCSize; - ip += litCSize; - srcSize -= litCSize; - } - if (sizeof(size_t) > 4) /* do not enable prefetching on 32-bits x86, as it's performance detrimental */ - /* likely because of register pressure */ - /* if that's the correct cause, then 32-bits ARM should be affected differently */ - /* it would be good to test this on ARM real hardware, to see if prefetch version improves speed */ - if (dctx->fParams.windowSize > (1<<23)) - return ZSTD_decompressSequencesLong(dctx, dst, dstCapacity, ip, srcSize); - return ZSTD_decompressSequences(dctx, dst, dstCapacity, ip, srcSize); + /* Decode literals section */ + { size_t const litCSize = ZSTD_decodeLiteralsBlock(dctx, src, srcSize); + if (ZSTD_isError(litCSize)) return litCSize; + ip += litCSize; + srcSize -= litCSize; + } + if (sizeof(size_t) > 4) /* do not enable prefetching on 32-bits x86, as it's performance detrimental */ + /* likely because of register pressure */ + /* if that's the correct cause, then 32-bits ARM should be affected differently */ + /* it would be good to test this on ARM real hardware, to see if prefetch version improves speed */ + if (dctx->fParams.windowSize > (1<<23)) + return ZSTD_decompressSequencesLong(dctx, dst, dstCapacity, ip, srcSize); + return ZSTD_decompressSequences(dctx, dst, dstCapacity, ip, srcSize); } static void ZSTD_checkContinuity(ZSTD_DCtx* dctx, const void* dst) { - if (dst != dctx->previousDstEnd) { /* not contiguous */ - dctx->dictEnd = dctx->previousDstEnd; - dctx->vBase = (const char*)dst - ((const char*)(dctx->previousDstEnd) - (const char*)(dctx->base)); - dctx->base = dst; - dctx->previousDstEnd = dst; - } + if (dst != dctx->previousDstEnd) { /* not contiguous */ + dctx->dictEnd = dctx->previousDstEnd; + dctx->vBase = (const char*)dst - ((const char*)(dctx->previousDstEnd) - (const char*)(dctx->base)); + dctx->base = dst; + dctx->previousDstEnd = dst; + } } size_t ZSTD_decompressBlock(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize) { - size_t dSize; - ZSTD_checkContinuity(dctx, dst); - dSize = ZSTD_decompressBlock_internal(dctx, dst, dstCapacity, src, srcSize); - dctx->previousDstEnd = (char*)dst + dSize; - return dSize; + size_t dSize; + ZSTD_checkContinuity(dctx, dst); + dSize = ZSTD_decompressBlock_internal(dctx, dst, dstCapacity, src, srcSize); + dctx->previousDstEnd = (char*)dst + dSize; + return dSize; } /** ZSTD_insertBlock() : - insert `src` block into `dctx` history. Useful to track uncompressed blocks. */ + insert `src` block into `dctx` history. Useful to track uncompressed blocks. */ ZSTDLIB_API size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize) { - ZSTD_checkContinuity(dctx, blockStart); - dctx->previousDstEnd = (const char*)blockStart + blockSize; - return blockSize; + ZSTD_checkContinuity(dctx, blockStart); + dctx->previousDstEnd = (const char*)blockStart + blockSize; + return blockSize; } size_t ZSTD_generateNxBytes(void* dst, size_t dstCapacity, BYTE byte, size_t length) { - if (length > dstCapacity) return ERROR(dstSize_tooSmall); - memset(dst, byte, length); - return length; + if (length > dstCapacity) return ERROR(dstSize_tooSmall); + memset(dst, byte, length); + return length; } /** ZSTD_findFrameCompressedSize() : @@ -1473,241 +1473,241 @@ size_t ZSTD_generateNxBytes(void* dst, size_t dstCapacity, BYTE byte, size_t len size_t ZSTD_findFrameCompressedSize(const void *src, size_t srcSize) { #if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) - if (ZSTD_isLegacy(src, srcSize)) return ZSTD_findFrameCompressedSizeLegacy(src, srcSize); + if (ZSTD_isLegacy(src, srcSize)) return ZSTD_findFrameCompressedSizeLegacy(src, srcSize); #endif - if (srcSize >= ZSTD_skippableHeaderSize && - (MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { - return ZSTD_skippableHeaderSize + MEM_readLE32((const BYTE*)src + 4); - } else { - const BYTE* ip = (const BYTE*)src; - const BYTE* const ipstart = ip; - size_t remainingSize = srcSize; - ZSTD_frameParams fParams; + if (srcSize >= ZSTD_skippableHeaderSize && + (MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { + return ZSTD_skippableHeaderSize + MEM_readLE32((const BYTE*)src + 4); + } else { + const BYTE* ip = (const BYTE*)src; + const BYTE* const ipstart = ip; + size_t remainingSize = srcSize; + ZSTD_frameParams fParams; - size_t const headerSize = ZSTD_frameHeaderSize(ip, remainingSize); - if (ZSTD_isError(headerSize)) return headerSize; + size_t const headerSize = ZSTD_frameHeaderSize(ip, remainingSize); + if (ZSTD_isError(headerSize)) return headerSize; - /* Frame Header */ - { size_t const ret = ZSTD_getFrameParams(&fParams, ip, remainingSize); - if (ZSTD_isError(ret)) return ret; - if (ret > 0) return ERROR(srcSize_wrong); - } + /* Frame Header */ + { size_t const ret = ZSTD_getFrameParams(&fParams, ip, remainingSize); + if (ZSTD_isError(ret)) return ret; + if (ret > 0) return ERROR(srcSize_wrong); + } - ip += headerSize; - remainingSize -= headerSize; + ip += headerSize; + remainingSize -= headerSize; - /* Loop on each block */ - while (1) { - blockProperties_t blockProperties; - size_t const cBlockSize = ZSTD_getcBlockSize(ip, remainingSize, &blockProperties); - if (ZSTD_isError(cBlockSize)) return cBlockSize; + /* Loop on each block */ + while (1) { + blockProperties_t blockProperties; + size_t const cBlockSize = ZSTD_getcBlockSize(ip, remainingSize, &blockProperties); + if (ZSTD_isError(cBlockSize)) return cBlockSize; - if (ZSTD_blockHeaderSize + cBlockSize > remainingSize) return ERROR(srcSize_wrong); + if (ZSTD_blockHeaderSize + cBlockSize > remainingSize) return ERROR(srcSize_wrong); - ip += ZSTD_blockHeaderSize + cBlockSize; - remainingSize -= ZSTD_blockHeaderSize + cBlockSize; + ip += ZSTD_blockHeaderSize + cBlockSize; + remainingSize -= ZSTD_blockHeaderSize + cBlockSize; - if (blockProperties.lastBlock) break; - } + if (blockProperties.lastBlock) break; + } - if (fParams.checksumFlag) { /* Frame content checksum */ - if (remainingSize < 4) return ERROR(srcSize_wrong); - ip += 4; - remainingSize -= 4; - } + if (fParams.checksumFlag) { /* Frame content checksum */ + if (remainingSize < 4) return ERROR(srcSize_wrong); + ip += 4; + remainingSize -= 4; + } - return ip - ipstart; - } + return ip - ipstart; + } } /*! ZSTD_decompressFrame() : * @dctx must be properly initialized */ static size_t ZSTD_decompressFrame(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void** srcPtr, size_t *srcSizePtr) + void* dst, size_t dstCapacity, + const void** srcPtr, size_t *srcSizePtr) { - const BYTE* ip = (const BYTE*)(*srcPtr); - BYTE* const ostart = (BYTE* const)dst; - BYTE* const oend = ostart + dstCapacity; - BYTE* op = ostart; - size_t remainingSize = *srcSizePtr; + const BYTE* ip = (const BYTE*)(*srcPtr); + BYTE* const ostart = (BYTE* const)dst; + BYTE* const oend = ostart + dstCapacity; + BYTE* op = ostart; + size_t remainingSize = *srcSizePtr; - /* check */ - if (remainingSize < ZSTD_frameHeaderSize_min+ZSTD_blockHeaderSize) return ERROR(srcSize_wrong); + /* check */ + if (remainingSize < ZSTD_frameHeaderSize_min+ZSTD_blockHeaderSize) return ERROR(srcSize_wrong); - /* Frame Header */ - { size_t const frameHeaderSize = ZSTD_frameHeaderSize(ip, ZSTD_frameHeaderSize_prefix); - if (ZSTD_isError(frameHeaderSize)) return frameHeaderSize; - if (remainingSize < frameHeaderSize+ZSTD_blockHeaderSize) return ERROR(srcSize_wrong); - CHECK_F(ZSTD_decodeFrameHeader(dctx, ip, frameHeaderSize)); - ip += frameHeaderSize; remainingSize -= frameHeaderSize; - } + /* Frame Header */ + { size_t const frameHeaderSize = ZSTD_frameHeaderSize(ip, ZSTD_frameHeaderSize_prefix); + if (ZSTD_isError(frameHeaderSize)) return frameHeaderSize; + if (remainingSize < frameHeaderSize+ZSTD_blockHeaderSize) return ERROR(srcSize_wrong); + CHECK_F(ZSTD_decodeFrameHeader(dctx, ip, frameHeaderSize)); + ip += frameHeaderSize; remainingSize -= frameHeaderSize; + } - /* Loop on each block */ - while (1) { - size_t decodedSize; - blockProperties_t blockProperties; - size_t const cBlockSize = ZSTD_getcBlockSize(ip, remainingSize, &blockProperties); - if (ZSTD_isError(cBlockSize)) return cBlockSize; + /* Loop on each block */ + while (1) { + size_t decodedSize; + blockProperties_t blockProperties; + size_t const cBlockSize = ZSTD_getcBlockSize(ip, remainingSize, &blockProperties); + if (ZSTD_isError(cBlockSize)) return cBlockSize; - ip += ZSTD_blockHeaderSize; - remainingSize -= ZSTD_blockHeaderSize; - if (cBlockSize > remainingSize) return ERROR(srcSize_wrong); + ip += ZSTD_blockHeaderSize; + remainingSize -= ZSTD_blockHeaderSize; + if (cBlockSize > remainingSize) return ERROR(srcSize_wrong); - switch(blockProperties.blockType) - { - case bt_compressed: - decodedSize = ZSTD_decompressBlock_internal(dctx, op, oend-op, ip, cBlockSize); - break; - case bt_raw : - decodedSize = ZSTD_copyRawBlock(op, oend-op, ip, cBlockSize); - break; - case bt_rle : - decodedSize = ZSTD_generateNxBytes(op, oend-op, *ip, blockProperties.origSize); - break; - case bt_reserved : - default: - return ERROR(corruption_detected); - } + switch(blockProperties.blockType) + { + case bt_compressed: + decodedSize = ZSTD_decompressBlock_internal(dctx, op, oend-op, ip, cBlockSize); + break; + case bt_raw : + decodedSize = ZSTD_copyRawBlock(op, oend-op, ip, cBlockSize); + break; + case bt_rle : + decodedSize = ZSTD_generateNxBytes(op, oend-op, *ip, blockProperties.origSize); + break; + case bt_reserved : + default: + return ERROR(corruption_detected); + } - if (ZSTD_isError(decodedSize)) return decodedSize; - if (dctx->fParams.checksumFlag) XXH64_update(&dctx->xxhState, op, decodedSize); - op += decodedSize; - ip += cBlockSize; - remainingSize -= cBlockSize; - if (blockProperties.lastBlock) break; - } + if (ZSTD_isError(decodedSize)) return decodedSize; + if (dctx->fParams.checksumFlag) XXH64_update(&dctx->xxhState, op, decodedSize); + op += decodedSize; + ip += cBlockSize; + remainingSize -= cBlockSize; + if (blockProperties.lastBlock) break; + } - if (dctx->fParams.checksumFlag) { /* Frame content checksum verification */ - U32 const checkCalc = (U32)XXH64_digest(&dctx->xxhState); - U32 checkRead; - if (remainingSize<4) return ERROR(checksum_wrong); - checkRead = MEM_readLE32(ip); - if (checkRead != checkCalc) return ERROR(checksum_wrong); - ip += 4; - remainingSize -= 4; - } + if (dctx->fParams.checksumFlag) { /* Frame content checksum verification */ + U32 const checkCalc = (U32)XXH64_digest(&dctx->xxhState); + U32 checkRead; + if (remainingSize<4) return ERROR(checksum_wrong); + checkRead = MEM_readLE32(ip); + if (checkRead != checkCalc) return ERROR(checksum_wrong); + ip += 4; + remainingSize -= 4; + } - /* Allow caller to get size read */ - *srcPtr = ip; - *srcSizePtr = remainingSize; - return op-ostart; + /* Allow caller to get size read */ + *srcPtr = ip; + *srcSizePtr = remainingSize; + return op-ostart; } static const void* ZSTD_DDictDictContent(const ZSTD_DDict* ddict); static size_t ZSTD_DDictDictSize(const ZSTD_DDict* ddict); static size_t ZSTD_decompressMultiFrame(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void *dict, size_t dictSize, - const ZSTD_DDict* ddict) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void *dict, size_t dictSize, + const ZSTD_DDict* ddict) { - void* const dststart = dst; + void* const dststart = dst; - if (ddict) { - if (dict) { - /* programmer error, these two cases should be mutually exclusive */ - return ERROR(GENERIC); - } + if (ddict) { + if (dict) { + /* programmer error, these two cases should be mutually exclusive */ + return ERROR(GENERIC); + } - dict = ZSTD_DDictDictContent(ddict); - dictSize = ZSTD_DDictDictSize(ddict); - } + dict = ZSTD_DDictDictContent(ddict); + dictSize = ZSTD_DDictDictSize(ddict); + } - while (srcSize >= ZSTD_frameHeaderSize_prefix) { - U32 magicNumber; + while (srcSize >= ZSTD_frameHeaderSize_prefix) { + U32 magicNumber; #if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) - if (ZSTD_isLegacy(src, srcSize)) { - size_t decodedSize; - size_t const frameSize = ZSTD_findFrameCompressedSizeLegacy(src, srcSize); - if (ZSTD_isError(frameSize)) return frameSize; + if (ZSTD_isLegacy(src, srcSize)) { + size_t decodedSize; + size_t const frameSize = ZSTD_findFrameCompressedSizeLegacy(src, srcSize); + if (ZSTD_isError(frameSize)) return frameSize; - decodedSize = ZSTD_decompressLegacy(dst, dstCapacity, src, frameSize, dict, dictSize); + decodedSize = ZSTD_decompressLegacy(dst, dstCapacity, src, frameSize, dict, dictSize); - dst = (BYTE*)dst + decodedSize; - dstCapacity -= decodedSize; + dst = (BYTE*)dst + decodedSize; + dstCapacity -= decodedSize; - src = (const BYTE*)src + frameSize; - srcSize -= frameSize; + src = (const BYTE*)src + frameSize; + srcSize -= frameSize; - continue; - } + continue; + } #endif - magicNumber = MEM_readLE32(src); - if (magicNumber != ZSTD_MAGICNUMBER) { - if ((magicNumber & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { - size_t skippableSize; - if (srcSize < ZSTD_skippableHeaderSize) - return ERROR(srcSize_wrong); - skippableSize = MEM_readLE32((const BYTE *)src + 4) + - ZSTD_skippableHeaderSize; - if (srcSize < skippableSize) { - return ERROR(srcSize_wrong); - } + magicNumber = MEM_readLE32(src); + if (magicNumber != ZSTD_MAGICNUMBER) { + if ((magicNumber & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { + size_t skippableSize; + if (srcSize < ZSTD_skippableHeaderSize) + return ERROR(srcSize_wrong); + skippableSize = MEM_readLE32((const BYTE *)src + 4) + + ZSTD_skippableHeaderSize; + if (srcSize < skippableSize) { + return ERROR(srcSize_wrong); + } - src = (const BYTE *)src + skippableSize; - srcSize -= skippableSize; - continue; - } else { - return ERROR(prefix_unknown); - } - } + src = (const BYTE *)src + skippableSize; + srcSize -= skippableSize; + continue; + } else { + return ERROR(prefix_unknown); + } + } - if (ddict) { - /* we were called from ZSTD_decompress_usingDDict */ - ZSTD_refDDict(dctx, ddict); - } else { - /* this will initialize correctly with no dict if dict == NULL, so - * use this in all cases but ddict */ - CHECK_F(ZSTD_decompressBegin_usingDict(dctx, dict, dictSize)); - } - ZSTD_checkContinuity(dctx, dst); + if (ddict) { + /* we were called from ZSTD_decompress_usingDDict */ + ZSTD_refDDict(dctx, ddict); + } else { + /* this will initialize correctly with no dict if dict == NULL, so + * use this in all cases but ddict */ + CHECK_F(ZSTD_decompressBegin_usingDict(dctx, dict, dictSize)); + } + ZSTD_checkContinuity(dctx, dst); - { const size_t res = ZSTD_decompressFrame(dctx, dst, dstCapacity, - &src, &srcSize); - if (ZSTD_isError(res)) return res; - /* don't need to bounds check this, ZSTD_decompressFrame will have - * already */ - dst = (BYTE*)dst + res; - dstCapacity -= res; - } - } + { const size_t res = ZSTD_decompressFrame(dctx, dst, dstCapacity, + &src, &srcSize); + if (ZSTD_isError(res)) return res; + /* don't need to bounds check this, ZSTD_decompressFrame will have + * already */ + dst = (BYTE*)dst + res; + dstCapacity -= res; + } + } - if (srcSize) return ERROR(srcSize_wrong); /* input not entirely consumed */ + if (srcSize) return ERROR(srcSize_wrong); /* input not entirely consumed */ - return (BYTE*)dst - (BYTE*)dststart; + return (BYTE*)dst - (BYTE*)dststart; } size_t ZSTD_decompress_usingDict(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict, size_t dictSize) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const void* dict, size_t dictSize) { - return ZSTD_decompressMultiFrame(dctx, dst, dstCapacity, src, srcSize, dict, dictSize, NULL); + return ZSTD_decompressMultiFrame(dctx, dst, dstCapacity, src, srcSize, dict, dictSize, NULL); } size_t ZSTD_decompressDCtx(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - return ZSTD_decompress_usingDict(dctx, dst, dstCapacity, src, srcSize, NULL, 0); + return ZSTD_decompress_usingDict(dctx, dst, dstCapacity, src, srcSize, NULL, 0); } size_t ZSTD_decompress(void* dst, size_t dstCapacity, const void* src, size_t srcSize) { #if defined(ZSTD_HEAPMODE) && (ZSTD_HEAPMODE==1) - size_t regenSize; - ZSTD_DCtx* const dctx = ZSTD_createDCtx(); - if (dctx==NULL) return ERROR(memory_allocation); - regenSize = ZSTD_decompressDCtx(dctx, dst, dstCapacity, src, srcSize); - ZSTD_freeDCtx(dctx); - return regenSize; + size_t regenSize; + ZSTD_DCtx* const dctx = ZSTD_createDCtx(); + if (dctx==NULL) return ERROR(memory_allocation); + regenSize = ZSTD_decompressDCtx(dctx, dst, dstCapacity, src, srcSize); + ZSTD_freeDCtx(dctx); + return regenSize; #else /* stack mode */ - ZSTD_DCtx dctx; - return ZSTD_decompressDCtx(&dctx, dst, dstCapacity, src, srcSize); + ZSTD_DCtx dctx; + return ZSTD_decompressDCtx(&dctx, dst, dstCapacity, src, srcSize); #endif } @@ -1719,24 +1719,24 @@ size_t ZSTD_decompress(void* dst, size_t dstCapacity, const void* src, size_t sr size_t ZSTD_nextSrcSizeToDecompress(ZSTD_DCtx* dctx) { return dctx->expected; } ZSTD_nextInputType_e ZSTD_nextInputType(ZSTD_DCtx* dctx) { - switch(dctx->stage) - { - default: /* should not happen */ - case ZSTDds_getFrameHeaderSize: - case ZSTDds_decodeFrameHeader: - return ZSTDnit_frameHeader; - case ZSTDds_decodeBlockHeader: - return ZSTDnit_blockHeader; - case ZSTDds_decompressBlock: - return ZSTDnit_block; - case ZSTDds_decompressLastBlock: - return ZSTDnit_lastBlock; - case ZSTDds_checkChecksum: - return ZSTDnit_checksum; - case ZSTDds_decodeSkippableHeader: - case ZSTDds_skipFrame: - return ZSTDnit_skippableFrame; - } + switch(dctx->stage) + { + default: /* should not happen */ + case ZSTDds_getFrameHeaderSize: + case ZSTDds_decodeFrameHeader: + return ZSTDnit_frameHeader; + case ZSTDds_decodeBlockHeader: + return ZSTDnit_blockHeader; + case ZSTDds_decompressBlock: + return ZSTDnit_block; + case ZSTDds_decompressLastBlock: + return ZSTDnit_lastBlock; + case ZSTDds_checkChecksum: + return ZSTDnit_checksum; + case ZSTDds_decodeSkippableHeader: + case ZSTDds_skipFrame: + return ZSTDnit_skippableFrame; + } } int ZSTD_isSkipFrame(ZSTD_DCtx* dctx) { return dctx->stage == ZSTDds_skipFrame; } /* for zbuff */ @@ -1746,131 +1746,131 @@ int ZSTD_isSkipFrame(ZSTD_DCtx* dctx) { return dctx->stage == ZSTDds_skipFrame; * or an error code, which can be tested using ZSTD_isError() */ size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - /* Sanity check */ - if (srcSize != dctx->expected) return ERROR(srcSize_wrong); - if (dstCapacity) ZSTD_checkContinuity(dctx, dst); + /* Sanity check */ + if (srcSize != dctx->expected) return ERROR(srcSize_wrong); + if (dstCapacity) ZSTD_checkContinuity(dctx, dst); - switch (dctx->stage) - { - case ZSTDds_getFrameHeaderSize : - if (srcSize != ZSTD_frameHeaderSize_prefix) return ERROR(srcSize_wrong); /* impossible */ - if ((MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { /* skippable frame */ - memcpy(dctx->headerBuffer, src, ZSTD_frameHeaderSize_prefix); - dctx->expected = ZSTD_skippableHeaderSize - ZSTD_frameHeaderSize_prefix; /* magic number + skippable frame length */ - dctx->stage = ZSTDds_decodeSkippableHeader; - return 0; - } - dctx->headerSize = ZSTD_frameHeaderSize(src, ZSTD_frameHeaderSize_prefix); - if (ZSTD_isError(dctx->headerSize)) return dctx->headerSize; - memcpy(dctx->headerBuffer, src, ZSTD_frameHeaderSize_prefix); - if (dctx->headerSize > ZSTD_frameHeaderSize_prefix) { - dctx->expected = dctx->headerSize - ZSTD_frameHeaderSize_prefix; - dctx->stage = ZSTDds_decodeFrameHeader; - return 0; - } - dctx->expected = 0; /* not necessary to copy more */ + switch (dctx->stage) + { + case ZSTDds_getFrameHeaderSize : + if (srcSize != ZSTD_frameHeaderSize_prefix) return ERROR(srcSize_wrong); /* impossible */ + if ((MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { /* skippable frame */ + memcpy(dctx->headerBuffer, src, ZSTD_frameHeaderSize_prefix); + dctx->expected = ZSTD_skippableHeaderSize - ZSTD_frameHeaderSize_prefix; /* magic number + skippable frame length */ + dctx->stage = ZSTDds_decodeSkippableHeader; + return 0; + } + dctx->headerSize = ZSTD_frameHeaderSize(src, ZSTD_frameHeaderSize_prefix); + if (ZSTD_isError(dctx->headerSize)) return dctx->headerSize; + memcpy(dctx->headerBuffer, src, ZSTD_frameHeaderSize_prefix); + if (dctx->headerSize > ZSTD_frameHeaderSize_prefix) { + dctx->expected = dctx->headerSize - ZSTD_frameHeaderSize_prefix; + dctx->stage = ZSTDds_decodeFrameHeader; + return 0; + } + dctx->expected = 0; /* not necessary to copy more */ - case ZSTDds_decodeFrameHeader: - memcpy(dctx->headerBuffer + ZSTD_frameHeaderSize_prefix, src, dctx->expected); - CHECK_F(ZSTD_decodeFrameHeader(dctx, dctx->headerBuffer, dctx->headerSize)); - dctx->expected = ZSTD_blockHeaderSize; - dctx->stage = ZSTDds_decodeBlockHeader; - return 0; + case ZSTDds_decodeFrameHeader: + memcpy(dctx->headerBuffer + ZSTD_frameHeaderSize_prefix, src, dctx->expected); + CHECK_F(ZSTD_decodeFrameHeader(dctx, dctx->headerBuffer, dctx->headerSize)); + dctx->expected = ZSTD_blockHeaderSize; + dctx->stage = ZSTDds_decodeBlockHeader; + return 0; - case ZSTDds_decodeBlockHeader: - { blockProperties_t bp; - size_t const cBlockSize = ZSTD_getcBlockSize(src, ZSTD_blockHeaderSize, &bp); - if (ZSTD_isError(cBlockSize)) return cBlockSize; - dctx->expected = cBlockSize; - dctx->bType = bp.blockType; - dctx->rleSize = bp.origSize; - if (cBlockSize) { - dctx->stage = bp.lastBlock ? ZSTDds_decompressLastBlock : ZSTDds_decompressBlock; - return 0; - } - /* empty block */ - if (bp.lastBlock) { - if (dctx->fParams.checksumFlag) { - dctx->expected = 4; - dctx->stage = ZSTDds_checkChecksum; - } else { - dctx->expected = 0; /* end of frame */ - dctx->stage = ZSTDds_getFrameHeaderSize; - } - } else { - dctx->expected = 3; /* go directly to next header */ - dctx->stage = ZSTDds_decodeBlockHeader; - } - return 0; - } - case ZSTDds_decompressLastBlock: - case ZSTDds_decompressBlock: - { size_t rSize; - switch(dctx->bType) - { - case bt_compressed: - rSize = ZSTD_decompressBlock_internal(dctx, dst, dstCapacity, src, srcSize); - break; - case bt_raw : - rSize = ZSTD_copyRawBlock(dst, dstCapacity, src, srcSize); - break; - case bt_rle : - rSize = ZSTD_setRleBlock(dst, dstCapacity, src, srcSize, dctx->rleSize); - break; - case bt_reserved : /* should never happen */ - default: - return ERROR(corruption_detected); - } - if (ZSTD_isError(rSize)) return rSize; - if (dctx->fParams.checksumFlag) XXH64_update(&dctx->xxhState, dst, rSize); + case ZSTDds_decodeBlockHeader: + { blockProperties_t bp; + size_t const cBlockSize = ZSTD_getcBlockSize(src, ZSTD_blockHeaderSize, &bp); + if (ZSTD_isError(cBlockSize)) return cBlockSize; + dctx->expected = cBlockSize; + dctx->bType = bp.blockType; + dctx->rleSize = bp.origSize; + if (cBlockSize) { + dctx->stage = bp.lastBlock ? ZSTDds_decompressLastBlock : ZSTDds_decompressBlock; + return 0; + } + /* empty block */ + if (bp.lastBlock) { + if (dctx->fParams.checksumFlag) { + dctx->expected = 4; + dctx->stage = ZSTDds_checkChecksum; + } else { + dctx->expected = 0; /* end of frame */ + dctx->stage = ZSTDds_getFrameHeaderSize; + } + } else { + dctx->expected = 3; /* go directly to next header */ + dctx->stage = ZSTDds_decodeBlockHeader; + } + return 0; + } + case ZSTDds_decompressLastBlock: + case ZSTDds_decompressBlock: + { size_t rSize; + switch(dctx->bType) + { + case bt_compressed: + rSize = ZSTD_decompressBlock_internal(dctx, dst, dstCapacity, src, srcSize); + break; + case bt_raw : + rSize = ZSTD_copyRawBlock(dst, dstCapacity, src, srcSize); + break; + case bt_rle : + rSize = ZSTD_setRleBlock(dst, dstCapacity, src, srcSize, dctx->rleSize); + break; + case bt_reserved : /* should never happen */ + default: + return ERROR(corruption_detected); + } + if (ZSTD_isError(rSize)) return rSize; + if (dctx->fParams.checksumFlag) XXH64_update(&dctx->xxhState, dst, rSize); - if (dctx->stage == ZSTDds_decompressLastBlock) { /* end of frame */ - if (dctx->fParams.checksumFlag) { /* another round for frame checksum */ - dctx->expected = 4; - dctx->stage = ZSTDds_checkChecksum; - } else { - dctx->expected = 0; /* ends here */ - dctx->stage = ZSTDds_getFrameHeaderSize; - } - } else { - dctx->stage = ZSTDds_decodeBlockHeader; - dctx->expected = ZSTD_blockHeaderSize; - dctx->previousDstEnd = (char*)dst + rSize; - } - return rSize; - } - case ZSTDds_checkChecksum: - { U32 const h32 = (U32)XXH64_digest(&dctx->xxhState); - U32 const check32 = MEM_readLE32(src); /* srcSize == 4, guaranteed by dctx->expected */ - if (check32 != h32) return ERROR(checksum_wrong); - dctx->expected = 0; - dctx->stage = ZSTDds_getFrameHeaderSize; - return 0; - } - case ZSTDds_decodeSkippableHeader: - { memcpy(dctx->headerBuffer + ZSTD_frameHeaderSize_prefix, src, dctx->expected); - dctx->expected = MEM_readLE32(dctx->headerBuffer + 4); - dctx->stage = ZSTDds_skipFrame; - return 0; - } - case ZSTDds_skipFrame: - { dctx->expected = 0; - dctx->stage = ZSTDds_getFrameHeaderSize; - return 0; - } - default: - return ERROR(GENERIC); /* impossible */ - } + if (dctx->stage == ZSTDds_decompressLastBlock) { /* end of frame */ + if (dctx->fParams.checksumFlag) { /* another round for frame checksum */ + dctx->expected = 4; + dctx->stage = ZSTDds_checkChecksum; + } else { + dctx->expected = 0; /* ends here */ + dctx->stage = ZSTDds_getFrameHeaderSize; + } + } else { + dctx->stage = ZSTDds_decodeBlockHeader; + dctx->expected = ZSTD_blockHeaderSize; + dctx->previousDstEnd = (char*)dst + rSize; + } + return rSize; + } + case ZSTDds_checkChecksum: + { U32 const h32 = (U32)XXH64_digest(&dctx->xxhState); + U32 const check32 = MEM_readLE32(src); /* srcSize == 4, guaranteed by dctx->expected */ + if (check32 != h32) return ERROR(checksum_wrong); + dctx->expected = 0; + dctx->stage = ZSTDds_getFrameHeaderSize; + return 0; + } + case ZSTDds_decodeSkippableHeader: + { memcpy(dctx->headerBuffer + ZSTD_frameHeaderSize_prefix, src, dctx->expected); + dctx->expected = MEM_readLE32(dctx->headerBuffer + 4); + dctx->stage = ZSTDds_skipFrame; + return 0; + } + case ZSTDds_skipFrame: + { dctx->expected = 0; + dctx->stage = ZSTDds_getFrameHeaderSize; + return 0; + } + default: + return ERROR(GENERIC); /* impossible */ + } } static size_t ZSTD_refDictContent(ZSTD_DCtx* dctx, const void* dict, size_t dictSize) { - dctx->dictEnd = dctx->previousDstEnd; - dctx->vBase = (const char*)dict - ((const char*)(dctx->previousDstEnd) - (const char*)(dctx->base)); - dctx->base = dict; - dctx->previousDstEnd = (const char*)dict + dictSize; - return 0; + dctx->dictEnd = dctx->previousDstEnd; + dctx->vBase = (const char*)dict - ((const char*)(dctx->previousDstEnd) - (const char*)(dctx->base)); + dctx->base = dict; + dctx->previousDstEnd = (const char*)dict + dictSize; + return 0; } /* ZSTD_loadEntropy() : @@ -1878,181 +1878,181 @@ static size_t ZSTD_refDictContent(ZSTD_DCtx* dctx, const void* dict, size_t dict * @return : size of entropy tables read */ static size_t ZSTD_loadEntropy(ZSTD_entropyTables_t* entropy, const void* const dict, size_t const dictSize) { - const BYTE* dictPtr = (const BYTE*)dict; - const BYTE* const dictEnd = dictPtr + dictSize; + const BYTE* dictPtr = (const BYTE*)dict; + const BYTE* const dictEnd = dictPtr + dictSize; - if (dictSize <= 8) return ERROR(dictionary_corrupted); - dictPtr += 8; /* skip header = magic + dictID */ + if (dictSize <= 8) return ERROR(dictionary_corrupted); + dictPtr += 8; /* skip header = magic + dictID */ - { size_t const hSize = HUF_readDTableX4(entropy->hufTable, dictPtr, dictEnd-dictPtr); - if (HUF_isError(hSize)) return ERROR(dictionary_corrupted); - dictPtr += hSize; - } + { size_t const hSize = HUF_readDTableX4(entropy->hufTable, dictPtr, dictEnd-dictPtr); + if (HUF_isError(hSize)) return ERROR(dictionary_corrupted); + dictPtr += hSize; + } - { short offcodeNCount[MaxOff+1]; - U32 offcodeMaxValue = MaxOff, offcodeLog; - size_t const offcodeHeaderSize = FSE_readNCount(offcodeNCount, &offcodeMaxValue, &offcodeLog, dictPtr, dictEnd-dictPtr); - if (FSE_isError(offcodeHeaderSize)) return ERROR(dictionary_corrupted); - if (offcodeLog > OffFSELog) return ERROR(dictionary_corrupted); - CHECK_E(FSE_buildDTable(entropy->OFTable, offcodeNCount, offcodeMaxValue, offcodeLog), dictionary_corrupted); - dictPtr += offcodeHeaderSize; - } + { short offcodeNCount[MaxOff+1]; + U32 offcodeMaxValue = MaxOff, offcodeLog; + size_t const offcodeHeaderSize = FSE_readNCount(offcodeNCount, &offcodeMaxValue, &offcodeLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(offcodeHeaderSize)) return ERROR(dictionary_corrupted); + if (offcodeLog > OffFSELog) return ERROR(dictionary_corrupted); + CHECK_E(FSE_buildDTable(entropy->OFTable, offcodeNCount, offcodeMaxValue, offcodeLog), dictionary_corrupted); + dictPtr += offcodeHeaderSize; + } - { short matchlengthNCount[MaxML+1]; - unsigned matchlengthMaxValue = MaxML, matchlengthLog; - size_t const matchlengthHeaderSize = FSE_readNCount(matchlengthNCount, &matchlengthMaxValue, &matchlengthLog, dictPtr, dictEnd-dictPtr); - if (FSE_isError(matchlengthHeaderSize)) return ERROR(dictionary_corrupted); - if (matchlengthLog > MLFSELog) return ERROR(dictionary_corrupted); - CHECK_E(FSE_buildDTable(entropy->MLTable, matchlengthNCount, matchlengthMaxValue, matchlengthLog), dictionary_corrupted); - dictPtr += matchlengthHeaderSize; - } + { short matchlengthNCount[MaxML+1]; + unsigned matchlengthMaxValue = MaxML, matchlengthLog; + size_t const matchlengthHeaderSize = FSE_readNCount(matchlengthNCount, &matchlengthMaxValue, &matchlengthLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(matchlengthHeaderSize)) return ERROR(dictionary_corrupted); + if (matchlengthLog > MLFSELog) return ERROR(dictionary_corrupted); + CHECK_E(FSE_buildDTable(entropy->MLTable, matchlengthNCount, matchlengthMaxValue, matchlengthLog), dictionary_corrupted); + dictPtr += matchlengthHeaderSize; + } - { short litlengthNCount[MaxLL+1]; - unsigned litlengthMaxValue = MaxLL, litlengthLog; - size_t const litlengthHeaderSize = FSE_readNCount(litlengthNCount, &litlengthMaxValue, &litlengthLog, dictPtr, dictEnd-dictPtr); - if (FSE_isError(litlengthHeaderSize)) return ERROR(dictionary_corrupted); - if (litlengthLog > LLFSELog) return ERROR(dictionary_corrupted); - CHECK_E(FSE_buildDTable(entropy->LLTable, litlengthNCount, litlengthMaxValue, litlengthLog), dictionary_corrupted); - dictPtr += litlengthHeaderSize; - } + { short litlengthNCount[MaxLL+1]; + unsigned litlengthMaxValue = MaxLL, litlengthLog; + size_t const litlengthHeaderSize = FSE_readNCount(litlengthNCount, &litlengthMaxValue, &litlengthLog, dictPtr, dictEnd-dictPtr); + if (FSE_isError(litlengthHeaderSize)) return ERROR(dictionary_corrupted); + if (litlengthLog > LLFSELog) return ERROR(dictionary_corrupted); + CHECK_E(FSE_buildDTable(entropy->LLTable, litlengthNCount, litlengthMaxValue, litlengthLog), dictionary_corrupted); + dictPtr += litlengthHeaderSize; + } - if (dictPtr+12 > dictEnd) return ERROR(dictionary_corrupted); - { int i; - size_t const dictContentSize = (size_t)(dictEnd - (dictPtr+12)); - for (i=0; i<3; i++) { - U32 const rep = MEM_readLE32(dictPtr); dictPtr += 4; - if (rep==0 || rep >= dictContentSize) return ERROR(dictionary_corrupted); - entropy->rep[i] = rep; - } } + if (dictPtr+12 > dictEnd) return ERROR(dictionary_corrupted); + { int i; + size_t const dictContentSize = (size_t)(dictEnd - (dictPtr+12)); + for (i=0; i<3; i++) { + U32 const rep = MEM_readLE32(dictPtr); dictPtr += 4; + if (rep==0 || rep >= dictContentSize) return ERROR(dictionary_corrupted); + entropy->rep[i] = rep; + } } - return dictPtr - (const BYTE*)dict; + return dictPtr - (const BYTE*)dict; } static size_t ZSTD_decompress_insertDictionary(ZSTD_DCtx* dctx, const void* dict, size_t dictSize) { - if (dictSize < 8) return ZSTD_refDictContent(dctx, dict, dictSize); - { U32 const magic = MEM_readLE32(dict); - if (magic != ZSTD_DICT_MAGIC) { - return ZSTD_refDictContent(dctx, dict, dictSize); /* pure content mode */ - } } - dctx->dictID = MEM_readLE32((const char*)dict + 4); + if (dictSize < 8) return ZSTD_refDictContent(dctx, dict, dictSize); + { U32 const magic = MEM_readLE32(dict); + if (magic != ZSTD_DICT_MAGIC) { + return ZSTD_refDictContent(dctx, dict, dictSize); /* pure content mode */ + } } + dctx->dictID = MEM_readLE32((const char*)dict + 4); - /* load entropy tables */ - { size_t const eSize = ZSTD_loadEntropy(&dctx->entropy, dict, dictSize); - if (ZSTD_isError(eSize)) return ERROR(dictionary_corrupted); - dict = (const char*)dict + eSize; - dictSize -= eSize; - } - dctx->litEntropy = dctx->fseEntropy = 1; + /* load entropy tables */ + { size_t const eSize = ZSTD_loadEntropy(&dctx->entropy, dict, dictSize); + if (ZSTD_isError(eSize)) return ERROR(dictionary_corrupted); + dict = (const char*)dict + eSize; + dictSize -= eSize; + } + dctx->litEntropy = dctx->fseEntropy = 1; - /* reference dictionary content */ - return ZSTD_refDictContent(dctx, dict, dictSize); + /* reference dictionary content */ + return ZSTD_refDictContent(dctx, dict, dictSize); } size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx* dctx, const void* dict, size_t dictSize) { - CHECK_F(ZSTD_decompressBegin(dctx)); - if (dict && dictSize) CHECK_E(ZSTD_decompress_insertDictionary(dctx, dict, dictSize), dictionary_corrupted); - return 0; + CHECK_F(ZSTD_decompressBegin(dctx)); + if (dict && dictSize) CHECK_E(ZSTD_decompress_insertDictionary(dctx, dict, dictSize), dictionary_corrupted); + return 0; } /* ====== ZSTD_DDict ====== */ struct ZSTD_DDict_s { - void* dictBuffer; - const void* dictContent; - size_t dictSize; - ZSTD_entropyTables_t entropy; - U32 dictID; - U32 entropyPresent; - ZSTD_customMem cMem; + void* dictBuffer; + const void* dictContent; + size_t dictSize; + ZSTD_entropyTables_t entropy; + U32 dictID; + U32 entropyPresent; + ZSTD_customMem cMem; }; /* typedef'd to ZSTD_DDict within "zstd.h" */ static const void* ZSTD_DDictDictContent(const ZSTD_DDict* ddict) { - return ddict->dictContent; + return ddict->dictContent; } static size_t ZSTD_DDictDictSize(const ZSTD_DDict* ddict) { - return ddict->dictSize; + return ddict->dictSize; } static void ZSTD_refDDict(ZSTD_DCtx* dstDCtx, const ZSTD_DDict* ddict) { - ZSTD_decompressBegin(dstDCtx); /* init */ - if (ddict) { /* support refDDict on NULL */ - dstDCtx->dictID = ddict->dictID; - dstDCtx->base = ddict->dictContent; - dstDCtx->vBase = ddict->dictContent; - dstDCtx->dictEnd = (const BYTE*)ddict->dictContent + ddict->dictSize; - dstDCtx->previousDstEnd = dstDCtx->dictEnd; - if (ddict->entropyPresent) { - dstDCtx->litEntropy = 1; - dstDCtx->fseEntropy = 1; - dstDCtx->LLTptr = ddict->entropy.LLTable; - dstDCtx->MLTptr = ddict->entropy.MLTable; - dstDCtx->OFTptr = ddict->entropy.OFTable; - dstDCtx->HUFptr = ddict->entropy.hufTable; - dstDCtx->entropy.rep[0] = ddict->entropy.rep[0]; - dstDCtx->entropy.rep[1] = ddict->entropy.rep[1]; - dstDCtx->entropy.rep[2] = ddict->entropy.rep[2]; - } else { - dstDCtx->litEntropy = 0; - dstDCtx->fseEntropy = 0; - } - } + ZSTD_decompressBegin(dstDCtx); /* init */ + if (ddict) { /* support refDDict on NULL */ + dstDCtx->dictID = ddict->dictID; + dstDCtx->base = ddict->dictContent; + dstDCtx->vBase = ddict->dictContent; + dstDCtx->dictEnd = (const BYTE*)ddict->dictContent + ddict->dictSize; + dstDCtx->previousDstEnd = dstDCtx->dictEnd; + if (ddict->entropyPresent) { + dstDCtx->litEntropy = 1; + dstDCtx->fseEntropy = 1; + dstDCtx->LLTptr = ddict->entropy.LLTable; + dstDCtx->MLTptr = ddict->entropy.MLTable; + dstDCtx->OFTptr = ddict->entropy.OFTable; + dstDCtx->HUFptr = ddict->entropy.hufTable; + dstDCtx->entropy.rep[0] = ddict->entropy.rep[0]; + dstDCtx->entropy.rep[1] = ddict->entropy.rep[1]; + dstDCtx->entropy.rep[2] = ddict->entropy.rep[2]; + } else { + dstDCtx->litEntropy = 0; + dstDCtx->fseEntropy = 0; + } + } } static size_t ZSTD_loadEntropy_inDDict(ZSTD_DDict* ddict) { - ddict->dictID = 0; - ddict->entropyPresent = 0; - if (ddict->dictSize < 8) return 0; - { U32 const magic = MEM_readLE32(ddict->dictContent); - if (magic != ZSTD_DICT_MAGIC) return 0; /* pure content mode */ - } - ddict->dictID = MEM_readLE32((const char*)ddict->dictContent + 4); + ddict->dictID = 0; + ddict->entropyPresent = 0; + if (ddict->dictSize < 8) return 0; + { U32 const magic = MEM_readLE32(ddict->dictContent); + if (magic != ZSTD_DICT_MAGIC) return 0; /* pure content mode */ + } + ddict->dictID = MEM_readLE32((const char*)ddict->dictContent + 4); - /* load entropy tables */ - CHECK_E( ZSTD_loadEntropy(&ddict->entropy, ddict->dictContent, ddict->dictSize), dictionary_corrupted ); - ddict->entropyPresent = 1; - return 0; + /* load entropy tables */ + CHECK_E( ZSTD_loadEntropy(&ddict->entropy, ddict->dictContent, ddict->dictSize), dictionary_corrupted ); + ddict->entropyPresent = 1; + return 0; } ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize, unsigned byReference, ZSTD_customMem customMem) { - if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; - if (!customMem.customAlloc || !customMem.customFree) return NULL; + if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; + if (!customMem.customAlloc || !customMem.customFree) return NULL; - { ZSTD_DDict* const ddict = (ZSTD_DDict*) ZSTD_malloc(sizeof(ZSTD_DDict), customMem); - if (!ddict) return NULL; - ddict->cMem = customMem; + { ZSTD_DDict* const ddict = (ZSTD_DDict*) ZSTD_malloc(sizeof(ZSTD_DDict), customMem); + if (!ddict) return NULL; + ddict->cMem = customMem; - if ((byReference) || (!dict) || (!dictSize)) { - ddict->dictBuffer = NULL; - ddict->dictContent = dict; - } else { - void* const internalBuffer = ZSTD_malloc(dictSize, customMem); - if (!internalBuffer) { ZSTD_freeDDict(ddict); return NULL; } - memcpy(internalBuffer, dict, dictSize); - ddict->dictBuffer = internalBuffer; - ddict->dictContent = internalBuffer; - } - ddict->dictSize = dictSize; - ddict->entropy.hufTable[0] = (HUF_DTable)((HufLog)*0x1000001); /* cover both little and big endian */ - /* parse dictionary content */ - { size_t const errorCode = ZSTD_loadEntropy_inDDict(ddict); - if (ZSTD_isError(errorCode)) { - ZSTD_freeDDict(ddict); - return NULL; - } } + if ((byReference) || (!dict) || (!dictSize)) { + ddict->dictBuffer = NULL; + ddict->dictContent = dict; + } else { + void* const internalBuffer = ZSTD_malloc(dictSize, customMem); + if (!internalBuffer) { ZSTD_freeDDict(ddict); return NULL; } + memcpy(internalBuffer, dict, dictSize); + ddict->dictBuffer = internalBuffer; + ddict->dictContent = internalBuffer; + } + ddict->dictSize = dictSize; + ddict->entropy.hufTable[0] = (HUF_DTable)((HufLog)*0x1000001); /* cover both little and big endian */ + /* parse dictionary content */ + { size_t const errorCode = ZSTD_loadEntropy_inDDict(ddict); + if (ZSTD_isError(errorCode)) { + ZSTD_freeDDict(ddict); + return NULL; + } } - return ddict; - } + return ddict; + } } /*! ZSTD_createDDict() : @@ -2061,8 +2061,8 @@ ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize, unsigne * Consequently, `dict` can be released after `ZSTD_DDict` creation */ ZSTD_DDict* ZSTD_createDDict(const void* dict, size_t dictSize) { - ZSTD_customMem const allocator = { NULL, NULL, NULL }; - return ZSTD_createDDict_advanced(dict, dictSize, 0, allocator); + ZSTD_customMem const allocator = { NULL, NULL, NULL }; + return ZSTD_createDDict_advanced(dict, dictSize, 0, allocator); } @@ -2072,25 +2072,25 @@ ZSTD_DDict* ZSTD_createDDict(const void* dict, size_t dictSize) * Warning : dictBuffer must outlive DDict (DDict must be freed before dictBuffer) */ ZSTD_DDict* ZSTD_createDDict_byReference(const void* dictBuffer, size_t dictSize) { - ZSTD_customMem const allocator = { NULL, NULL, NULL }; - return ZSTD_createDDict_advanced(dictBuffer, dictSize, 1, allocator); + ZSTD_customMem const allocator = { NULL, NULL, NULL }; + return ZSTD_createDDict_advanced(dictBuffer, dictSize, 1, allocator); } size_t ZSTD_freeDDict(ZSTD_DDict* ddict) { - if (ddict==NULL) return 0; /* support free on NULL */ - { ZSTD_customMem const cMem = ddict->cMem; - ZSTD_free(ddict->dictBuffer, cMem); - ZSTD_free(ddict, cMem); - return 0; - } + if (ddict==NULL) return 0; /* support free on NULL */ + { ZSTD_customMem const cMem = ddict->cMem; + ZSTD_free(ddict->dictBuffer, cMem); + ZSTD_free(ddict, cMem); + return 0; + } } size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict) { - if (ddict==NULL) return 0; /* support sizeof on NULL */ - return sizeof(*ddict) + (ddict->dictBuffer ? ddict->dictSize : 0) ; + if (ddict==NULL) return 0; /* support sizeof on NULL */ + return sizeof(*ddict) + (ddict->dictBuffer ? ddict->dictSize : 0) ; } /*! ZSTD_getDictID_fromDict() : @@ -2099,9 +2099,9 @@ size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict) * It can still be loaded, but as a content-only dictionary. */ unsigned ZSTD_getDictID_fromDict(const void* dict, size_t dictSize) { - if (dictSize < 8) return 0; - if (MEM_readLE32(dict) != ZSTD_DICT_MAGIC) return 0; - return MEM_readLE32((const char*)dict + 4); + if (dictSize < 8) return 0; + if (MEM_readLE32(dict) != ZSTD_DICT_MAGIC) return 0; + return MEM_readLE32((const char*)dict + 4); } /*! ZSTD_getDictID_fromDDict() : @@ -2110,8 +2110,8 @@ unsigned ZSTD_getDictID_fromDict(const void* dict, size_t dictSize) * Non-conformant dictionaries can still be loaded, but as content-only dictionaries. */ unsigned ZSTD_getDictID_fromDDict(const ZSTD_DDict* ddict) { - if (ddict==NULL) return 0; - return ZSTD_getDictID_fromDict(ddict->dictContent, ddict->dictSize); + if (ddict==NULL) return 0; + return ZSTD_getDictID_fromDict(ddict->dictContent, ddict->dictSize); } /*! ZSTD_getDictID_fromFrame() : @@ -2126,10 +2126,10 @@ unsigned ZSTD_getDictID_fromDDict(const ZSTD_DDict* ddict) * When identifying the exact failure cause, it's possible to used ZSTD_getFrameParams(), which will provide a more precise error code. */ unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize) { - ZSTD_frameParams zfp = { 0 , 0 , 0 , 0 }; - size_t const hError = ZSTD_getFrameParams(&zfp, src, srcSize); - if (ZSTD_isError(hError)) return 0; - return zfp.dictID; + ZSTD_frameParams zfp = { 0 , 0 , 0 , 0 }; + size_t const hError = ZSTD_getFrameParams(&zfp, src, srcSize); + if (ZSTD_isError(hError)) return 0; + return zfp.dictID; } @@ -2137,14 +2137,14 @@ unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize) * Decompression using a pre-digested Dictionary * Use dictionary without significant overhead. */ size_t ZSTD_decompress_usingDDict(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const ZSTD_DDict* ddict) + void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + const ZSTD_DDict* ddict) { - /* pass content and size in case legacy frames are encountered */ - return ZSTD_decompressMultiFrame(dctx, dst, dstCapacity, src, srcSize, - NULL, 0, - ddict); + /* pass content and size in case legacy frames are encountered */ + return ZSTD_decompressMultiFrame(dctx, dst, dstCapacity, src, srcSize, + NULL, 0, + ddict); } @@ -2153,76 +2153,76 @@ size_t ZSTD_decompress_usingDDict(ZSTD_DCtx* dctx, *====================================*/ typedef enum { zdss_init, zdss_loadHeader, - zdss_read, zdss_load, zdss_flush } ZSTD_dStreamStage; + zdss_read, zdss_load, zdss_flush } ZSTD_dStreamStage; /* *** Resource management *** */ struct ZSTD_DStream_s { - ZSTD_DCtx* dctx; - ZSTD_DDict* ddictLocal; - const ZSTD_DDict* ddict; - ZSTD_frameParams fParams; - ZSTD_dStreamStage stage; - char* inBuff; - size_t inBuffSize; - size_t inPos; - size_t maxWindowSize; - char* outBuff; - size_t outBuffSize; - size_t outStart; - size_t outEnd; - size_t blockSize; - BYTE headerBuffer[ZSTD_FRAMEHEADERSIZE_MAX]; /* tmp buffer to store frame header */ - size_t lhSize; - ZSTD_customMem customMem; - void* legacyContext; - U32 previousLegacyVersion; - U32 legacyVersion; - U32 hostageByte; + ZSTD_DCtx* dctx; + ZSTD_DDict* ddictLocal; + const ZSTD_DDict* ddict; + ZSTD_frameParams fParams; + ZSTD_dStreamStage stage; + char* inBuff; + size_t inBuffSize; + size_t inPos; + size_t maxWindowSize; + char* outBuff; + size_t outBuffSize; + size_t outStart; + size_t outEnd; + size_t blockSize; + BYTE headerBuffer[ZSTD_FRAMEHEADERSIZE_MAX]; /* tmp buffer to store frame header */ + size_t lhSize; + ZSTD_customMem customMem; + void* legacyContext; + U32 previousLegacyVersion; + U32 legacyVersion; + U32 hostageByte; }; /* typedef'd to ZSTD_DStream within "zstd.h" */ ZSTD_DStream* ZSTD_createDStream(void) { - return ZSTD_createDStream_advanced(defaultCustomMem); + return ZSTD_createDStream_advanced(defaultCustomMem); } ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem) { - ZSTD_DStream* zds; + ZSTD_DStream* zds; - if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; - if (!customMem.customAlloc || !customMem.customFree) return NULL; + if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; + if (!customMem.customAlloc || !customMem.customFree) return NULL; - zds = (ZSTD_DStream*) ZSTD_malloc(sizeof(ZSTD_DStream), customMem); - if (zds==NULL) return NULL; - memset(zds, 0, sizeof(ZSTD_DStream)); - memcpy(&zds->customMem, &customMem, sizeof(ZSTD_customMem)); - zds->dctx = ZSTD_createDCtx_advanced(customMem); - if (zds->dctx == NULL) { ZSTD_freeDStream(zds); return NULL; } - zds->stage = zdss_init; - zds->maxWindowSize = ZSTD_MAXWINDOWSIZE_DEFAULT; - return zds; + zds = (ZSTD_DStream*) ZSTD_malloc(sizeof(ZSTD_DStream), customMem); + if (zds==NULL) return NULL; + memset(zds, 0, sizeof(ZSTD_DStream)); + memcpy(&zds->customMem, &customMem, sizeof(ZSTD_customMem)); + zds->dctx = ZSTD_createDCtx_advanced(customMem); + if (zds->dctx == NULL) { ZSTD_freeDStream(zds); return NULL; } + zds->stage = zdss_init; + zds->maxWindowSize = ZSTD_MAXWINDOWSIZE_DEFAULT; + return zds; } size_t ZSTD_freeDStream(ZSTD_DStream* zds) { - if (zds==NULL) return 0; /* support free on null */ - { ZSTD_customMem const cMem = zds->customMem; - ZSTD_freeDCtx(zds->dctx); - zds->dctx = NULL; - ZSTD_freeDDict(zds->ddictLocal); - zds->ddictLocal = NULL; - ZSTD_free(zds->inBuff, cMem); - zds->inBuff = NULL; - ZSTD_free(zds->outBuff, cMem); - zds->outBuff = NULL; + if (zds==NULL) return 0; /* support free on null */ + { ZSTD_customMem const cMem = zds->customMem; + ZSTD_freeDCtx(zds->dctx); + zds->dctx = NULL; + ZSTD_freeDDict(zds->ddictLocal); + zds->ddictLocal = NULL; + ZSTD_free(zds->inBuff, cMem); + zds->inBuff = NULL; + ZSTD_free(zds->outBuff, cMem); + zds->outBuff = NULL; #if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) - if (zds->legacyContext) - ZSTD_freeLegacyStreamContext(zds->legacyContext, zds->previousLegacyVersion); + if (zds->legacyContext) + ZSTD_freeLegacyStreamContext(zds->legacyContext, zds->previousLegacyVersion); #endif - ZSTD_free(zds, cMem); - return 0; - } + ZSTD_free(zds, cMem); + return 0; + } } @@ -2233,56 +2233,56 @@ size_t ZSTD_DStreamOutSize(void) { return ZSTD_BLOCKSIZE_ABSOLUTEMAX; } size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize) { - zds->stage = zdss_loadHeader; - zds->lhSize = zds->inPos = zds->outStart = zds->outEnd = 0; - ZSTD_freeDDict(zds->ddictLocal); - if (dict && dictSize >= 8) { - zds->ddictLocal = ZSTD_createDDict(dict, dictSize); - if (zds->ddictLocal == NULL) return ERROR(memory_allocation); - } else zds->ddictLocal = NULL; - zds->ddict = zds->ddictLocal; - zds->legacyVersion = 0; - zds->hostageByte = 0; - return ZSTD_frameHeaderSize_prefix; + zds->stage = zdss_loadHeader; + zds->lhSize = zds->inPos = zds->outStart = zds->outEnd = 0; + ZSTD_freeDDict(zds->ddictLocal); + if (dict && dictSize >= 8) { + zds->ddictLocal = ZSTD_createDDict(dict, dictSize); + if (zds->ddictLocal == NULL) return ERROR(memory_allocation); + } else zds->ddictLocal = NULL; + zds->ddict = zds->ddictLocal; + zds->legacyVersion = 0; + zds->hostageByte = 0; + return ZSTD_frameHeaderSize_prefix; } size_t ZSTD_initDStream(ZSTD_DStream* zds) { - return ZSTD_initDStream_usingDict(zds, NULL, 0); + return ZSTD_initDStream_usingDict(zds, NULL, 0); } size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, const ZSTD_DDict* ddict) /**< note : ddict will just be referenced, and must outlive decompression session */ { - size_t const initResult = ZSTD_initDStream(zds); - zds->ddict = ddict; - return initResult; + size_t const initResult = ZSTD_initDStream(zds); + zds->ddict = ddict; + return initResult; } size_t ZSTD_resetDStream(ZSTD_DStream* zds) { - zds->stage = zdss_loadHeader; - zds->lhSize = zds->inPos = zds->outStart = zds->outEnd = 0; - zds->legacyVersion = 0; - zds->hostageByte = 0; - return ZSTD_frameHeaderSize_prefix; + zds->stage = zdss_loadHeader; + zds->lhSize = zds->inPos = zds->outStart = zds->outEnd = 0; + zds->legacyVersion = 0; + zds->hostageByte = 0; + return ZSTD_frameHeaderSize_prefix; } size_t ZSTD_setDStreamParameter(ZSTD_DStream* zds, - ZSTD_DStreamParameter_e paramType, unsigned paramValue) + ZSTD_DStreamParameter_e paramType, unsigned paramValue) { - switch(paramType) - { - default : return ERROR(parameter_unknown); - case DStream_p_maxWindowSize : zds->maxWindowSize = paramValue ? paramValue : (U32)(-1); break; - } - return 0; + switch(paramType) + { + default : return ERROR(parameter_unknown); + case DStream_p_maxWindowSize : zds->maxWindowSize = paramValue ? paramValue : (U32)(-1); break; + } + return 0; } size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds) { - if (zds==NULL) return 0; /* support sizeof on NULL */ - return sizeof(*zds) + ZSTD_sizeof_DCtx(zds->dctx) + ZSTD_sizeof_DDict(zds->ddictLocal) + zds->inBuffSize + zds->outBuffSize; + if (zds==NULL) return 0; /* support sizeof on NULL */ + return sizeof(*zds) + ZSTD_sizeof_DCtx(zds->dctx) + ZSTD_sizeof_DDict(zds->ddictLocal) + zds->inBuffSize + zds->outBuffSize; } @@ -2290,195 +2290,195 @@ size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds) MEM_STATIC size_t ZSTD_limitCopy(void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - size_t const length = MIN(dstCapacity, srcSize); - memcpy(dst, src, length); - return length; + size_t const length = MIN(dstCapacity, srcSize); + memcpy(dst, src, length); + return length; } size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input) { - const char* const istart = (const char*)(input->src) + input->pos; - const char* const iend = (const char*)(input->src) + input->size; - const char* ip = istart; - char* const ostart = (char*)(output->dst) + output->pos; - char* const oend = (char*)(output->dst) + output->size; - char* op = ostart; - U32 someMoreWork = 1; + const char* const istart = (const char*)(input->src) + input->pos; + const char* const iend = (const char*)(input->src) + input->size; + const char* ip = istart; + char* const ostart = (char*)(output->dst) + output->pos; + char* const oend = (char*)(output->dst) + output->size; + char* op = ostart; + U32 someMoreWork = 1; #if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1) - if (zds->legacyVersion) - return ZSTD_decompressLegacyStream(zds->legacyContext, zds->legacyVersion, output, input); + if (zds->legacyVersion) + return ZSTD_decompressLegacyStream(zds->legacyContext, zds->legacyVersion, output, input); #endif - while (someMoreWork) { - switch(zds->stage) - { - case zdss_init : - ZSTD_resetDStream(zds); /* transparent reset on starting decoding a new frame */ - /* fall-through */ + while (someMoreWork) { + switch(zds->stage) + { + case zdss_init : + ZSTD_resetDStream(zds); /* transparent reset on starting decoding a new frame */ + /* fall-through */ - case zdss_loadHeader : - { size_t const hSize = ZSTD_getFrameParams(&zds->fParams, zds->headerBuffer, zds->lhSize); - if (ZSTD_isError(hSize)) + case zdss_loadHeader : + { size_t const hSize = ZSTD_getFrameParams(&zds->fParams, zds->headerBuffer, zds->lhSize); + if (ZSTD_isError(hSize)) #if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1) - { U32 const legacyVersion = ZSTD_isLegacy(istart, iend-istart); - if (legacyVersion) { - const void* const dict = zds->ddict ? zds->ddict->dictContent : NULL; - size_t const dictSize = zds->ddict ? zds->ddict->dictSize : 0; - CHECK_F(ZSTD_initLegacyStream(&zds->legacyContext, zds->previousLegacyVersion, legacyVersion, - dict, dictSize)); - zds->legacyVersion = zds->previousLegacyVersion = legacyVersion; - return ZSTD_decompressLegacyStream(zds->legacyContext, zds->legacyVersion, output, input); - } else { - return hSize; /* error */ - } } + { U32 const legacyVersion = ZSTD_isLegacy(istart, iend-istart); + if (legacyVersion) { + const void* const dict = zds->ddict ? zds->ddict->dictContent : NULL; + size_t const dictSize = zds->ddict ? zds->ddict->dictSize : 0; + CHECK_F(ZSTD_initLegacyStream(&zds->legacyContext, zds->previousLegacyVersion, legacyVersion, + dict, dictSize)); + zds->legacyVersion = zds->previousLegacyVersion = legacyVersion; + return ZSTD_decompressLegacyStream(zds->legacyContext, zds->legacyVersion, output, input); + } else { + return hSize; /* error */ + } } #else - return hSize; + return hSize; #endif - if (hSize != 0) { /* need more input */ - size_t const toLoad = hSize - zds->lhSize; /* if hSize!=0, hSize > zds->lhSize */ - if (toLoad > (size_t)(iend-ip)) { /* not enough input to load full header */ - memcpy(zds->headerBuffer + zds->lhSize, ip, iend-ip); - zds->lhSize += iend-ip; - input->pos = input->size; - return (MAX(ZSTD_frameHeaderSize_min, hSize) - zds->lhSize) + ZSTD_blockHeaderSize; /* remaining header bytes + next block header */ - } - memcpy(zds->headerBuffer + zds->lhSize, ip, toLoad); zds->lhSize = hSize; ip += toLoad; - break; - } } + if (hSize != 0) { /* need more input */ + size_t const toLoad = hSize - zds->lhSize; /* if hSize!=0, hSize > zds->lhSize */ + if (toLoad > (size_t)(iend-ip)) { /* not enough input to load full header */ + memcpy(zds->headerBuffer + zds->lhSize, ip, iend-ip); + zds->lhSize += iend-ip; + input->pos = input->size; + return (MAX(ZSTD_frameHeaderSize_min, hSize) - zds->lhSize) + ZSTD_blockHeaderSize; /* remaining header bytes + next block header */ + } + memcpy(zds->headerBuffer + zds->lhSize, ip, toLoad); zds->lhSize = hSize; ip += toLoad; + break; + } } - /* check for single-pass mode opportunity */ - if (zds->fParams.frameContentSize && zds->fParams.windowSize /* skippable frame if == 0 */ - && (U64)(size_t)(oend-op) >= zds->fParams.frameContentSize) { - size_t const cSize = ZSTD_findFrameCompressedSize(istart, iend-istart); - if (cSize <= (size_t)(iend-istart)) { - size_t const decompressedSize = ZSTD_decompress_usingDDict(zds->dctx, op, oend-op, istart, cSize, zds->ddict); - if (ZSTD_isError(decompressedSize)) return decompressedSize; - ip = istart + cSize; - op += decompressedSize; - zds->dctx->expected = 0; - zds->stage = zdss_init; - someMoreWork = 0; - break; - } } + /* check for single-pass mode opportunity */ + if (zds->fParams.frameContentSize && zds->fParams.windowSize /* skippable frame if == 0 */ + && (U64)(size_t)(oend-op) >= zds->fParams.frameContentSize) { + size_t const cSize = ZSTD_findFrameCompressedSize(istart, iend-istart); + if (cSize <= (size_t)(iend-istart)) { + size_t const decompressedSize = ZSTD_decompress_usingDDict(zds->dctx, op, oend-op, istart, cSize, zds->ddict); + if (ZSTD_isError(decompressedSize)) return decompressedSize; + ip = istart + cSize; + op += decompressedSize; + zds->dctx->expected = 0; + zds->stage = zdss_init; + someMoreWork = 0; + break; + } } - /* Consume header */ - ZSTD_refDDict(zds->dctx, zds->ddict); - { size_t const h1Size = ZSTD_nextSrcSizeToDecompress(zds->dctx); /* == ZSTD_frameHeaderSize_prefix */ - CHECK_F(ZSTD_decompressContinue(zds->dctx, NULL, 0, zds->headerBuffer, h1Size)); - { size_t const h2Size = ZSTD_nextSrcSizeToDecompress(zds->dctx); - CHECK_F(ZSTD_decompressContinue(zds->dctx, NULL, 0, zds->headerBuffer+h1Size, h2Size)); - } } + /* Consume header */ + ZSTD_refDDict(zds->dctx, zds->ddict); + { size_t const h1Size = ZSTD_nextSrcSizeToDecompress(zds->dctx); /* == ZSTD_frameHeaderSize_prefix */ + CHECK_F(ZSTD_decompressContinue(zds->dctx, NULL, 0, zds->headerBuffer, h1Size)); + { size_t const h2Size = ZSTD_nextSrcSizeToDecompress(zds->dctx); + CHECK_F(ZSTD_decompressContinue(zds->dctx, NULL, 0, zds->headerBuffer+h1Size, h2Size)); + } } - zds->fParams.windowSize = MAX(zds->fParams.windowSize, 1U << ZSTD_WINDOWLOG_ABSOLUTEMIN); - if (zds->fParams.windowSize > zds->maxWindowSize) return ERROR(frameParameter_windowTooLarge); + zds->fParams.windowSize = MAX(zds->fParams.windowSize, 1U << ZSTD_WINDOWLOG_ABSOLUTEMIN); + if (zds->fParams.windowSize > zds->maxWindowSize) return ERROR(frameParameter_windowTooLarge); - /* Adapt buffer sizes to frame header instructions */ - { size_t const blockSize = MIN(zds->fParams.windowSize, ZSTD_BLOCKSIZE_ABSOLUTEMAX); - size_t const neededOutSize = zds->fParams.windowSize + blockSize + WILDCOPY_OVERLENGTH * 2; - zds->blockSize = blockSize; - if (zds->inBuffSize < blockSize) { - ZSTD_free(zds->inBuff, zds->customMem); - zds->inBuffSize = blockSize; - zds->inBuff = (char*)ZSTD_malloc(blockSize, zds->customMem); - if (zds->inBuff == NULL) return ERROR(memory_allocation); - } - if (zds->outBuffSize < neededOutSize) { - ZSTD_free(zds->outBuff, zds->customMem); - zds->outBuffSize = neededOutSize; - zds->outBuff = (char*)ZSTD_malloc(neededOutSize, zds->customMem); - if (zds->outBuff == NULL) return ERROR(memory_allocation); - } } - zds->stage = zdss_read; - /* pass-through */ + /* Adapt buffer sizes to frame header instructions */ + { size_t const blockSize = MIN(zds->fParams.windowSize, ZSTD_BLOCKSIZE_ABSOLUTEMAX); + size_t const neededOutSize = zds->fParams.windowSize + blockSize + WILDCOPY_OVERLENGTH * 2; + zds->blockSize = blockSize; + if (zds->inBuffSize < blockSize) { + ZSTD_free(zds->inBuff, zds->customMem); + zds->inBuffSize = blockSize; + zds->inBuff = (char*)ZSTD_malloc(blockSize, zds->customMem); + if (zds->inBuff == NULL) return ERROR(memory_allocation); + } + if (zds->outBuffSize < neededOutSize) { + ZSTD_free(zds->outBuff, zds->customMem); + zds->outBuffSize = neededOutSize; + zds->outBuff = (char*)ZSTD_malloc(neededOutSize, zds->customMem); + if (zds->outBuff == NULL) return ERROR(memory_allocation); + } } + zds->stage = zdss_read; + /* pass-through */ - case zdss_read: - { size_t const neededInSize = ZSTD_nextSrcSizeToDecompress(zds->dctx); - if (neededInSize==0) { /* end of frame */ - zds->stage = zdss_init; - someMoreWork = 0; - break; - } - if ((size_t)(iend-ip) >= neededInSize) { /* decode directly from src */ - const int isSkipFrame = ZSTD_isSkipFrame(zds->dctx); - size_t const decodedSize = ZSTD_decompressContinue(zds->dctx, - zds->outBuff + zds->outStart, (isSkipFrame ? 0 : zds->outBuffSize - zds->outStart), - ip, neededInSize); - if (ZSTD_isError(decodedSize)) return decodedSize; - ip += neededInSize; - if (!decodedSize && !isSkipFrame) break; /* this was just a header */ - zds->outEnd = zds->outStart + decodedSize; - zds->stage = zdss_flush; - break; - } - if (ip==iend) { someMoreWork = 0; break; } /* no more input */ - zds->stage = zdss_load; - /* pass-through */ - } + case zdss_read: + { size_t const neededInSize = ZSTD_nextSrcSizeToDecompress(zds->dctx); + if (neededInSize==0) { /* end of frame */ + zds->stage = zdss_init; + someMoreWork = 0; + break; + } + if ((size_t)(iend-ip) >= neededInSize) { /* decode directly from src */ + const int isSkipFrame = ZSTD_isSkipFrame(zds->dctx); + size_t const decodedSize = ZSTD_decompressContinue(zds->dctx, + zds->outBuff + zds->outStart, (isSkipFrame ? 0 : zds->outBuffSize - zds->outStart), + ip, neededInSize); + if (ZSTD_isError(decodedSize)) return decodedSize; + ip += neededInSize; + if (!decodedSize && !isSkipFrame) break; /* this was just a header */ + zds->outEnd = zds->outStart + decodedSize; + zds->stage = zdss_flush; + break; + } + if (ip==iend) { someMoreWork = 0; break; } /* no more input */ + zds->stage = zdss_load; + /* pass-through */ + } - case zdss_load: - { size_t const neededInSize = ZSTD_nextSrcSizeToDecompress(zds->dctx); - size_t const toLoad = neededInSize - zds->inPos; /* should always be <= remaining space within inBuff */ - size_t loadedSize; - if (toLoad > zds->inBuffSize - zds->inPos) return ERROR(corruption_detected); /* should never happen */ - loadedSize = ZSTD_limitCopy(zds->inBuff + zds->inPos, toLoad, ip, iend-ip); - ip += loadedSize; - zds->inPos += loadedSize; - if (loadedSize < toLoad) { someMoreWork = 0; break; } /* not enough input, wait for more */ + case zdss_load: + { size_t const neededInSize = ZSTD_nextSrcSizeToDecompress(zds->dctx); + size_t const toLoad = neededInSize - zds->inPos; /* should always be <= remaining space within inBuff */ + size_t loadedSize; + if (toLoad > zds->inBuffSize - zds->inPos) return ERROR(corruption_detected); /* should never happen */ + loadedSize = ZSTD_limitCopy(zds->inBuff + zds->inPos, toLoad, ip, iend-ip); + ip += loadedSize; + zds->inPos += loadedSize; + if (loadedSize < toLoad) { someMoreWork = 0; break; } /* not enough input, wait for more */ - /* decode loaded input */ - { const int isSkipFrame = ZSTD_isSkipFrame(zds->dctx); - size_t const decodedSize = ZSTD_decompressContinue(zds->dctx, - zds->outBuff + zds->outStart, zds->outBuffSize - zds->outStart, - zds->inBuff, neededInSize); - if (ZSTD_isError(decodedSize)) return decodedSize; - zds->inPos = 0; /* input is consumed */ - if (!decodedSize && !isSkipFrame) { zds->stage = zdss_read; break; } /* this was just a header */ - zds->outEnd = zds->outStart + decodedSize; - zds->stage = zdss_flush; - /* pass-through */ - } } + /* decode loaded input */ + { const int isSkipFrame = ZSTD_isSkipFrame(zds->dctx); + size_t const decodedSize = ZSTD_decompressContinue(zds->dctx, + zds->outBuff + zds->outStart, zds->outBuffSize - zds->outStart, + zds->inBuff, neededInSize); + if (ZSTD_isError(decodedSize)) return decodedSize; + zds->inPos = 0; /* input is consumed */ + if (!decodedSize && !isSkipFrame) { zds->stage = zdss_read; break; } /* this was just a header */ + zds->outEnd = zds->outStart + decodedSize; + zds->stage = zdss_flush; + /* pass-through */ + } } - case zdss_flush: - { size_t const toFlushSize = zds->outEnd - zds->outStart; - size_t const flushedSize = ZSTD_limitCopy(op, oend-op, zds->outBuff + zds->outStart, toFlushSize); - op += flushedSize; - zds->outStart += flushedSize; - if (flushedSize == toFlushSize) { /* flush completed */ - zds->stage = zdss_read; - if (zds->outStart + zds->blockSize > zds->outBuffSize) - zds->outStart = zds->outEnd = 0; - break; - } - /* cannot complete flush */ - someMoreWork = 0; - break; - } - default: return ERROR(GENERIC); /* impossible */ - } } + case zdss_flush: + { size_t const toFlushSize = zds->outEnd - zds->outStart; + size_t const flushedSize = ZSTD_limitCopy(op, oend-op, zds->outBuff + zds->outStart, toFlushSize); + op += flushedSize; + zds->outStart += flushedSize; + if (flushedSize == toFlushSize) { /* flush completed */ + zds->stage = zdss_read; + if (zds->outStart + zds->blockSize > zds->outBuffSize) + zds->outStart = zds->outEnd = 0; + break; + } + /* cannot complete flush */ + someMoreWork = 0; + break; + } + default: return ERROR(GENERIC); /* impossible */ + } } - /* result */ - input->pos += (size_t)(ip-istart); - output->pos += (size_t)(op-ostart); - { size_t nextSrcSizeHint = ZSTD_nextSrcSizeToDecompress(zds->dctx); - if (!nextSrcSizeHint) { /* frame fully decoded */ - if (zds->outEnd == zds->outStart) { /* output fully flushed */ - if (zds->hostageByte) { - if (input->pos >= input->size) { zds->stage = zdss_read; return 1; } /* can't release hostage (not present) */ - input->pos++; /* release hostage */ - } - return 0; - } - if (!zds->hostageByte) { /* output not fully flushed; keep last byte as hostage; will be released when all output is flushed */ - input->pos--; /* note : pos > 0, otherwise, impossible to finish reading last block */ - zds->hostageByte=1; - } - return 1; - } - nextSrcSizeHint += ZSTD_blockHeaderSize * (ZSTD_nextInputType(zds->dctx) == ZSTDnit_block); /* preload header of next block */ - if (zds->inPos > nextSrcSizeHint) return ERROR(GENERIC); /* should never happen */ - nextSrcSizeHint -= zds->inPos; /* already loaded*/ - return nextSrcSizeHint; - } + /* result */ + input->pos += (size_t)(ip-istart); + output->pos += (size_t)(op-ostart); + { size_t nextSrcSizeHint = ZSTD_nextSrcSizeToDecompress(zds->dctx); + if (!nextSrcSizeHint) { /* frame fully decoded */ + if (zds->outEnd == zds->outStart) { /* output fully flushed */ + if (zds->hostageByte) { + if (input->pos >= input->size) { zds->stage = zdss_read; return 1; } /* can't release hostage (not present) */ + input->pos++; /* release hostage */ + } + return 0; + } + if (!zds->hostageByte) { /* output not fully flushed; keep last byte as hostage; will be released when all output is flushed */ + input->pos--; /* note : pos > 0, otherwise, impossible to finish reading last block */ + zds->hostageByte=1; + } + return 1; + } + nextSrcSizeHint += ZSTD_blockHeaderSize * (ZSTD_nextInputType(zds->dctx) == ZSTDnit_block); /* preload header of next block */ + if (zds->inPos > nextSrcSizeHint) return ERROR(GENERIC); /* should never happen */ + nextSrcSizeHint -= zds->inPos; /* already loaded*/ + return nextSrcSizeHint; + } } diff --git a/contrib/linux-kernel/lib/zstd_errors.h b/contrib/linux-kernel/lib/zstd_errors.h index 3d579d969..1cf0b0adb 100644 --- a/contrib/linux-kernel/lib/zstd_errors.h +++ b/contrib/linux-kernel/lib/zstd_errors.h @@ -62,8 +62,8 @@ typedef enum { } ZSTD_ErrorCode; /*! ZSTD_getErrorCode() : - convert a `size_t` function result into a `ZSTD_ErrorCode` enum type, - which can be used to compare directly with enum list published into "error_public.h" */ + convert a `size_t` function result into a `ZSTD_ErrorCode` enum type, + which can be used to compare directly with enum list published into "error_public.h" */ ZSTDERRORLIB_API ZSTD_ErrorCode ZSTD_getErrorCode(size_t functionResult); ZSTDERRORLIB_API const char* ZSTD_getErrorString(ZSTD_ErrorCode code); diff --git a/contrib/linux-kernel/lib/zstd_internal.h b/contrib/linux-kernel/lib/zstd_internal.h index 5c5b28732..8b2c27f53 100644 --- a/contrib/linux-kernel/lib/zstd_internal.h +++ b/contrib/linux-kernel/lib/zstd_internal.h @@ -117,27 +117,27 @@ typedef enum { set_basic, set_rle, set_compressed, set_repeat } symbolEncodingTy #define OffFSELog 8 static const U32 LL_bits[MaxLL+1] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 1, 1, 1, 1, 2, 2, 3, 3, 4, 6, 7, 8, 9,10,11,12, - 13,14,15,16 }; + 1, 1, 1, 1, 2, 2, 3, 3, 4, 6, 7, 8, 9,10,11,12, + 13,14,15,16 }; static const S16 LL_defaultNorm[MaxLL+1] = { 4, 3, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 1, 1, 1, - 2, 2, 2, 2, 2, 2, 2, 2, 2, 3, 2, 1, 1, 1, 1, 1, - -1,-1,-1,-1 }; + 2, 2, 2, 2, 2, 2, 2, 2, 2, 3, 2, 1, 1, 1, 1, 1, + -1,-1,-1,-1 }; #define LL_DEFAULTNORMLOG 6 /* for static allocation */ static const U32 LL_defaultNormLog = LL_DEFAULTNORMLOG; static const U32 ML_bits[MaxML+1] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 1, 1, 1, 1, 2, 2, 3, 3, 4, 4, 5, 7, 8, 9,10,11, - 12,13,14,15,16 }; + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 1, 1, 1, 1, 2, 2, 3, 3, 4, 4, 5, 7, 8, 9,10,11, + 12,13,14,15,16 }; static const S16 ML_defaultNorm[MaxML+1] = { 1, 4, 3, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,-1,-1, - -1,-1,-1,-1,-1 }; + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,-1,-1, + -1,-1,-1,-1,-1 }; #define ML_DEFAULTNORMLOG 6 /* for static allocation */ static const U32 ML_defaultNormLog = ML_DEFAULTNORMLOG; static const S16 OF_defaultNorm[MaxOff+1] = { 1, 1, 1, 1, 1, 1, 2, 2, 2, 1, 1, 1, 1, 1, 1, 1, - 1, 1, 1, 1, 1, 1, 1, 1,-1,-1,-1,-1,-1 }; + 1, 1, 1, 1, 1, 1, 1, 1,-1,-1,-1,-1,-1 }; #define OF_DEFAULTNORMLOG 5 /* for static allocation */ static const U32 OF_defaultNormLog = OF_DEFAULTNORMLOG; @@ -153,22 +153,22 @@ static void ZSTD_copy8(void* dst, const void* src) { memcpy(dst, src, 8); } #define WILDCOPY_OVERLENGTH 8 MEM_STATIC void ZSTD_wildcopy(void* dst, const void* src, ptrdiff_t length) { - const BYTE* ip = (const BYTE*)src; - BYTE* op = (BYTE*)dst; - BYTE* const oend = op + length; - do - COPY8(op, ip) - while (op < oend); + const BYTE* ip = (const BYTE*)src; + BYTE* op = (BYTE*)dst; + BYTE* const oend = op + length; + do + COPY8(op, ip) + while (op < oend); } MEM_STATIC void ZSTD_wildcopy_e(void* dst, const void* src, void* dstEnd) /* should be faster for decoding, but strangely, not verified on all platform */ { - const BYTE* ip = (const BYTE*)src; - BYTE* op = (BYTE*)dst; - BYTE* const oend = (BYTE*)dstEnd; - do - COPY8(op, ip) - while (op < oend); + const BYTE* ip = (const BYTE*)src; + BYTE* op = (BYTE*)dst; + BYTE* const oend = (BYTE*)dstEnd; + do + COPY8(op, ip) + while (op < oend); } @@ -178,58 +178,58 @@ MEM_STATIC void ZSTD_wildcopy_e(void* dst, const void* src, void* dstEnd) /* s typedef struct ZSTD_stats_s ZSTD_stats_t; typedef struct { - U32 off; - U32 len; + U32 off; + U32 len; } ZSTD_match_t; typedef struct { - U32 price; - U32 off; - U32 mlen; - U32 litlen; - U32 rep[ZSTD_REP_NUM]; + U32 price; + U32 off; + U32 mlen; + U32 litlen; + U32 rep[ZSTD_REP_NUM]; } ZSTD_optimal_t; typedef struct seqDef_s { - U32 offset; - U16 litLength; - U16 matchLength; + U32 offset; + U16 litLength; + U16 matchLength; } seqDef; typedef struct { - seqDef* sequencesStart; - seqDef* sequences; - BYTE* litStart; - BYTE* lit; - BYTE* llCode; - BYTE* mlCode; - BYTE* ofCode; - U32 longLengthID; /* 0 == no longLength; 1 == Lit.longLength; 2 == Match.longLength; */ - U32 longLengthPos; - /* opt */ - ZSTD_optimal_t* priceTable; - ZSTD_match_t* matchTable; - U32* matchLengthFreq; - U32* litLengthFreq; - U32* litFreq; - U32* offCodeFreq; - U32 matchLengthSum; - U32 matchSum; - U32 litLengthSum; - U32 litSum; - U32 offCodeSum; - U32 log2matchLengthSum; - U32 log2matchSum; - U32 log2litLengthSum; - U32 log2litSum; - U32 log2offCodeSum; - U32 factor; - U32 staticPrices; - U32 cachedPrice; - U32 cachedLitLength; - const BYTE* cachedLiterals; + seqDef* sequencesStart; + seqDef* sequences; + BYTE* litStart; + BYTE* lit; + BYTE* llCode; + BYTE* mlCode; + BYTE* ofCode; + U32 longLengthID; /* 0 == no longLength; 1 == Lit.longLength; 2 == Match.longLength; */ + U32 longLengthPos; + /* opt */ + ZSTD_optimal_t* priceTable; + ZSTD_match_t* matchTable; + U32* matchLengthFreq; + U32* litLengthFreq; + U32* litFreq; + U32* offCodeFreq; + U32 matchLengthSum; + U32 matchSum; + U32 litLengthSum; + U32 litSum; + U32 offCodeSum; + U32 log2matchLengthSum; + U32 log2matchSum; + U32 log2litLengthSum; + U32 log2litSum; + U32 log2offCodeSum; + U32 factor; + U32 staticPrices; + U32 cachedPrice; + U32 cachedLitLength; + const BYTE* cachedLiterals; } seqStore_t; const seqStore_t* ZSTD_getSeqStore(const ZSTD_CCtx* ctx); @@ -251,22 +251,22 @@ void ZSTD_free(void* ptr, ZSTD_customMem customMem); MEM_STATIC U32 ZSTD_highbit32(U32 val) { # if defined(_MSC_VER) /* Visual */ - unsigned long r=0; - _BitScanReverse(&r, val); - return (unsigned)r; + unsigned long r=0; + _BitScanReverse(&r, val); + return (unsigned)r; # elif defined(__GNUC__) && (__GNUC__ >= 3) /* GCC Intrinsic */ - return 31 - __builtin_clz(val); + return 31 - __builtin_clz(val); # else /* Software version */ - static const int DeBruijnClz[32] = { 0, 9, 1, 10, 13, 21, 2, 29, 11, 14, 16, 18, 22, 25, 3, 30, 8, 12, 20, 28, 15, 17, 24, 7, 19, 27, 23, 6, 26, 5, 4, 31 }; - U32 v = val; - int r; - v |= v >> 1; - v |= v >> 2; - v |= v >> 4; - v |= v >> 8; - v |= v >> 16; - r = DeBruijnClz[(U32)(v * 0x07C4ACDDU) >> 27]; - return r; + static const int DeBruijnClz[32] = { 0, 9, 1, 10, 13, 21, 2, 29, 11, 14, 16, 18, 22, 25, 3, 30, 8, 12, 20, 28, 15, 17, 24, 7, 19, 27, 23, 6, 26, 5, 4, 31 }; + U32 v = val; + int r; + v |= v >> 1; + v |= v >> 2; + v |= v >> 4; + v |= v >> 8; + v |= v >> 16; + r = DeBruijnClz[(U32)(v * 0x07C4ACDDU) >> 27]; + return r; # endif } diff --git a/contrib/linux-kernel/lib/zstd_opt.h b/contrib/linux-kernel/lib/zstd_opt.h index 543761191..297a71559 100644 --- a/contrib/linux-kernel/lib/zstd_opt.h +++ b/contrib/linux-kernel/lib/zstd_opt.h @@ -24,181 +24,181 @@ ***************************************/ FORCE_INLINE void ZSTD_setLog2Prices(seqStore_t* ssPtr) { - ssPtr->log2matchLengthSum = ZSTD_highbit32(ssPtr->matchLengthSum+1); - ssPtr->log2litLengthSum = ZSTD_highbit32(ssPtr->litLengthSum+1); - ssPtr->log2litSum = ZSTD_highbit32(ssPtr->litSum+1); - ssPtr->log2offCodeSum = ZSTD_highbit32(ssPtr->offCodeSum+1); - ssPtr->factor = 1 + ((ssPtr->litSum>>5) / ssPtr->litLengthSum) + ((ssPtr->litSum<<1) / (ssPtr->litSum + ssPtr->matchSum)); + ssPtr->log2matchLengthSum = ZSTD_highbit32(ssPtr->matchLengthSum+1); + ssPtr->log2litLengthSum = ZSTD_highbit32(ssPtr->litLengthSum+1); + ssPtr->log2litSum = ZSTD_highbit32(ssPtr->litSum+1); + ssPtr->log2offCodeSum = ZSTD_highbit32(ssPtr->offCodeSum+1); + ssPtr->factor = 1 + ((ssPtr->litSum>>5) / ssPtr->litLengthSum) + ((ssPtr->litSum<<1) / (ssPtr->litSum + ssPtr->matchSum)); } MEM_STATIC void ZSTD_rescaleFreqs(seqStore_t* ssPtr, const BYTE* src, size_t srcSize) { - unsigned u; + unsigned u; - ssPtr->cachedLiterals = NULL; - ssPtr->cachedPrice = ssPtr->cachedLitLength = 0; - ssPtr->staticPrices = 0; + ssPtr->cachedLiterals = NULL; + ssPtr->cachedPrice = ssPtr->cachedLitLength = 0; + ssPtr->staticPrices = 0; - if (ssPtr->litLengthSum == 0) { - if (srcSize <= 1024) ssPtr->staticPrices = 1; + if (ssPtr->litLengthSum == 0) { + if (srcSize <= 1024) ssPtr->staticPrices = 1; - for (u=0; u<=MaxLit; u++) - ssPtr->litFreq[u] = 0; - for (u=0; ulitFreq[src[u]]++; + for (u=0; u<=MaxLit; u++) + ssPtr->litFreq[u] = 0; + for (u=0; ulitFreq[src[u]]++; - ssPtr->litSum = 0; - ssPtr->litLengthSum = MaxLL+1; - ssPtr->matchLengthSum = MaxML+1; - ssPtr->offCodeSum = (MaxOff+1); - ssPtr->matchSum = (ZSTD_LITFREQ_ADD<litSum = 0; + ssPtr->litLengthSum = MaxLL+1; + ssPtr->matchLengthSum = MaxML+1; + ssPtr->offCodeSum = (MaxOff+1); + ssPtr->matchSum = (ZSTD_LITFREQ_ADD<litFreq[u] = 1 + (ssPtr->litFreq[u]>>ZSTD_FREQ_DIV); - ssPtr->litSum += ssPtr->litFreq[u]; - } - for (u=0; u<=MaxLL; u++) - ssPtr->litLengthFreq[u] = 1; - for (u=0; u<=MaxML; u++) - ssPtr->matchLengthFreq[u] = 1; - for (u=0; u<=MaxOff; u++) - ssPtr->offCodeFreq[u] = 1; - } else { - ssPtr->matchLengthSum = 0; - ssPtr->litLengthSum = 0; - ssPtr->offCodeSum = 0; - ssPtr->matchSum = 0; - ssPtr->litSum = 0; + for (u=0; u<=MaxLit; u++) { + ssPtr->litFreq[u] = 1 + (ssPtr->litFreq[u]>>ZSTD_FREQ_DIV); + ssPtr->litSum += ssPtr->litFreq[u]; + } + for (u=0; u<=MaxLL; u++) + ssPtr->litLengthFreq[u] = 1; + for (u=0; u<=MaxML; u++) + ssPtr->matchLengthFreq[u] = 1; + for (u=0; u<=MaxOff; u++) + ssPtr->offCodeFreq[u] = 1; + } else { + ssPtr->matchLengthSum = 0; + ssPtr->litLengthSum = 0; + ssPtr->offCodeSum = 0; + ssPtr->matchSum = 0; + ssPtr->litSum = 0; - for (u=0; u<=MaxLit; u++) { - ssPtr->litFreq[u] = 1 + (ssPtr->litFreq[u]>>(ZSTD_FREQ_DIV+1)); - ssPtr->litSum += ssPtr->litFreq[u]; - } - for (u=0; u<=MaxLL; u++) { - ssPtr->litLengthFreq[u] = 1 + (ssPtr->litLengthFreq[u]>>(ZSTD_FREQ_DIV+1)); - ssPtr->litLengthSum += ssPtr->litLengthFreq[u]; - } - for (u=0; u<=MaxML; u++) { - ssPtr->matchLengthFreq[u] = 1 + (ssPtr->matchLengthFreq[u]>>ZSTD_FREQ_DIV); - ssPtr->matchLengthSum += ssPtr->matchLengthFreq[u]; - ssPtr->matchSum += ssPtr->matchLengthFreq[u] * (u + 3); - } - ssPtr->matchSum *= ZSTD_LITFREQ_ADD; - for (u=0; u<=MaxOff; u++) { - ssPtr->offCodeFreq[u] = 1 + (ssPtr->offCodeFreq[u]>>ZSTD_FREQ_DIV); - ssPtr->offCodeSum += ssPtr->offCodeFreq[u]; - } - } + for (u=0; u<=MaxLit; u++) { + ssPtr->litFreq[u] = 1 + (ssPtr->litFreq[u]>>(ZSTD_FREQ_DIV+1)); + ssPtr->litSum += ssPtr->litFreq[u]; + } + for (u=0; u<=MaxLL; u++) { + ssPtr->litLengthFreq[u] = 1 + (ssPtr->litLengthFreq[u]>>(ZSTD_FREQ_DIV+1)); + ssPtr->litLengthSum += ssPtr->litLengthFreq[u]; + } + for (u=0; u<=MaxML; u++) { + ssPtr->matchLengthFreq[u] = 1 + (ssPtr->matchLengthFreq[u]>>ZSTD_FREQ_DIV); + ssPtr->matchLengthSum += ssPtr->matchLengthFreq[u]; + ssPtr->matchSum += ssPtr->matchLengthFreq[u] * (u + 3); + } + ssPtr->matchSum *= ZSTD_LITFREQ_ADD; + for (u=0; u<=MaxOff; u++) { + ssPtr->offCodeFreq[u] = 1 + (ssPtr->offCodeFreq[u]>>ZSTD_FREQ_DIV); + ssPtr->offCodeSum += ssPtr->offCodeFreq[u]; + } + } - ZSTD_setLog2Prices(ssPtr); + ZSTD_setLog2Prices(ssPtr); } FORCE_INLINE U32 ZSTD_getLiteralPrice(seqStore_t* ssPtr, U32 litLength, const BYTE* literals) { - U32 price, u; + U32 price, u; - if (ssPtr->staticPrices) - return ZSTD_highbit32((U32)litLength+1) + (litLength*6); + if (ssPtr->staticPrices) + return ZSTD_highbit32((U32)litLength+1) + (litLength*6); - if (litLength == 0) - return ssPtr->log2litLengthSum - ZSTD_highbit32(ssPtr->litLengthFreq[0]+1); + if (litLength == 0) + return ssPtr->log2litLengthSum - ZSTD_highbit32(ssPtr->litLengthFreq[0]+1); - /* literals */ - if (ssPtr->cachedLiterals == literals) { - U32 const additional = litLength - ssPtr->cachedLitLength; - const BYTE* literals2 = ssPtr->cachedLiterals + ssPtr->cachedLitLength; - price = ssPtr->cachedPrice + additional * ssPtr->log2litSum; - for (u=0; u < additional; u++) - price -= ZSTD_highbit32(ssPtr->litFreq[literals2[u]]+1); - ssPtr->cachedPrice = price; - ssPtr->cachedLitLength = litLength; - } else { - price = litLength * ssPtr->log2litSum; - for (u=0; u < litLength; u++) - price -= ZSTD_highbit32(ssPtr->litFreq[literals[u]]+1); + /* literals */ + if (ssPtr->cachedLiterals == literals) { + U32 const additional = litLength - ssPtr->cachedLitLength; + const BYTE* literals2 = ssPtr->cachedLiterals + ssPtr->cachedLitLength; + price = ssPtr->cachedPrice + additional * ssPtr->log2litSum; + for (u=0; u < additional; u++) + price -= ZSTD_highbit32(ssPtr->litFreq[literals2[u]]+1); + ssPtr->cachedPrice = price; + ssPtr->cachedLitLength = litLength; + } else { + price = litLength * ssPtr->log2litSum; + for (u=0; u < litLength; u++) + price -= ZSTD_highbit32(ssPtr->litFreq[literals[u]]+1); - if (litLength >= 12) { - ssPtr->cachedLiterals = literals; - ssPtr->cachedPrice = price; - ssPtr->cachedLitLength = litLength; - } - } + if (litLength >= 12) { + ssPtr->cachedLiterals = literals; + ssPtr->cachedPrice = price; + ssPtr->cachedLitLength = litLength; + } + } - /* literal Length */ - { const BYTE LL_deltaCode = 19; - const BYTE llCode = (litLength>63) ? (BYTE)ZSTD_highbit32(litLength) + LL_deltaCode : LL_Code[litLength]; - price += LL_bits[llCode] + ssPtr->log2litLengthSum - ZSTD_highbit32(ssPtr->litLengthFreq[llCode]+1); - } + /* literal Length */ + { const BYTE LL_deltaCode = 19; + const BYTE llCode = (litLength>63) ? (BYTE)ZSTD_highbit32(litLength) + LL_deltaCode : LL_Code[litLength]; + price += LL_bits[llCode] + ssPtr->log2litLengthSum - ZSTD_highbit32(ssPtr->litLengthFreq[llCode]+1); + } - return price; + return price; } FORCE_INLINE U32 ZSTD_getPrice(seqStore_t* seqStorePtr, U32 litLength, const BYTE* literals, U32 offset, U32 matchLength, const int ultra) { - /* offset */ - U32 price; - BYTE const offCode = (BYTE)ZSTD_highbit32(offset+1); + /* offset */ + U32 price; + BYTE const offCode = (BYTE)ZSTD_highbit32(offset+1); - if (seqStorePtr->staticPrices) - return ZSTD_getLiteralPrice(seqStorePtr, litLength, literals) + ZSTD_highbit32((U32)matchLength+1) + 16 + offCode; + if (seqStorePtr->staticPrices) + return ZSTD_getLiteralPrice(seqStorePtr, litLength, literals) + ZSTD_highbit32((U32)matchLength+1) + 16 + offCode; - price = offCode + seqStorePtr->log2offCodeSum - ZSTD_highbit32(seqStorePtr->offCodeFreq[offCode]+1); - if (!ultra && offCode >= 20) price += (offCode-19)*2; + price = offCode + seqStorePtr->log2offCodeSum - ZSTD_highbit32(seqStorePtr->offCodeFreq[offCode]+1); + if (!ultra && offCode >= 20) price += (offCode-19)*2; - /* match Length */ - { const BYTE ML_deltaCode = 36; - const BYTE mlCode = (matchLength>127) ? (BYTE)ZSTD_highbit32(matchLength) + ML_deltaCode : ML_Code[matchLength]; - price += ML_bits[mlCode] + seqStorePtr->log2matchLengthSum - ZSTD_highbit32(seqStorePtr->matchLengthFreq[mlCode]+1); - } + /* match Length */ + { const BYTE ML_deltaCode = 36; + const BYTE mlCode = (matchLength>127) ? (BYTE)ZSTD_highbit32(matchLength) + ML_deltaCode : ML_Code[matchLength]; + price += ML_bits[mlCode] + seqStorePtr->log2matchLengthSum - ZSTD_highbit32(seqStorePtr->matchLengthFreq[mlCode]+1); + } - return price + ZSTD_getLiteralPrice(seqStorePtr, litLength, literals) + seqStorePtr->factor; + return price + ZSTD_getLiteralPrice(seqStorePtr, litLength, literals) + seqStorePtr->factor; } MEM_STATIC void ZSTD_updatePrice(seqStore_t* seqStorePtr, U32 litLength, const BYTE* literals, U32 offset, U32 matchLength) { - U32 u; + U32 u; - /* literals */ - seqStorePtr->litSum += litLength*ZSTD_LITFREQ_ADD; - for (u=0; u < litLength; u++) - seqStorePtr->litFreq[literals[u]] += ZSTD_LITFREQ_ADD; + /* literals */ + seqStorePtr->litSum += litLength*ZSTD_LITFREQ_ADD; + for (u=0; u < litLength; u++) + seqStorePtr->litFreq[literals[u]] += ZSTD_LITFREQ_ADD; - /* literal Length */ - { const BYTE LL_deltaCode = 19; - const BYTE llCode = (litLength>63) ? (BYTE)ZSTD_highbit32(litLength) + LL_deltaCode : LL_Code[litLength]; - seqStorePtr->litLengthFreq[llCode]++; - seqStorePtr->litLengthSum++; - } + /* literal Length */ + { const BYTE LL_deltaCode = 19; + const BYTE llCode = (litLength>63) ? (BYTE)ZSTD_highbit32(litLength) + LL_deltaCode : LL_Code[litLength]; + seqStorePtr->litLengthFreq[llCode]++; + seqStorePtr->litLengthSum++; + } - /* match offset */ - { BYTE const offCode = (BYTE)ZSTD_highbit32(offset+1); - seqStorePtr->offCodeSum++; - seqStorePtr->offCodeFreq[offCode]++; - } + /* match offset */ + { BYTE const offCode = (BYTE)ZSTD_highbit32(offset+1); + seqStorePtr->offCodeSum++; + seqStorePtr->offCodeFreq[offCode]++; + } - /* match Length */ - { const BYTE ML_deltaCode = 36; - const BYTE mlCode = (matchLength>127) ? (BYTE)ZSTD_highbit32(matchLength) + ML_deltaCode : ML_Code[matchLength]; - seqStorePtr->matchLengthFreq[mlCode]++; - seqStorePtr->matchLengthSum++; - } + /* match Length */ + { const BYTE ML_deltaCode = 36; + const BYTE mlCode = (matchLength>127) ? (BYTE)ZSTD_highbit32(matchLength) + ML_deltaCode : ML_Code[matchLength]; + seqStorePtr->matchLengthFreq[mlCode]++; + seqStorePtr->matchLengthSum++; + } - ZSTD_setLog2Prices(seqStorePtr); + ZSTD_setLog2Prices(seqStorePtr); } #define SET_PRICE(pos, mlen_, offset_, litlen_, price_) \ - { \ - while (last_pos < pos) { opt[last_pos+1].price = ZSTD_MAX_PRICE; last_pos++; } \ - opt[pos].mlen = mlen_; \ - opt[pos].off = offset_; \ - opt[pos].litlen = litlen_; \ - opt[pos].price = price_; \ - } + { \ + while (last_pos < pos) { opt[last_pos+1].price = ZSTD_MAX_PRICE; last_pos++; } \ + opt[pos].mlen = mlen_; \ + opt[pos].off = offset_; \ + opt[pos].litlen = litlen_; \ + opt[pos].price = price_; \ + } @@ -207,19 +207,19 @@ MEM_STATIC void ZSTD_updatePrice(seqStore_t* seqStorePtr, U32 litLength, const B FORCE_INLINE U32 ZSTD_insertAndFindFirstIndexHash3 (ZSTD_CCtx* zc, const BYTE* ip) { - U32* const hashTable3 = zc->hashTable3; - U32 const hashLog3 = zc->hashLog3; - const BYTE* const base = zc->base; - U32 idx = zc->nextToUpdate3; - const U32 target = zc->nextToUpdate3 = (U32)(ip - base); - const size_t hash3 = ZSTD_hash3Ptr(ip, hashLog3); + U32* const hashTable3 = zc->hashTable3; + U32 const hashLog3 = zc->hashLog3; + const BYTE* const base = zc->base; + U32 idx = zc->nextToUpdate3; + const U32 target = zc->nextToUpdate3 = (U32)(ip - base); + const size_t hash3 = ZSTD_hash3Ptr(ip, hashLog3); - while(idx < target) { - hashTable3[ZSTD_hash3Ptr(base+idx, hashLog3)] = idx; - idx++; - } + while(idx < target) { + hashTable3[ZSTD_hash3Ptr(base+idx, hashLog3)] = idx; + idx++; + } - return hashTable3[hash3]; + return hashTable3[hash3]; } @@ -227,170 +227,170 @@ U32 ZSTD_insertAndFindFirstIndexHash3 (ZSTD_CCtx* zc, const BYTE* ip) * Binary Tree search ***************************************/ static U32 ZSTD_insertBtAndGetAllMatches ( - ZSTD_CCtx* zc, - const BYTE* const ip, const BYTE* const iLimit, - U32 nbCompares, const U32 mls, - U32 extDict, ZSTD_match_t* matches, const U32 minMatchLen) + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iLimit, + U32 nbCompares, const U32 mls, + U32 extDict, ZSTD_match_t* matches, const U32 minMatchLen) { - const BYTE* const base = zc->base; - const U32 current = (U32)(ip-base); - const U32 hashLog = zc->params.cParams.hashLog; - const size_t h = ZSTD_hashPtr(ip, hashLog, mls); - U32* const hashTable = zc->hashTable; - U32 matchIndex = hashTable[h]; - U32* const bt = zc->chainTable; - const U32 btLog = zc->params.cParams.chainLog - 1; - const U32 btMask= (1U << btLog) - 1; - size_t commonLengthSmaller=0, commonLengthLarger=0; - const BYTE* const dictBase = zc->dictBase; - const U32 dictLimit = zc->dictLimit; - const BYTE* const dictEnd = dictBase + dictLimit; - const BYTE* const prefixStart = base + dictLimit; - const U32 btLow = btMask >= current ? 0 : current - btMask; - const U32 windowLow = zc->lowLimit; - U32* smallerPtr = bt + 2*(current&btMask); - U32* largerPtr = bt + 2*(current&btMask) + 1; - U32 matchEndIdx = current+8; - U32 dummy32; /* to be nullified at the end */ - U32 mnum = 0; + const BYTE* const base = zc->base; + const U32 current = (U32)(ip-base); + const U32 hashLog = zc->params.cParams.hashLog; + const size_t h = ZSTD_hashPtr(ip, hashLog, mls); + U32* const hashTable = zc->hashTable; + U32 matchIndex = hashTable[h]; + U32* const bt = zc->chainTable; + const U32 btLog = zc->params.cParams.chainLog - 1; + const U32 btMask= (1U << btLog) - 1; + size_t commonLengthSmaller=0, commonLengthLarger=0; + const BYTE* const dictBase = zc->dictBase; + const U32 dictLimit = zc->dictLimit; + const BYTE* const dictEnd = dictBase + dictLimit; + const BYTE* const prefixStart = base + dictLimit; + const U32 btLow = btMask >= current ? 0 : current - btMask; + const U32 windowLow = zc->lowLimit; + U32* smallerPtr = bt + 2*(current&btMask); + U32* largerPtr = bt + 2*(current&btMask) + 1; + U32 matchEndIdx = current+8; + U32 dummy32; /* to be nullified at the end */ + U32 mnum = 0; - const U32 minMatch = (mls == 3) ? 3 : 4; - size_t bestLength = minMatchLen-1; + const U32 minMatch = (mls == 3) ? 3 : 4; + size_t bestLength = minMatchLen-1; - if (minMatch == 3) { /* HC3 match finder */ - U32 const matchIndex3 = ZSTD_insertAndFindFirstIndexHash3 (zc, ip); - if (matchIndex3>windowLow && (current - matchIndex3 < (1<<18))) { - const BYTE* match; - size_t currentMl=0; - if ((!extDict) || matchIndex3 >= dictLimit) { - match = base + matchIndex3; - if (match[bestLength] == ip[bestLength]) currentMl = ZSTD_count(ip, match, iLimit); - } else { - match = dictBase + matchIndex3; - if (MEM_readMINMATCH(match, MINMATCH) == MEM_readMINMATCH(ip, MINMATCH)) /* assumption : matchIndex3 <= dictLimit-4 (by table construction) */ - currentMl = ZSTD_count_2segments(ip+MINMATCH, match+MINMATCH, iLimit, dictEnd, prefixStart) + MINMATCH; - } + if (minMatch == 3) { /* HC3 match finder */ + U32 const matchIndex3 = ZSTD_insertAndFindFirstIndexHash3 (zc, ip); + if (matchIndex3>windowLow && (current - matchIndex3 < (1<<18))) { + const BYTE* match; + size_t currentMl=0; + if ((!extDict) || matchIndex3 >= dictLimit) { + match = base + matchIndex3; + if (match[bestLength] == ip[bestLength]) currentMl = ZSTD_count(ip, match, iLimit); + } else { + match = dictBase + matchIndex3; + if (MEM_readMINMATCH(match, MINMATCH) == MEM_readMINMATCH(ip, MINMATCH)) /* assumption : matchIndex3 <= dictLimit-4 (by table construction) */ + currentMl = ZSTD_count_2segments(ip+MINMATCH, match+MINMATCH, iLimit, dictEnd, prefixStart) + MINMATCH; + } - /* save best solution */ - if (currentMl > bestLength) { - bestLength = currentMl; - matches[mnum].off = ZSTD_REP_MOVE_OPT + current - matchIndex3; - matches[mnum].len = (U32)currentMl; - mnum++; - if (currentMl > ZSTD_OPT_NUM) goto update; - if (ip+currentMl == iLimit) goto update; /* best possible, and avoid read overflow*/ - } - } - } + /* save best solution */ + if (currentMl > bestLength) { + bestLength = currentMl; + matches[mnum].off = ZSTD_REP_MOVE_OPT + current - matchIndex3; + matches[mnum].len = (U32)currentMl; + mnum++; + if (currentMl > ZSTD_OPT_NUM) goto update; + if (ip+currentMl == iLimit) goto update; /* best possible, and avoid read overflow*/ + } + } + } - hashTable[h] = current; /* Update Hash Table */ + hashTable[h] = current; /* Update Hash Table */ - while (nbCompares-- && (matchIndex > windowLow)) { - U32* nextPtr = bt + 2*(matchIndex & btMask); - size_t matchLength = MIN(commonLengthSmaller, commonLengthLarger); /* guaranteed minimum nb of common bytes */ - const BYTE* match; + while (nbCompares-- && (matchIndex > windowLow)) { + U32* nextPtr = bt + 2*(matchIndex & btMask); + size_t matchLength = MIN(commonLengthSmaller, commonLengthLarger); /* guaranteed minimum nb of common bytes */ + const BYTE* match; - if ((!extDict) || (matchIndex+matchLength >= dictLimit)) { - match = base + matchIndex; - if (match[matchLength] == ip[matchLength]) { - matchLength += ZSTD_count(ip+matchLength+1, match+matchLength+1, iLimit) +1; - } - } else { - match = dictBase + matchIndex; - matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iLimit, dictEnd, prefixStart); - if (matchIndex+matchLength >= dictLimit) - match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ - } + if ((!extDict) || (matchIndex+matchLength >= dictLimit)) { + match = base + matchIndex; + if (match[matchLength] == ip[matchLength]) { + matchLength += ZSTD_count(ip+matchLength+1, match+matchLength+1, iLimit) +1; + } + } else { + match = dictBase + matchIndex; + matchLength += ZSTD_count_2segments(ip+matchLength, match+matchLength, iLimit, dictEnd, prefixStart); + if (matchIndex+matchLength >= dictLimit) + match = base + matchIndex; /* to prepare for next usage of match[matchLength] */ + } - if (matchLength > bestLength) { - if (matchLength > matchEndIdx - matchIndex) matchEndIdx = matchIndex + (U32)matchLength; - bestLength = matchLength; - matches[mnum].off = ZSTD_REP_MOVE_OPT + current - matchIndex; - matches[mnum].len = (U32)matchLength; - mnum++; - if (matchLength > ZSTD_OPT_NUM) break; - if (ip+matchLength == iLimit) /* equal : no way to know if inf or sup */ - break; /* drop, to guarantee consistency (miss a little bit of compression) */ - } + if (matchLength > bestLength) { + if (matchLength > matchEndIdx - matchIndex) matchEndIdx = matchIndex + (U32)matchLength; + bestLength = matchLength; + matches[mnum].off = ZSTD_REP_MOVE_OPT + current - matchIndex; + matches[mnum].len = (U32)matchLength; + mnum++; + if (matchLength > ZSTD_OPT_NUM) break; + if (ip+matchLength == iLimit) /* equal : no way to know if inf or sup */ + break; /* drop, to guarantee consistency (miss a little bit of compression) */ + } - if (match[matchLength] < ip[matchLength]) { - /* match is smaller than current */ - *smallerPtr = matchIndex; /* update smaller idx */ - commonLengthSmaller = matchLength; /* all smaller will now have at least this guaranteed common length */ - if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ - smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ - matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ - } else { - /* match is larger than current */ - *largerPtr = matchIndex; - commonLengthLarger = matchLength; - if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ - largerPtr = nextPtr; - matchIndex = nextPtr[0]; - } } + if (match[matchLength] < ip[matchLength]) { + /* match is smaller than current */ + *smallerPtr = matchIndex; /* update smaller idx */ + commonLengthSmaller = matchLength; /* all smaller will now have at least this guaranteed common length */ + if (matchIndex <= btLow) { smallerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + smallerPtr = nextPtr+1; /* new "smaller" => larger of match */ + matchIndex = nextPtr[1]; /* new matchIndex larger than previous (closer to current) */ + } else { + /* match is larger than current */ + *largerPtr = matchIndex; + commonLengthLarger = matchLength; + if (matchIndex <= btLow) { largerPtr=&dummy32; break; } /* beyond tree size, stop the search */ + largerPtr = nextPtr; + matchIndex = nextPtr[0]; + } } - *smallerPtr = *largerPtr = 0; + *smallerPtr = *largerPtr = 0; update: - zc->nextToUpdate = (matchEndIdx > current + 8) ? matchEndIdx - 8 : current+1; - return mnum; + zc->nextToUpdate = (matchEndIdx > current + 8) ? matchEndIdx - 8 : current+1; + return mnum; } /** Tree updater, providing best match */ static U32 ZSTD_BtGetAllMatches ( - ZSTD_CCtx* zc, - const BYTE* const ip, const BYTE* const iLimit, - const U32 maxNbAttempts, const U32 mls, ZSTD_match_t* matches, const U32 minMatchLen) + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iLimit, + const U32 maxNbAttempts, const U32 mls, ZSTD_match_t* matches, const U32 minMatchLen) { - if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ - ZSTD_updateTree(zc, ip, iLimit, maxNbAttempts, mls); - return ZSTD_insertBtAndGetAllMatches(zc, ip, iLimit, maxNbAttempts, mls, 0, matches, minMatchLen); + if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ + ZSTD_updateTree(zc, ip, iLimit, maxNbAttempts, mls); + return ZSTD_insertBtAndGetAllMatches(zc, ip, iLimit, maxNbAttempts, mls, 0, matches, minMatchLen); } static U32 ZSTD_BtGetAllMatches_selectMLS ( - ZSTD_CCtx* zc, /* Index table will be updated */ - const BYTE* ip, const BYTE* const iHighLimit, - const U32 maxNbAttempts, const U32 matchLengthSearch, ZSTD_match_t* matches, const U32 minMatchLen) + ZSTD_CCtx* zc, /* Index table will be updated */ + const BYTE* ip, const BYTE* const iHighLimit, + const U32 maxNbAttempts, const U32 matchLengthSearch, ZSTD_match_t* matches, const U32 minMatchLen) { - switch(matchLengthSearch) - { - case 3 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 3, matches, minMatchLen); - default : - case 4 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 4, matches, minMatchLen); - case 5 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 5, matches, minMatchLen); - case 7 : - case 6 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 6, matches, minMatchLen); - } + switch(matchLengthSearch) + { + case 3 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 3, matches, minMatchLen); + default : + case 4 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 4, matches, minMatchLen); + case 5 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 5, matches, minMatchLen); + case 7 : + case 6 : return ZSTD_BtGetAllMatches(zc, ip, iHighLimit, maxNbAttempts, 6, matches, minMatchLen); + } } /** Tree updater, providing best match */ static U32 ZSTD_BtGetAllMatches_extDict ( - ZSTD_CCtx* zc, - const BYTE* const ip, const BYTE* const iLimit, - const U32 maxNbAttempts, const U32 mls, ZSTD_match_t* matches, const U32 minMatchLen) + ZSTD_CCtx* zc, + const BYTE* const ip, const BYTE* const iLimit, + const U32 maxNbAttempts, const U32 mls, ZSTD_match_t* matches, const U32 minMatchLen) { - if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ - ZSTD_updateTree_extDict(zc, ip, iLimit, maxNbAttempts, mls); - return ZSTD_insertBtAndGetAllMatches(zc, ip, iLimit, maxNbAttempts, mls, 1, matches, minMatchLen); + if (ip < zc->base + zc->nextToUpdate) return 0; /* skipped area */ + ZSTD_updateTree_extDict(zc, ip, iLimit, maxNbAttempts, mls); + return ZSTD_insertBtAndGetAllMatches(zc, ip, iLimit, maxNbAttempts, mls, 1, matches, minMatchLen); } static U32 ZSTD_BtGetAllMatches_selectMLS_extDict ( - ZSTD_CCtx* zc, /* Index table will be updated */ - const BYTE* ip, const BYTE* const iHighLimit, - const U32 maxNbAttempts, const U32 matchLengthSearch, ZSTD_match_t* matches, const U32 minMatchLen) + ZSTD_CCtx* zc, /* Index table will be updated */ + const BYTE* ip, const BYTE* const iHighLimit, + const U32 maxNbAttempts, const U32 matchLengthSearch, ZSTD_match_t* matches, const U32 minMatchLen) { - switch(matchLengthSearch) - { - case 3 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 3, matches, minMatchLen); - default : - case 4 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 4, matches, minMatchLen); - case 5 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 5, matches, minMatchLen); - case 7 : - case 6 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 6, matches, minMatchLen); - } + switch(matchLengthSearch) + { + case 3 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 3, matches, minMatchLen); + default : + case 4 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 4, matches, minMatchLen); + case 5 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 5, matches, minMatchLen); + case 7 : + case 6 : return ZSTD_BtGetAllMatches_extDict(zc, ip, iHighLimit, maxNbAttempts, 6, matches, minMatchLen); + } } @@ -399,523 +399,523 @@ static U32 ZSTD_BtGetAllMatches_selectMLS_extDict ( *********************************/ FORCE_INLINE void ZSTD_compressBlock_opt_generic(ZSTD_CCtx* ctx, - const void* src, size_t srcSize, const int ultra) + const void* src, size_t srcSize, const int ultra) { - seqStore_t* seqStorePtr = &(ctx->seqStore); - const BYTE* const istart = (const BYTE*)src; - const BYTE* ip = istart; - const BYTE* anchor = istart; - const BYTE* const iend = istart + srcSize; - const BYTE* const ilimit = iend - 8; - const BYTE* const base = ctx->base; - const BYTE* const prefixStart = base + ctx->dictLimit; + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + const BYTE* const base = ctx->base; + const BYTE* const prefixStart = base + ctx->dictLimit; - const U32 maxSearches = 1U << ctx->params.cParams.searchLog; - const U32 sufficient_len = ctx->params.cParams.targetLength; - const U32 mls = ctx->params.cParams.searchLength; - const U32 minMatch = (ctx->params.cParams.searchLength == 3) ? 3 : 4; + const U32 maxSearches = 1U << ctx->params.cParams.searchLog; + const U32 sufficient_len = ctx->params.cParams.targetLength; + const U32 mls = ctx->params.cParams.searchLength; + const U32 minMatch = (ctx->params.cParams.searchLength == 3) ? 3 : 4; - ZSTD_optimal_t* opt = seqStorePtr->priceTable; - ZSTD_match_t* matches = seqStorePtr->matchTable; - const BYTE* inr; - U32 offset, rep[ZSTD_REP_NUM]; + ZSTD_optimal_t* opt = seqStorePtr->priceTable; + ZSTD_match_t* matches = seqStorePtr->matchTable; + const BYTE* inr; + U32 offset, rep[ZSTD_REP_NUM]; - /* init */ - ctx->nextToUpdate3 = ctx->nextToUpdate; - ZSTD_rescaleFreqs(seqStorePtr, (const BYTE*)src, srcSize); - ip += (ip==prefixStart); - { U32 i; for (i=0; irep[i]; } + /* init */ + ctx->nextToUpdate3 = ctx->nextToUpdate; + ZSTD_rescaleFreqs(seqStorePtr, (const BYTE*)src, srcSize); + ip += (ip==prefixStart); + { U32 i; for (i=0; irep[i]; } - /* Match Loop */ - while (ip < ilimit) { - U32 cur, match_num, last_pos, litlen, price; - U32 u, mlen, best_mlen, best_off, litLength; - memset(opt, 0, sizeof(ZSTD_optimal_t)); - last_pos = 0; - litlen = (U32)(ip - anchor); + /* Match Loop */ + while (ip < ilimit) { + U32 cur, match_num, last_pos, litlen, price; + U32 u, mlen, best_mlen, best_off, litLength; + memset(opt, 0, sizeof(ZSTD_optimal_t)); + last_pos = 0; + litlen = (U32)(ip - anchor); - /* check repCode */ - { U32 i, last_i = ZSTD_REP_CHECK + (ip==anchor); - for (i=(ip == anchor); i 0) && (repCur < (S32)(ip-prefixStart)) - && (MEM_readMINMATCH(ip, minMatch) == MEM_readMINMATCH(ip - repCur, minMatch))) { - mlen = (U32)ZSTD_count(ip+minMatch, ip+minMatch-repCur, iend) + minMatch; - if (mlen > sufficient_len || mlen >= ZSTD_OPT_NUM) { - best_mlen = mlen; best_off = i; cur = 0; last_pos = 1; - goto _storeSequence; - } - best_off = i - (ip == anchor); - do { - price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); - if (mlen > last_pos || price < opt[mlen].price) - SET_PRICE(mlen, mlen, i, litlen, price); /* note : macro modifies last_pos */ - mlen--; - } while (mlen >= minMatch); - } } } + /* check repCode */ + { U32 i, last_i = ZSTD_REP_CHECK + (ip==anchor); + for (i=(ip == anchor); i 0) && (repCur < (S32)(ip-prefixStart)) + && (MEM_readMINMATCH(ip, minMatch) == MEM_readMINMATCH(ip - repCur, minMatch))) { + mlen = (U32)ZSTD_count(ip+minMatch, ip+minMatch-repCur, iend) + minMatch; + if (mlen > sufficient_len || mlen >= ZSTD_OPT_NUM) { + best_mlen = mlen; best_off = i; cur = 0; last_pos = 1; + goto _storeSequence; + } + best_off = i - (ip == anchor); + do { + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); + if (mlen > last_pos || price < opt[mlen].price) + SET_PRICE(mlen, mlen, i, litlen, price); /* note : macro modifies last_pos */ + mlen--; + } while (mlen >= minMatch); + } } } - match_num = ZSTD_BtGetAllMatches_selectMLS(ctx, ip, iend, maxSearches, mls, matches, minMatch); + match_num = ZSTD_BtGetAllMatches_selectMLS(ctx, ip, iend, maxSearches, mls, matches, minMatch); - if (!last_pos && !match_num) { ip++; continue; } + if (!last_pos && !match_num) { ip++; continue; } - if (match_num && (matches[match_num-1].len > sufficient_len || matches[match_num-1].len >= ZSTD_OPT_NUM)) { - best_mlen = matches[match_num-1].len; - best_off = matches[match_num-1].off; - cur = 0; - last_pos = 1; - goto _storeSequence; - } + if (match_num && (matches[match_num-1].len > sufficient_len || matches[match_num-1].len >= ZSTD_OPT_NUM)) { + best_mlen = matches[match_num-1].len; + best_off = matches[match_num-1].off; + cur = 0; + last_pos = 1; + goto _storeSequence; + } - /* set prices using matches at position = 0 */ - best_mlen = (last_pos) ? last_pos : minMatch; - for (u = 0; u < match_num; u++) { - mlen = (u>0) ? matches[u-1].len+1 : best_mlen; - best_mlen = matches[u].len; - while (mlen <= best_mlen) { - price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); - if (mlen > last_pos || price < opt[mlen].price) - SET_PRICE(mlen, mlen, matches[u].off, litlen, price); /* note : macro modifies last_pos */ - mlen++; - } } + /* set prices using matches at position = 0 */ + best_mlen = (last_pos) ? last_pos : minMatch; + for (u = 0; u < match_num; u++) { + mlen = (u>0) ? matches[u-1].len+1 : best_mlen; + best_mlen = matches[u].len; + while (mlen <= best_mlen) { + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); + if (mlen > last_pos || price < opt[mlen].price) + SET_PRICE(mlen, mlen, matches[u].off, litlen, price); /* note : macro modifies last_pos */ + mlen++; + } } - if (last_pos < minMatch) { ip++; continue; } + if (last_pos < minMatch) { ip++; continue; } - /* initialize opt[0] */ - { U32 i ; for (i=0; i litlen) { - price = opt[cur - litlen].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-litlen); - } else - price = ZSTD_getLiteralPrice(seqStorePtr, litlen, anchor); - } else { - litlen = 1; - price = opt[cur - 1].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-1); - } + if (opt[cur-1].mlen == 1) { + litlen = opt[cur-1].litlen + 1; + if (cur > litlen) { + price = opt[cur - litlen].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-litlen); + } else + price = ZSTD_getLiteralPrice(seqStorePtr, litlen, anchor); + } else { + litlen = 1; + price = opt[cur - 1].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-1); + } - if (cur > last_pos || price <= opt[cur].price) - SET_PRICE(cur, 1, 0, litlen, price); + if (cur > last_pos || price <= opt[cur].price) + SET_PRICE(cur, 1, 0, litlen, price); - if (cur == last_pos) break; + if (cur == last_pos) break; - if (inr > ilimit) /* last match must start at a minimum distance of 8 from oend */ - continue; + if (inr > ilimit) /* last match must start at a minimum distance of 8 from oend */ + continue; - mlen = opt[cur].mlen; - if (opt[cur].off > ZSTD_REP_MOVE_OPT) { - opt[cur].rep[2] = opt[cur-mlen].rep[1]; - opt[cur].rep[1] = opt[cur-mlen].rep[0]; - opt[cur].rep[0] = opt[cur].off - ZSTD_REP_MOVE_OPT; - } else { - opt[cur].rep[2] = (opt[cur].off > 1) ? opt[cur-mlen].rep[1] : opt[cur-mlen].rep[2]; - opt[cur].rep[1] = (opt[cur].off > 0) ? opt[cur-mlen].rep[0] : opt[cur-mlen].rep[1]; - opt[cur].rep[0] = ((opt[cur].off==ZSTD_REP_MOVE_OPT) && (mlen != 1)) ? (opt[cur-mlen].rep[0] - 1) : (opt[cur-mlen].rep[opt[cur].off]); - } + mlen = opt[cur].mlen; + if (opt[cur].off > ZSTD_REP_MOVE_OPT) { + opt[cur].rep[2] = opt[cur-mlen].rep[1]; + opt[cur].rep[1] = opt[cur-mlen].rep[0]; + opt[cur].rep[0] = opt[cur].off - ZSTD_REP_MOVE_OPT; + } else { + opt[cur].rep[2] = (opt[cur].off > 1) ? opt[cur-mlen].rep[1] : opt[cur-mlen].rep[2]; + opt[cur].rep[1] = (opt[cur].off > 0) ? opt[cur-mlen].rep[0] : opt[cur-mlen].rep[1]; + opt[cur].rep[0] = ((opt[cur].off==ZSTD_REP_MOVE_OPT) && (mlen != 1)) ? (opt[cur-mlen].rep[0] - 1) : (opt[cur-mlen].rep[opt[cur].off]); + } - best_mlen = minMatch; - { U32 i, last_i = ZSTD_REP_CHECK + (mlen != 1); - for (i=(opt[cur].mlen != 1); i 0) && (repCur < (S32)(inr-prefixStart)) - && (MEM_readMINMATCH(inr, minMatch) == MEM_readMINMATCH(inr - repCur, minMatch))) { - mlen = (U32)ZSTD_count(inr+minMatch, inr+minMatch - repCur, iend) + minMatch; + best_mlen = minMatch; + { U32 i, last_i = ZSTD_REP_CHECK + (mlen != 1); + for (i=(opt[cur].mlen != 1); i 0) && (repCur < (S32)(inr-prefixStart)) + && (MEM_readMINMATCH(inr, minMatch) == MEM_readMINMATCH(inr - repCur, minMatch))) { + mlen = (U32)ZSTD_count(inr+minMatch, inr+minMatch - repCur, iend) + minMatch; - if (mlen > sufficient_len || cur + mlen >= ZSTD_OPT_NUM) { - best_mlen = mlen; best_off = i; last_pos = cur + 1; - goto _storeSequence; - } + if (mlen > sufficient_len || cur + mlen >= ZSTD_OPT_NUM) { + best_mlen = mlen; best_off = i; last_pos = cur + 1; + goto _storeSequence; + } - best_off = i - (opt[cur].mlen != 1); - if (mlen > best_mlen) best_mlen = mlen; + best_off = i - (opt[cur].mlen != 1); + if (mlen > best_mlen) best_mlen = mlen; - do { - if (opt[cur].mlen == 1) { - litlen = opt[cur].litlen; - if (cur > litlen) { - price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, inr-litlen, best_off, mlen - MINMATCH, ultra); - } else - price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); - } else { - litlen = 0; - price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, best_off, mlen - MINMATCH, ultra); - } + do { + if (opt[cur].mlen == 1) { + litlen = opt[cur].litlen; + if (cur > litlen) { + price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, inr-litlen, best_off, mlen - MINMATCH, ultra); + } else + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); + } else { + litlen = 0; + price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, best_off, mlen - MINMATCH, ultra); + } - if (cur + mlen > last_pos || price <= opt[cur + mlen].price) - SET_PRICE(cur + mlen, mlen, i, litlen, price); - mlen--; - } while (mlen >= minMatch); - } } } + if (cur + mlen > last_pos || price <= opt[cur + mlen].price) + SET_PRICE(cur + mlen, mlen, i, litlen, price); + mlen--; + } while (mlen >= minMatch); + } } } - match_num = ZSTD_BtGetAllMatches_selectMLS(ctx, inr, iend, maxSearches, mls, matches, best_mlen); + match_num = ZSTD_BtGetAllMatches_selectMLS(ctx, inr, iend, maxSearches, mls, matches, best_mlen); - if (match_num > 0 && (matches[match_num-1].len > sufficient_len || cur + matches[match_num-1].len >= ZSTD_OPT_NUM)) { - best_mlen = matches[match_num-1].len; - best_off = matches[match_num-1].off; - last_pos = cur + 1; - goto _storeSequence; - } + if (match_num > 0 && (matches[match_num-1].len > sufficient_len || cur + matches[match_num-1].len >= ZSTD_OPT_NUM)) { + best_mlen = matches[match_num-1].len; + best_off = matches[match_num-1].off; + last_pos = cur + 1; + goto _storeSequence; + } - /* set prices using matches at position = cur */ - for (u = 0; u < match_num; u++) { - mlen = (u>0) ? matches[u-1].len+1 : best_mlen; - best_mlen = matches[u].len; + /* set prices using matches at position = cur */ + for (u = 0; u < match_num; u++) { + mlen = (u>0) ? matches[u-1].len+1 : best_mlen; + best_mlen = matches[u].len; - while (mlen <= best_mlen) { - if (opt[cur].mlen == 1) { - litlen = opt[cur].litlen; - if (cur > litlen) - price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, ip+cur-litlen, matches[u].off-1, mlen - MINMATCH, ultra); - else - price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); - } else { - litlen = 0; - price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, matches[u].off-1, mlen - MINMATCH, ultra); - } + while (mlen <= best_mlen) { + if (opt[cur].mlen == 1) { + litlen = opt[cur].litlen; + if (cur > litlen) + price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, ip+cur-litlen, matches[u].off-1, mlen - MINMATCH, ultra); + else + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); + } else { + litlen = 0; + price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, matches[u].off-1, mlen - MINMATCH, ultra); + } - if (cur + mlen > last_pos || (price < opt[cur + mlen].price)) - SET_PRICE(cur + mlen, mlen, matches[u].off, litlen, price); + if (cur + mlen > last_pos || (price < opt[cur + mlen].price)) + SET_PRICE(cur + mlen, mlen, matches[u].off, litlen, price); - mlen++; - } } } + mlen++; + } } } - best_mlen = opt[last_pos].mlen; - best_off = opt[last_pos].off; - cur = last_pos - best_mlen; + best_mlen = opt[last_pos].mlen; + best_off = opt[last_pos].off; + cur = last_pos - best_mlen; - /* store sequence */ + /* store sequence */ _storeSequence: /* cur, last_pos, best_mlen, best_off have to be set */ - opt[0].mlen = 1; + opt[0].mlen = 1; - while (1) { - mlen = opt[cur].mlen; - offset = opt[cur].off; - opt[cur].mlen = best_mlen; - opt[cur].off = best_off; - best_mlen = mlen; - best_off = offset; - if (mlen > cur) break; - cur -= mlen; - } + while (1) { + mlen = opt[cur].mlen; + offset = opt[cur].off; + opt[cur].mlen = best_mlen; + opt[cur].off = best_off; + best_mlen = mlen; + best_off = offset; + if (mlen > cur) break; + cur -= mlen; + } - for (u = 0; u <= last_pos;) { - u += opt[u].mlen; - } + for (u = 0; u <= last_pos;) { + u += opt[u].mlen; + } - for (cur=0; cur < last_pos; ) { - mlen = opt[cur].mlen; - if (mlen == 1) { ip++; cur++; continue; } - offset = opt[cur].off; - cur += mlen; - litLength = (U32)(ip - anchor); + for (cur=0; cur < last_pos; ) { + mlen = opt[cur].mlen; + if (mlen == 1) { ip++; cur++; continue; } + offset = opt[cur].off; + cur += mlen; + litLength = (U32)(ip - anchor); - if (offset > ZSTD_REP_MOVE_OPT) { - rep[2] = rep[1]; - rep[1] = rep[0]; - rep[0] = offset - ZSTD_REP_MOVE_OPT; - offset--; - } else { - if (offset != 0) { - best_off = (offset==ZSTD_REP_MOVE_OPT) ? (rep[0] - 1) : (rep[offset]); - if (offset != 1) rep[2] = rep[1]; - rep[1] = rep[0]; - rep[0] = best_off; - } - if (litLength==0) offset--; - } + if (offset > ZSTD_REP_MOVE_OPT) { + rep[2] = rep[1]; + rep[1] = rep[0]; + rep[0] = offset - ZSTD_REP_MOVE_OPT; + offset--; + } else { + if (offset != 0) { + best_off = (offset==ZSTD_REP_MOVE_OPT) ? (rep[0] - 1) : (rep[offset]); + if (offset != 1) rep[2] = rep[1]; + rep[1] = rep[0]; + rep[0] = best_off; + } + if (litLength==0) offset--; + } - ZSTD_updatePrice(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); - ZSTD_storeSeq(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); - anchor = ip = ip + mlen; - } } /* for (cur=0; cur < last_pos; ) */ + ZSTD_updatePrice(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); + ZSTD_storeSeq(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); + anchor = ip = ip + mlen; + } } /* for (cur=0; cur < last_pos; ) */ - /* Save reps for next block */ - { int i; for (i=0; irepToConfirm[i] = rep[i]; } + /* Save reps for next block */ + { int i; for (i=0; irepToConfirm[i] = rep[i]; } - /* Last Literals */ - { size_t const lastLLSize = iend - anchor; - memcpy(seqStorePtr->lit, anchor, lastLLSize); - seqStorePtr->lit += lastLLSize; - } + /* Last Literals */ + { size_t const lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } } FORCE_INLINE void ZSTD_compressBlock_opt_extDict_generic(ZSTD_CCtx* ctx, - const void* src, size_t srcSize, const int ultra) + const void* src, size_t srcSize, const int ultra) { - seqStore_t* seqStorePtr = &(ctx->seqStore); - const BYTE* const istart = (const BYTE*)src; - const BYTE* ip = istart; - const BYTE* anchor = istart; - const BYTE* const iend = istart + srcSize; - const BYTE* const ilimit = iend - 8; - const BYTE* const base = ctx->base; - const U32 lowestIndex = ctx->lowLimit; - const U32 dictLimit = ctx->dictLimit; - const BYTE* const prefixStart = base + dictLimit; - const BYTE* const dictBase = ctx->dictBase; - const BYTE* const dictEnd = dictBase + dictLimit; + seqStore_t* seqStorePtr = &(ctx->seqStore); + const BYTE* const istart = (const BYTE*)src; + const BYTE* ip = istart; + const BYTE* anchor = istart; + const BYTE* const iend = istart + srcSize; + const BYTE* const ilimit = iend - 8; + const BYTE* const base = ctx->base; + const U32 lowestIndex = ctx->lowLimit; + const U32 dictLimit = ctx->dictLimit; + const BYTE* const prefixStart = base + dictLimit; + const BYTE* const dictBase = ctx->dictBase; + const BYTE* const dictEnd = dictBase + dictLimit; - const U32 maxSearches = 1U << ctx->params.cParams.searchLog; - const U32 sufficient_len = ctx->params.cParams.targetLength; - const U32 mls = ctx->params.cParams.searchLength; - const U32 minMatch = (ctx->params.cParams.searchLength == 3) ? 3 : 4; + const U32 maxSearches = 1U << ctx->params.cParams.searchLog; + const U32 sufficient_len = ctx->params.cParams.targetLength; + const U32 mls = ctx->params.cParams.searchLength; + const U32 minMatch = (ctx->params.cParams.searchLength == 3) ? 3 : 4; - ZSTD_optimal_t* opt = seqStorePtr->priceTable; - ZSTD_match_t* matches = seqStorePtr->matchTable; - const BYTE* inr; + ZSTD_optimal_t* opt = seqStorePtr->priceTable; + ZSTD_match_t* matches = seqStorePtr->matchTable; + const BYTE* inr; - /* init */ - U32 offset, rep[ZSTD_REP_NUM]; - { U32 i; for (i=0; irep[i]; } + /* init */ + U32 offset, rep[ZSTD_REP_NUM]; + { U32 i; for (i=0; irep[i]; } - ctx->nextToUpdate3 = ctx->nextToUpdate; - ZSTD_rescaleFreqs(seqStorePtr, (const BYTE*)src, srcSize); - ip += (ip==prefixStart); + ctx->nextToUpdate3 = ctx->nextToUpdate; + ZSTD_rescaleFreqs(seqStorePtr, (const BYTE*)src, srcSize); + ip += (ip==prefixStart); - /* Match Loop */ - while (ip < ilimit) { - U32 cur, match_num, last_pos, litlen, price; - U32 u, mlen, best_mlen, best_off, litLength; - U32 current = (U32)(ip-base); - memset(opt, 0, sizeof(ZSTD_optimal_t)); - last_pos = 0; - opt[0].litlen = (U32)(ip - anchor); + /* Match Loop */ + while (ip < ilimit) { + U32 cur, match_num, last_pos, litlen, price; + U32 u, mlen, best_mlen, best_off, litLength; + U32 current = (U32)(ip-base); + memset(opt, 0, sizeof(ZSTD_optimal_t)); + last_pos = 0; + opt[0].litlen = (U32)(ip - anchor); - /* check repCode */ - { U32 i, last_i = ZSTD_REP_CHECK + (ip==anchor); - for (i = (ip==anchor); i 0 && repCur <= (S32)current) - && (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex>lowestIndex)) /* intentional overflow */ - && (MEM_readMINMATCH(ip, minMatch) == MEM_readMINMATCH(repMatch, minMatch)) ) { - /* repcode detected we should take it */ - const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; - mlen = (U32)ZSTD_count_2segments(ip+minMatch, repMatch+minMatch, iend, repEnd, prefixStart) + minMatch; + /* check repCode */ + { U32 i, last_i = ZSTD_REP_CHECK + (ip==anchor); + for (i = (ip==anchor); i 0 && repCur <= (S32)current) + && (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex>lowestIndex)) /* intentional overflow */ + && (MEM_readMINMATCH(ip, minMatch) == MEM_readMINMATCH(repMatch, minMatch)) ) { + /* repcode detected we should take it */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + mlen = (U32)ZSTD_count_2segments(ip+minMatch, repMatch+minMatch, iend, repEnd, prefixStart) + minMatch; - if (mlen > sufficient_len || mlen >= ZSTD_OPT_NUM) { - best_mlen = mlen; best_off = i; cur = 0; last_pos = 1; - goto _storeSequence; - } + if (mlen > sufficient_len || mlen >= ZSTD_OPT_NUM) { + best_mlen = mlen; best_off = i; cur = 0; last_pos = 1; + goto _storeSequence; + } - best_off = i - (ip==anchor); - litlen = opt[0].litlen; - do { - price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); - if (mlen > last_pos || price < opt[mlen].price) - SET_PRICE(mlen, mlen, i, litlen, price); /* note : macro modifies last_pos */ - mlen--; - } while (mlen >= minMatch); - } } } + best_off = i - (ip==anchor); + litlen = opt[0].litlen; + do { + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); + if (mlen > last_pos || price < opt[mlen].price) + SET_PRICE(mlen, mlen, i, litlen, price); /* note : macro modifies last_pos */ + mlen--; + } while (mlen >= minMatch); + } } } - match_num = ZSTD_BtGetAllMatches_selectMLS_extDict(ctx, ip, iend, maxSearches, mls, matches, minMatch); /* first search (depth 0) */ + match_num = ZSTD_BtGetAllMatches_selectMLS_extDict(ctx, ip, iend, maxSearches, mls, matches, minMatch); /* first search (depth 0) */ - if (!last_pos && !match_num) { ip++; continue; } + if (!last_pos && !match_num) { ip++; continue; } - { U32 i; for (i=0; i sufficient_len || matches[match_num-1].len >= ZSTD_OPT_NUM)) { - best_mlen = matches[match_num-1].len; - best_off = matches[match_num-1].off; - cur = 0; - last_pos = 1; - goto _storeSequence; - } + if (match_num && (matches[match_num-1].len > sufficient_len || matches[match_num-1].len >= ZSTD_OPT_NUM)) { + best_mlen = matches[match_num-1].len; + best_off = matches[match_num-1].off; + cur = 0; + last_pos = 1; + goto _storeSequence; + } - best_mlen = (last_pos) ? last_pos : minMatch; + best_mlen = (last_pos) ? last_pos : minMatch; - /* set prices using matches at position = 0 */ - for (u = 0; u < match_num; u++) { - mlen = (u>0) ? matches[u-1].len+1 : best_mlen; - best_mlen = matches[u].len; - litlen = opt[0].litlen; - while (mlen <= best_mlen) { - price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); - if (mlen > last_pos || price < opt[mlen].price) - SET_PRICE(mlen, mlen, matches[u].off, litlen, price); - mlen++; - } } + /* set prices using matches at position = 0 */ + for (u = 0; u < match_num; u++) { + mlen = (u>0) ? matches[u-1].len+1 : best_mlen; + best_mlen = matches[u].len; + litlen = opt[0].litlen; + while (mlen <= best_mlen) { + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); + if (mlen > last_pos || price < opt[mlen].price) + SET_PRICE(mlen, mlen, matches[u].off, litlen, price); + mlen++; + } } - if (last_pos < minMatch) { - ip++; continue; - } + if (last_pos < minMatch) { + ip++; continue; + } - /* check further positions */ - for (cur = 1; cur <= last_pos; cur++) { - inr = ip + cur; + /* check further positions */ + for (cur = 1; cur <= last_pos; cur++) { + inr = ip + cur; - if (opt[cur-1].mlen == 1) { - litlen = opt[cur-1].litlen + 1; - if (cur > litlen) { - price = opt[cur - litlen].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-litlen); - } else - price = ZSTD_getLiteralPrice(seqStorePtr, litlen, anchor); - } else { - litlen = 1; - price = opt[cur - 1].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-1); - } + if (opt[cur-1].mlen == 1) { + litlen = opt[cur-1].litlen + 1; + if (cur > litlen) { + price = opt[cur - litlen].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-litlen); + } else + price = ZSTD_getLiteralPrice(seqStorePtr, litlen, anchor); + } else { + litlen = 1; + price = opt[cur - 1].price + ZSTD_getLiteralPrice(seqStorePtr, litlen, inr-1); + } - if (cur > last_pos || price <= opt[cur].price) - SET_PRICE(cur, 1, 0, litlen, price); + if (cur > last_pos || price <= opt[cur].price) + SET_PRICE(cur, 1, 0, litlen, price); - if (cur == last_pos) break; + if (cur == last_pos) break; - if (inr > ilimit) /* last match must start at a minimum distance of 8 from oend */ - continue; + if (inr > ilimit) /* last match must start at a minimum distance of 8 from oend */ + continue; - mlen = opt[cur].mlen; - if (opt[cur].off > ZSTD_REP_MOVE_OPT) { - opt[cur].rep[2] = opt[cur-mlen].rep[1]; - opt[cur].rep[1] = opt[cur-mlen].rep[0]; - opt[cur].rep[0] = opt[cur].off - ZSTD_REP_MOVE_OPT; - } else { - opt[cur].rep[2] = (opt[cur].off > 1) ? opt[cur-mlen].rep[1] : opt[cur-mlen].rep[2]; - opt[cur].rep[1] = (opt[cur].off > 0) ? opt[cur-mlen].rep[0] : opt[cur-mlen].rep[1]; - opt[cur].rep[0] = ((opt[cur].off==ZSTD_REP_MOVE_OPT) && (mlen != 1)) ? (opt[cur-mlen].rep[0] - 1) : (opt[cur-mlen].rep[opt[cur].off]); - } + mlen = opt[cur].mlen; + if (opt[cur].off > ZSTD_REP_MOVE_OPT) { + opt[cur].rep[2] = opt[cur-mlen].rep[1]; + opt[cur].rep[1] = opt[cur-mlen].rep[0]; + opt[cur].rep[0] = opt[cur].off - ZSTD_REP_MOVE_OPT; + } else { + opt[cur].rep[2] = (opt[cur].off > 1) ? opt[cur-mlen].rep[1] : opt[cur-mlen].rep[2]; + opt[cur].rep[1] = (opt[cur].off > 0) ? opt[cur-mlen].rep[0] : opt[cur-mlen].rep[1]; + opt[cur].rep[0] = ((opt[cur].off==ZSTD_REP_MOVE_OPT) && (mlen != 1)) ? (opt[cur-mlen].rep[0] - 1) : (opt[cur-mlen].rep[opt[cur].off]); + } - best_mlen = minMatch; - { U32 i, last_i = ZSTD_REP_CHECK + (mlen != 1); - for (i = (mlen != 1); i 0 && repCur <= (S32)(current+cur)) - && (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex>lowestIndex)) /* intentional overflow */ - && (MEM_readMINMATCH(inr, minMatch) == MEM_readMINMATCH(repMatch, minMatch)) ) { - /* repcode detected */ - const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; - mlen = (U32)ZSTD_count_2segments(inr+minMatch, repMatch+minMatch, iend, repEnd, prefixStart) + minMatch; + best_mlen = minMatch; + { U32 i, last_i = ZSTD_REP_CHECK + (mlen != 1); + for (i = (mlen != 1); i 0 && repCur <= (S32)(current+cur)) + && (((U32)((dictLimit-1) - repIndex) >= 3) & (repIndex>lowestIndex)) /* intentional overflow */ + && (MEM_readMINMATCH(inr, minMatch) == MEM_readMINMATCH(repMatch, minMatch)) ) { + /* repcode detected */ + const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend; + mlen = (U32)ZSTD_count_2segments(inr+minMatch, repMatch+minMatch, iend, repEnd, prefixStart) + minMatch; - if (mlen > sufficient_len || cur + mlen >= ZSTD_OPT_NUM) { - best_mlen = mlen; best_off = i; last_pos = cur + 1; - goto _storeSequence; - } + if (mlen > sufficient_len || cur + mlen >= ZSTD_OPT_NUM) { + best_mlen = mlen; best_off = i; last_pos = cur + 1; + goto _storeSequence; + } - best_off = i - (opt[cur].mlen != 1); - if (mlen > best_mlen) best_mlen = mlen; + best_off = i - (opt[cur].mlen != 1); + if (mlen > best_mlen) best_mlen = mlen; - do { - if (opt[cur].mlen == 1) { - litlen = opt[cur].litlen; - if (cur > litlen) { - price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, inr-litlen, best_off, mlen - MINMATCH, ultra); - } else - price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); - } else { - litlen = 0; - price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, best_off, mlen - MINMATCH, ultra); - } + do { + if (opt[cur].mlen == 1) { + litlen = opt[cur].litlen; + if (cur > litlen) { + price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, inr-litlen, best_off, mlen - MINMATCH, ultra); + } else + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, best_off, mlen - MINMATCH, ultra); + } else { + litlen = 0; + price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, best_off, mlen - MINMATCH, ultra); + } - if (cur + mlen > last_pos || price <= opt[cur + mlen].price) - SET_PRICE(cur + mlen, mlen, i, litlen, price); - mlen--; - } while (mlen >= minMatch); - } } } + if (cur + mlen > last_pos || price <= opt[cur + mlen].price) + SET_PRICE(cur + mlen, mlen, i, litlen, price); + mlen--; + } while (mlen >= minMatch); + } } } - match_num = ZSTD_BtGetAllMatches_selectMLS_extDict(ctx, inr, iend, maxSearches, mls, matches, minMatch); + match_num = ZSTD_BtGetAllMatches_selectMLS_extDict(ctx, inr, iend, maxSearches, mls, matches, minMatch); - if (match_num > 0 && (matches[match_num-1].len > sufficient_len || cur + matches[match_num-1].len >= ZSTD_OPT_NUM)) { - best_mlen = matches[match_num-1].len; - best_off = matches[match_num-1].off; - last_pos = cur + 1; - goto _storeSequence; - } + if (match_num > 0 && (matches[match_num-1].len > sufficient_len || cur + matches[match_num-1].len >= ZSTD_OPT_NUM)) { + best_mlen = matches[match_num-1].len; + best_off = matches[match_num-1].off; + last_pos = cur + 1; + goto _storeSequence; + } - /* set prices using matches at position = cur */ - for (u = 0; u < match_num; u++) { - mlen = (u>0) ? matches[u-1].len+1 : best_mlen; - best_mlen = matches[u].len; + /* set prices using matches at position = cur */ + for (u = 0; u < match_num; u++) { + mlen = (u>0) ? matches[u-1].len+1 : best_mlen; + best_mlen = matches[u].len; - while (mlen <= best_mlen) { - if (opt[cur].mlen == 1) { - litlen = opt[cur].litlen; - if (cur > litlen) - price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, ip+cur-litlen, matches[u].off-1, mlen - MINMATCH, ultra); - else - price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); - } else { - litlen = 0; - price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, matches[u].off-1, mlen - MINMATCH, ultra); - } + while (mlen <= best_mlen) { + if (opt[cur].mlen == 1) { + litlen = opt[cur].litlen; + if (cur > litlen) + price = opt[cur - litlen].price + ZSTD_getPrice(seqStorePtr, litlen, ip+cur-litlen, matches[u].off-1, mlen - MINMATCH, ultra); + else + price = ZSTD_getPrice(seqStorePtr, litlen, anchor, matches[u].off-1, mlen - MINMATCH, ultra); + } else { + litlen = 0; + price = opt[cur].price + ZSTD_getPrice(seqStorePtr, 0, NULL, matches[u].off-1, mlen - MINMATCH, ultra); + } - if (cur + mlen > last_pos || (price < opt[cur + mlen].price)) - SET_PRICE(cur + mlen, mlen, matches[u].off, litlen, price); + if (cur + mlen > last_pos || (price < opt[cur + mlen].price)) + SET_PRICE(cur + mlen, mlen, matches[u].off, litlen, price); - mlen++; - } } } /* for (cur = 1; cur <= last_pos; cur++) */ + mlen++; + } } } /* for (cur = 1; cur <= last_pos; cur++) */ - best_mlen = opt[last_pos].mlen; - best_off = opt[last_pos].off; - cur = last_pos - best_mlen; + best_mlen = opt[last_pos].mlen; + best_off = opt[last_pos].off; + cur = last_pos - best_mlen; - /* store sequence */ + /* store sequence */ _storeSequence: /* cur, last_pos, best_mlen, best_off have to be set */ - opt[0].mlen = 1; + opt[0].mlen = 1; - while (1) { - mlen = opt[cur].mlen; - offset = opt[cur].off; - opt[cur].mlen = best_mlen; - opt[cur].off = best_off; - best_mlen = mlen; - best_off = offset; - if (mlen > cur) break; - cur -= mlen; - } + while (1) { + mlen = opt[cur].mlen; + offset = opt[cur].off; + opt[cur].mlen = best_mlen; + opt[cur].off = best_off; + best_mlen = mlen; + best_off = offset; + if (mlen > cur) break; + cur -= mlen; + } - for (u = 0; u <= last_pos; ) { - u += opt[u].mlen; - } + for (u = 0; u <= last_pos; ) { + u += opt[u].mlen; + } - for (cur=0; cur < last_pos; ) { - mlen = opt[cur].mlen; - if (mlen == 1) { ip++; cur++; continue; } - offset = opt[cur].off; - cur += mlen; - litLength = (U32)(ip - anchor); + for (cur=0; cur < last_pos; ) { + mlen = opt[cur].mlen; + if (mlen == 1) { ip++; cur++; continue; } + offset = opt[cur].off; + cur += mlen; + litLength = (U32)(ip - anchor); - if (offset > ZSTD_REP_MOVE_OPT) { - rep[2] = rep[1]; - rep[1] = rep[0]; - rep[0] = offset - ZSTD_REP_MOVE_OPT; - offset--; - } else { - if (offset != 0) { - best_off = (offset==ZSTD_REP_MOVE_OPT) ? (rep[0] - 1) : (rep[offset]); - if (offset != 1) rep[2] = rep[1]; - rep[1] = rep[0]; - rep[0] = best_off; - } + if (offset > ZSTD_REP_MOVE_OPT) { + rep[2] = rep[1]; + rep[1] = rep[0]; + rep[0] = offset - ZSTD_REP_MOVE_OPT; + offset--; + } else { + if (offset != 0) { + best_off = (offset==ZSTD_REP_MOVE_OPT) ? (rep[0] - 1) : (rep[offset]); + if (offset != 1) rep[2] = rep[1]; + rep[1] = rep[0]; + rep[0] = best_off; + } - if (litLength==0) offset--; - } + if (litLength==0) offset--; + } - ZSTD_updatePrice(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); - ZSTD_storeSeq(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); - anchor = ip = ip + mlen; - } } /* for (cur=0; cur < last_pos; ) */ + ZSTD_updatePrice(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); + ZSTD_storeSeq(seqStorePtr, litLength, anchor, offset, mlen-MINMATCH); + anchor = ip = ip + mlen; + } } /* for (cur=0; cur < last_pos; ) */ - /* Save reps for next block */ - { int i; for (i=0; irepToConfirm[i] = rep[i]; } + /* Save reps for next block */ + { int i; for (i=0; irepToConfirm[i] = rep[i]; } - /* Last Literals */ - { size_t lastLLSize = iend - anchor; - memcpy(seqStorePtr->lit, anchor, lastLLSize); - seqStorePtr->lit += lastLLSize; - } + /* Last Literals */ + { size_t lastLLSize = iend - anchor; + memcpy(seqStorePtr->lit, anchor, lastLLSize); + seqStorePtr->lit += lastLLSize; + } } #endif /* ZSTD_OPT_H_91842398743 */ From b3b41d0f6dc4cf79334d9c72fffc784da0575b3b Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Wed, 29 Mar 2017 20:26:07 -0700 Subject: [PATCH 097/305] Remove platform specific macros --- contrib/linux-kernel/include/zstd.h | 21 +--- contrib/linux-kernel/lib/bitstream.h | 26 ---- contrib/linux-kernel/lib/error_private.h | 22 +--- contrib/linux-kernel/lib/fse.h | 18 +-- contrib/linux-kernel/lib/fse_compress.c | 17 +-- contrib/linux-kernel/lib/fse_decompress.c | 17 +-- contrib/linux-kernel/lib/huf.h | 9 -- contrib/linux-kernel/lib/huf_compress.c | 7 -- contrib/linux-kernel/lib/huf_decompress.c | 17 +-- contrib/linux-kernel/lib/mem.h | 125 ++----------------- contrib/linux-kernel/lib/xxhash.c | 137 ++------------------- contrib/linux-kernel/lib/xxhash.h | 58 +-------- contrib/linux-kernel/lib/zstd_decompress.c | 93 +------------- contrib/linux-kernel/lib/zstd_errors.h | 22 +--- contrib/linux-kernel/lib/zstd_internal.h | 29 +---- 15 files changed, 35 insertions(+), 583 deletions(-) diff --git a/contrib/linux-kernel/include/zstd.h b/contrib/linux-kernel/include/zstd.h index d7f4f5a09..ff7f86a6d 100644 --- a/contrib/linux-kernel/include/zstd.h +++ b/contrib/linux-kernel/include/zstd.h @@ -7,10 +7,6 @@ * of patent rights can be found in the PATENTS file in the same directory. */ -#if defined (__cplusplus) -extern "C" { -#endif - #ifndef ZSTD_H_235446 #define ZSTD_H_235446 @@ -19,18 +15,7 @@ extern "C" { /* ===== ZSTDLIB_API : control library symbols visibility ===== */ -#if defined(__GNUC__) && (__GNUC__ >= 4) -# define ZSTDLIB_VISIBILITY __attribute__ ((visibility ("default"))) -#else -# define ZSTDLIB_VISIBILITY -#endif -#if defined(ZSTD_DLL_EXPORT) && (ZSTD_DLL_EXPORT==1) -# define ZSTDLIB_API __declspec(dllexport) ZSTDLIB_VISIBILITY -#elif defined(ZSTD_DLL_IMPORT) && (ZSTD_DLL_IMPORT==1) -# define ZSTDLIB_API __declspec(dllimport) ZSTDLIB_VISIBILITY /* It isn't required but allows to generate better code, saving a function pointer load from the IAT and an indirect jump.*/ -#else -# define ZSTDLIB_API ZSTDLIB_VISIBILITY -#endif +#define ZSTDLIB_API /******************************************************************************************************* @@ -769,7 +754,3 @@ ZSTDLIB_API size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, siz #endif /* ZSTD_H_ZSTD_STATIC_LINKING_ONLY */ - -#if defined (__cplusplus) -} -#endif diff --git a/contrib/linux-kernel/lib/bitstream.h b/contrib/linux-kernel/lib/bitstream.h index 546f6582b..9d2154082 100644 --- a/contrib/linux-kernel/lib/bitstream.h +++ b/contrib/linux-kernel/lib/bitstream.h @@ -35,11 +35,6 @@ #ifndef BITSTREAM_H_MODULE #define BITSTREAM_H_MODULE -#if defined (__cplusplus) -extern "C" { -#endif - - /* * This API consists of small unitary functions, which must be inlined for best performance. * Since link-time-optimization is not available for all compilers, @@ -56,10 +51,6 @@ extern "C" { /*========================================= * Target specific =========================================*/ -#if defined(__BMI__) && defined(__GNUC__) -# include /* support for bextr (experimental) */ -#endif - #define STREAM_ACCUMULATOR_MIN_32 25 #define STREAM_ACCUMULATOR_MIN_64 57 #define STREAM_ACCUMULATOR_MIN ((U32)(MEM_32bits() ? STREAM_ACCUMULATOR_MIN_32 : STREAM_ACCUMULATOR_MIN_64)) @@ -301,16 +292,7 @@ MEM_STATIC size_t BIT_getUpperBits(size_t bitContainer, U32 const start) MEM_STATIC size_t BIT_getMiddleBits(size_t bitContainer, U32 const start, U32 const nbBits) { -#if defined(__BMI__) && defined(__GNUC__) && __GNUC__*1000+__GNUC_MINOR__ >= 4008 /* experimental */ -# if defined(__x86_64__) - if (sizeof(bitContainer)==8) - return _bextr_u64(bitContainer, start, nbBits); - else -# endif - return _bextr_u32(bitContainer, start, nbBits); -#else return (bitContainer >> start) & BIT_mask[nbBits]; -#endif } MEM_STATIC size_t BIT_getLowerBits(size_t bitContainer, U32 const nbBits) @@ -327,12 +309,8 @@ MEM_STATIC size_t BIT_getLowerBits(size_t bitContainer, U32 const nbBits) */ MEM_STATIC size_t BIT_lookBits(const BIT_DStream_t* bitD, U32 nbBits) { -#if defined(__BMI__) && defined(__GNUC__) /* experimental; fails if bitD->bitsConsumed + nbBits > sizeof(bitD->bitContainer)*8 */ - return BIT_getMiddleBits(bitD->bitContainer, (sizeof(bitD->bitContainer)*8) - bitD->bitsConsumed - nbBits, nbBits); -#else U32 const bitMask = sizeof(bitD->bitContainer)*8 - 1; return ((bitD->bitContainer << (bitD->bitsConsumed & bitMask)) >> 1) >> ((bitMask-nbBits) & bitMask); -#endif } /*! BIT_lookBitsFast() : @@ -410,8 +388,4 @@ MEM_STATIC unsigned BIT_endOfDStream(const BIT_DStream_t* DStream) return ((DStream->ptr == DStream->start) && (DStream->bitsConsumed == sizeof(DStream->bitContainer)*8)); } -#if defined (__cplusplus) -} -#endif - #endif /* BITSTREAM_H_MODULE */ diff --git a/contrib/linux-kernel/lib/error_private.h b/contrib/linux-kernel/lib/error_private.h index dc160914b..e88f2fd17 100644 --- a/contrib/linux-kernel/lib/error_private.h +++ b/contrib/linux-kernel/lib/error_private.h @@ -12,11 +12,6 @@ #ifndef ERROR_H_MODULE #define ERROR_H_MODULE -#if defined (__cplusplus) -extern "C" { -#endif - - /* **************************************** * Dependencies ******************************************/ @@ -27,15 +22,7 @@ extern "C" { /* **************************************** * Compiler-specific ******************************************/ -#if defined(__GNUC__) -# define ERR_STATIC static __attribute__((unused)) -#elif defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) -# define ERR_STATIC static inline -#elif defined(_MSC_VER) -# define ERR_STATIC static __inline -#else -# define ERR_STATIC static /* this version may generate warnings for unused static functions; disable the relevant warning */ -#endif +#define ERR_STATIC static __attribute__((unused)) /*-**************************************** @@ -48,9 +35,6 @@ typedef ZSTD_ErrorCode ERR_enum; /*-**************************************** * Error codes handling ******************************************/ -#ifdef ERROR -# undef ERROR /* reported already defined on VS 2015 (Rich Geldreich) */ -#endif #define ERROR(name) ((size_t)-PREFIX(name)) ERR_STATIC unsigned ERR_isError(size_t code) { return (code > ERROR(maxCode)); } @@ -69,8 +53,4 @@ ERR_STATIC const char* ERR_getErrorName(size_t code) return ERR_getErrorString(ERR_getErrorCode(code)); } -#if defined (__cplusplus) -} -#endif - #endif /* ERROR_H_MODULE */ diff --git a/contrib/linux-kernel/lib/fse.h b/contrib/linux-kernel/lib/fse.h index bcb592e96..538d52427 100644 --- a/contrib/linux-kernel/lib/fse.h +++ b/contrib/linux-kernel/lib/fse.h @@ -34,10 +34,6 @@ #ifndef FSE_H #define FSE_H -#if defined (__cplusplus) -extern "C" { -#endif - /*-***************************************** * Dependencies @@ -48,15 +44,7 @@ extern "C" { /*-***************************************** * FSE_PUBLIC_API : control library symbols visibility ******************************************/ -#if defined(FSE_DLL_EXPORT) && (FSE_DLL_EXPORT==1) && defined(__GNUC__) && (__GNUC__ >= 4) -# define FSE_PUBLIC_API __attribute__ ((visibility ("default"))) -#elif defined(FSE_DLL_EXPORT) && (FSE_DLL_EXPORT==1) /* Visual expected */ -# define FSE_PUBLIC_API __declspec(dllexport) -#elif defined(FSE_DLL_IMPORT) && (FSE_DLL_IMPORT==1) -# define FSE_PUBLIC_API __declspec(dllimport) /* It isn't required but allows to generate better code, saving a function pointer load from the IAT and an indirect jump.*/ -#else -# define FSE_PUBLIC_API -#endif +#define FSE_PUBLIC_API /*------ Version ------*/ #define FSE_VERSION_MAJOR 0 @@ -687,8 +675,4 @@ MEM_STATIC unsigned FSE_endOfDState(const FSE_DState_t* DStatePtr) #endif /* FSE_STATIC_LINKING_ONLY */ -#if defined (__cplusplus) -} -#endif - #endif /* FSE_H */ diff --git a/contrib/linux-kernel/lib/fse_compress.c b/contrib/linux-kernel/lib/fse_compress.c index 7340d413a..dbfa510c6 100644 --- a/contrib/linux-kernel/lib/fse_compress.c +++ b/contrib/linux-kernel/lib/fse_compress.c @@ -35,22 +35,7 @@ /* ************************************************************** * Compiler specifics ****************************************************************/ -#ifdef _MSC_VER /* Visual Studio */ -# define FORCE_INLINE static __forceinline -# include /* For Visual 2005 */ -# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ -# pragma warning(disable : 4214) /* disable: C4214: non-int bitfields */ -#else -# if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ -# ifdef __GNUC__ -# define FORCE_INLINE static inline __attribute__((always_inline)) -# else -# define FORCE_INLINE static inline -# endif -# else -# define FORCE_INLINE static -# endif /* __STDC_VERSION__ */ -#endif +#define FORCE_INLINE static __attribute__((always_inline)) /* ************************************************************** diff --git a/contrib/linux-kernel/lib/fse_decompress.c b/contrib/linux-kernel/lib/fse_decompress.c index dd2dadc20..e53d99fb4 100644 --- a/contrib/linux-kernel/lib/fse_decompress.c +++ b/contrib/linux-kernel/lib/fse_decompress.c @@ -36,22 +36,7 @@ /* ************************************************************** * Compiler specifics ****************************************************************/ -#ifdef _MSC_VER /* Visual Studio */ -# define FORCE_INLINE static __forceinline -# include /* For Visual 2005 */ -# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ -# pragma warning(disable : 4214) /* disable: C4214: non-int bitfields */ -#else -# if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ -# ifdef __GNUC__ -# define FORCE_INLINE static inline __attribute__((always_inline)) -# else -# define FORCE_INLINE static inline -# endif -# else -# define FORCE_INLINE static -# endif /* __STDC_VERSION__ */ -#endif +#define FORCE_INLINE static __attribute__((always_inline)) /* ************************************************************** diff --git a/contrib/linux-kernel/lib/huf.h b/contrib/linux-kernel/lib/huf.h index 8ad3e2b42..f22a7d4f1 100644 --- a/contrib/linux-kernel/lib/huf.h +++ b/contrib/linux-kernel/lib/huf.h @@ -34,10 +34,6 @@ #ifndef HUF_H_298734234 #define HUF_H_298734234 -#if defined (__cplusplus) -extern "C" { -#endif - /* *** Dependencies *** */ #include /* size_t */ @@ -252,9 +248,4 @@ size_t HUF_decompress1X4_usingDTable(void* dst, size_t maxDstSize, const void* c #endif /* HUF_STATIC_LINKING_ONLY */ - -#if defined (__cplusplus) -} -#endif - #endif /* HUF_H_298734234 */ diff --git a/contrib/linux-kernel/lib/huf_compress.c b/contrib/linux-kernel/lib/huf_compress.c index 4a6ea9963..fd5dd0edc 100644 --- a/contrib/linux-kernel/lib/huf_compress.c +++ b/contrib/linux-kernel/lib/huf_compress.c @@ -32,13 +32,6 @@ - Public forum : https://groups.google.com/forum/#!forum/lz4c ****************************************************************** */ -/* ************************************************************** -* Compiler specifics -****************************************************************/ -#ifdef _MSC_VER /* Visual Studio */ -# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ -#endif - /* ************************************************************** * Includes diff --git a/contrib/linux-kernel/lib/huf_decompress.c b/contrib/linux-kernel/lib/huf_decompress.c index a1e9ffc32..1e19666a7 100644 --- a/contrib/linux-kernel/lib/huf_decompress.c +++ b/contrib/linux-kernel/lib/huf_decompress.c @@ -35,26 +35,13 @@ /* ************************************************************** * Compiler specifics ****************************************************************/ -#ifdef _MSC_VER /* Visual Studio */ -# define FORCE_INLINE static __forceinline -# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ -#else -# if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ -# ifdef __GNUC__ -# define FORCE_INLINE static inline __attribute__((always_inline)) -# else -# define FORCE_INLINE static inline -# endif -# else -# define FORCE_INLINE static -# endif /* __STDC_VERSION__ */ -#endif +#define FORCE_INLINE static __attribute__((always_inline)) /* ************************************************************** * Dependencies ****************************************************************/ -#include /* memcpy, memset */ +#include /* memcpy, memset */ #include "bitstream.h" /* BIT_* */ #include "fse.h" /* header compression */ #define HUF_STATIC_LINKING_ONLY diff --git a/contrib/linux-kernel/lib/mem.h b/contrib/linux-kernel/lib/mem.h index 75e75088b..b79fc27c8 100644 --- a/contrib/linux-kernel/lib/mem.h +++ b/contrib/linux-kernel/lib/mem.h @@ -10,10 +10,6 @@ #ifndef MEM_H_MODULE #define MEM_H_MODULE -#if defined (__cplusplus) -extern "C" { -#endif - /*-**************************************** * Dependencies ******************************************/ @@ -24,19 +20,7 @@ extern "C" { /*-**************************************** * Compiler specifics ******************************************/ -#if defined(_MSC_VER) /* Visual Studio */ -# include /* _byteswap_ulong */ -# include /* _byteswap_* */ -#endif -#if defined(__GNUC__) -# define MEM_STATIC static __inline __attribute__((unused)) -#elif defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) -# define MEM_STATIC static inline -#elif defined(_MSC_VER) -# define MEM_STATIC static __inline -#else -# define MEM_STATIC static /* this version may generate warnings for unused static functions; disable the relevant warning */ -#endif +#define MEM_STATIC static __inline __attribute__((unused)) /* code only tested on 32 and 64 bits systems */ #define MEM_STATIC_ASSERT(c) { enum { MEM_static_assert = 1/(int)(!!(c)) }; } @@ -46,55 +30,21 @@ MEM_STATIC void MEM_check(void) { MEM_STATIC_ASSERT((sizeof(size_t)==4) || (size /*-************************************************************** * Basic Types *****************************************************************/ -#if !defined (__VMS) && (defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) ) -# include - typedef uint8_t BYTE; - typedef uint16_t U16; - typedef int16_t S16; - typedef uint32_t U32; - typedef int32_t S32; - typedef uint64_t U64; - typedef int64_t S64; - typedef intptr_t iPtrDiff; - typedef uintptr_t uPtrDiff; -#else - typedef unsigned char BYTE; - typedef unsigned short U16; - typedef signed short S16; - typedef unsigned int U32; - typedef signed int S32; - typedef unsigned long long U64; - typedef signed long long S64; - typedef ptrdiff_t iPtrDiff; - typedef size_t uPtrDiff; -#endif +#include +typedef uint8_t BYTE; +typedef uint16_t U16; +typedef int16_t S16; +typedef uint32_t U32; +typedef int32_t S32; +typedef uint64_t U64; +typedef int64_t S64; +typedef intptr_t iPtrDiff; +typedef uintptr_t uPtrDiff; /*-************************************************************** * Memory I/O *****************************************************************/ -/* MEM_FORCE_MEMORY_ACCESS : - * By default, access to unaligned memory is controlled by `memcpy()`, which is safe and portable. - * Unfortunately, on some target/compiler combinations, the generated assembly is sub-optimal. - * The below switch allow to select different access method for improved performance. - * Method 0 (default) : use `memcpy()`. Safe and portable. - * Method 1 : `__packed` statement. It depends on compiler extension (i.e., not portable). - * This method is safe if your compiler supports it, and *generally* as fast or faster than `memcpy`. - * Method 2 : direct access. This method is portable but violate C standard. - * It can generate buggy code on targets depending on alignment. - * In some circumstances, it's the only known way to get the most performance (i.e. GCC + ARMv6) - * See http://fastcompression.blogspot.fr/2015/08/accessing-unaligned-memory.html for details. - * Prefer these methods in priority order (0 > 1 > 2) - */ -#ifndef MEM_FORCE_MEMORY_ACCESS /* can be defined externally, on command line for example */ -# if defined(__GNUC__) && ( defined(__ARM_ARCH_6__) || defined(__ARM_ARCH_6J__) || defined(__ARM_ARCH_6K__) || defined(__ARM_ARCH_6Z__) || defined(__ARM_ARCH_6ZK__) || defined(__ARM_ARCH_6T2__) ) -# define MEM_FORCE_MEMORY_ACCESS 2 -# elif defined(__INTEL_COMPILER) /*|| defined(_MSC_VER)*/ || \ - (defined(__GNUC__) && ( defined(__ARM_ARCH_7__) || defined(__ARM_ARCH_7A__) || defined(__ARM_ARCH_7R__) || defined(__ARM_ARCH_7M__) || defined(__ARM_ARCH_7S__) )) -# define MEM_FORCE_MEMORY_ACCESS 1 -# endif -#endif - MEM_STATIC unsigned MEM_32bits(void) { return sizeof(size_t)==4; } MEM_STATIC unsigned MEM_64bits(void) { return sizeof(size_t)==8; } @@ -104,45 +54,6 @@ MEM_STATIC unsigned MEM_isLittleEndian(void) return one.c[0]; } -#if defined(MEM_FORCE_MEMORY_ACCESS) && (MEM_FORCE_MEMORY_ACCESS==2) - -/* violates C standard, by lying on structure alignment. -Only use if no other choice to achieve best performance on target platform */ -MEM_STATIC U16 MEM_read16(const void* memPtr) { return *(const U16*) memPtr; } -MEM_STATIC U32 MEM_read32(const void* memPtr) { return *(const U32*) memPtr; } -MEM_STATIC U64 MEM_read64(const void* memPtr) { return *(const U64*) memPtr; } -MEM_STATIC U64 MEM_readST(const void* memPtr) { return *(const size_t*) memPtr; } - -MEM_STATIC void MEM_write16(void* memPtr, U16 value) { *(U16*)memPtr = value; } -MEM_STATIC void MEM_write32(void* memPtr, U32 value) { *(U32*)memPtr = value; } -MEM_STATIC void MEM_write64(void* memPtr, U64 value) { *(U64*)memPtr = value; } - -#elif defined(MEM_FORCE_MEMORY_ACCESS) && (MEM_FORCE_MEMORY_ACCESS==1) - -/* __pack instructions are safer, but compiler specific, hence potentially problematic for some compilers */ -/* currently only defined for gcc and icc */ -#if defined(_MSC_VER) || (defined(__INTEL_COMPILER) && defined(WIN32)) - __pragma( pack(push, 1) ) - typedef union { U16 u16; U32 u32; U64 u64; size_t st; } unalign; - __pragma( pack(pop) ) -#else - typedef union { U16 u16; U32 u32; U64 u64; size_t st; } __attribute__((packed)) unalign; -#endif - -MEM_STATIC U16 MEM_read16(const void* ptr) { return ((const unalign*)ptr)->u16; } -MEM_STATIC U32 MEM_read32(const void* ptr) { return ((const unalign*)ptr)->u32; } -MEM_STATIC U64 MEM_read64(const void* ptr) { return ((const unalign*)ptr)->u64; } -MEM_STATIC U64 MEM_readST(const void* ptr) { return ((const unalign*)ptr)->st; } - -MEM_STATIC void MEM_write16(void* memPtr, U16 value) { ((unalign*)memPtr)->u16 = value; } -MEM_STATIC void MEM_write32(void* memPtr, U32 value) { ((unalign*)memPtr)->u32 = value; } -MEM_STATIC void MEM_write64(void* memPtr, U64 value) { ((unalign*)memPtr)->u64 = value; } - -#else - -/* default method, safe and standard. - can sometimes prove slower */ - MEM_STATIC U16 MEM_read16(const void* memPtr) { U16 val; memcpy(&val, memPtr, sizeof(val)); return val; @@ -178,13 +89,9 @@ MEM_STATIC void MEM_write64(void* memPtr, U64 value) memcpy(memPtr, &value, sizeof(value)); } -#endif /* MEM_FORCE_MEMORY_ACCESS */ - MEM_STATIC U32 MEM_swap32(U32 in) { -#if defined(_MSC_VER) /* Visual Studio */ - return _byteswap_ulong(in); -#elif defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403) +#if defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403) return __builtin_bswap32(in); #else return ((in << 24) & 0xff000000 ) | @@ -196,9 +103,7 @@ MEM_STATIC U32 MEM_swap32(U32 in) MEM_STATIC U64 MEM_swap64(U64 in) { -#if defined(_MSC_VER) /* Visual Studio */ - return _byteswap_uint64(in); -#elif defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403) +#if defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403) return __builtin_bswap64(in); #else return ((in << 56) & 0xff00000000000000ULL) | @@ -367,8 +272,4 @@ MEM_STATIC U32 MEM_readMINMATCH(const void* memPtr, U32 length) } } -#if defined (__cplusplus) -} -#endif - #endif /* MEM_H_MODULE */ diff --git a/contrib/linux-kernel/lib/xxhash.c b/contrib/linux-kernel/lib/xxhash.c index 19b8bb46b..808a843ee 100644 --- a/contrib/linux-kernel/lib/xxhash.c +++ b/contrib/linux-kernel/lib/xxhash.c @@ -36,28 +36,6 @@ /* ************************************* * Tuning parameters ***************************************/ -/*!XXH_FORCE_MEMORY_ACCESS : - * By default, access to unaligned memory is controlled by `memcpy()`, which is safe and portable. - * Unfortunately, on some target/compiler combinations, the generated assembly is sub-optimal. - * The below switch allow to select different access method for improved performance. - * Method 0 (default) : use `memcpy()`. Safe and portable. - * Method 1 : `__packed` statement. It depends on compiler extension (ie, not portable). - * This method is safe if your compiler supports it, and *generally* as fast or faster than `memcpy`. - * Method 2 : direct access. This method doesn't depend on compiler but violate C standard. - * It can generate buggy code on targets which do not support unaligned memory accesses. - * But in some circumstances, it's the only known way to get the most performance (ie GCC + ARMv6) - * See http://stackoverflow.com/a/32095106/646947 for details. - * Prefer these methods in priority order (0 > 1 > 2) - */ -#ifndef XXH_FORCE_MEMORY_ACCESS /* can be defined externally, on command line for example */ -# if defined(__GNUC__) && ( defined(__ARM_ARCH_6__) || defined(__ARM_ARCH_6J__) || defined(__ARM_ARCH_6K__) || defined(__ARM_ARCH_6Z__) || defined(__ARM_ARCH_6ZK__) || defined(__ARM_ARCH_6T2__) ) -# define XXH_FORCE_MEMORY_ACCESS 2 -# elif (defined(__INTEL_COMPILER) && !defined(WIN32)) || \ - (defined(__GNUC__) && ( defined(__ARM_ARCH_7__) || defined(__ARM_ARCH_7A__) || defined(__ARM_ARCH_7R__) || defined(__ARM_ARCH_7M__) || defined(__ARM_ARCH_7S__) )) -# define XXH_FORCE_MEMORY_ACCESS 1 -# endif -#endif - /*!XXH_ACCEPT_NULL_INPUT_POINTER : * If the input pointer is a null pointer, xxHash default behavior is to trigger a memory access error, since it is a bad pointer. * When this option is enabled, xxHash output for null input pointers will be the same as a null-length input. @@ -73,9 +51,7 @@ * to improve speed for Big-endian CPU. * This option has no impact on Little_Endian CPU. */ -#ifndef XXH_FORCE_NATIVE_FORMAT /* can be defined externally */ -# define XXH_FORCE_NATIVE_FORMAT 0 -#endif +#define XXH_FORCE_NATIVE_FORMAT 0 /*!XXH_FORCE_ALIGN_CHECK : * This is a minor performance trick, only useful with lots of very small keys. @@ -83,13 +59,7 @@ * The check costs one initial branch per hash; set to 0 when the input data * is guaranteed to be aligned. */ -#ifndef XXH_FORCE_ALIGN_CHECK /* can be defined externally */ -# if defined(__i386) || defined(_M_IX86) || defined(__x86_64__) || defined(_M_X64) -# define XXH_FORCE_ALIGN_CHECK 0 -# else -# define XXH_FORCE_ALIGN_CHECK 1 -# endif -#endif +#define XXH_FORCE_ALIGN_CHECK 0 /* ************************************* @@ -108,127 +78,40 @@ static void* XXH_memcpy(void* dest, const void* src, size_t size) { return memcp # define XXH_STATIC_LINKING_ONLY #endif #include "xxhash.h" +#include "mem.h" /* ************************************* * Compiler Specific Options ***************************************/ -#ifdef _MSC_VER /* Visual Studio */ -# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ -# define FORCE_INLINE static __forceinline -#else -# if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ -# ifdef __GNUC__ -# define FORCE_INLINE static inline __attribute__((always_inline)) -# else -# define FORCE_INLINE static inline -# endif -# else -# define FORCE_INLINE static -# endif /* __STDC_VERSION__ */ -#endif +#define FORCE_INLINE static __attribute__((always_inline)) -/* ************************************* -* Basic Types -***************************************/ -#ifndef MEM_MODULE -# define MEM_MODULE -# if !defined (__VMS) && (defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) ) -# include - typedef uint8_t BYTE; - typedef uint16_t U16; - typedef uint32_t U32; - typedef int32_t S32; - typedef uint64_t U64; -# else - typedef unsigned char BYTE; - typedef unsigned short U16; - typedef unsigned int U32; - typedef signed int S32; - typedef unsigned long long U64; /* if your compiler doesn't support unsigned long long, replace by another 64-bit type here. Note that xxhash.h will also need to be updated. */ -# endif -#endif - - -#if (defined(XXH_FORCE_MEMORY_ACCESS) && (XXH_FORCE_MEMORY_ACCESS==2)) - -/* Force direct memory access. Only works on CPU which support unaligned memory access in hardware */ -static U32 XXH_read32(const void* memPtr) { return *(const U32*) memPtr; } -static U64 XXH_read64(const void* memPtr) { return *(const U64*) memPtr; } - -#elif (defined(XXH_FORCE_MEMORY_ACCESS) && (XXH_FORCE_MEMORY_ACCESS==1)) - -/* __pack instructions are safer, but compiler specific, hence potentially problematic for some compilers */ -/* currently only defined for gcc and icc */ -typedef union { U32 u32; U64 u64; } __attribute__((packed)) unalign; - -static U32 XXH_read32(const void* ptr) { return ((const unalign*)ptr)->u32; } -static U64 XXH_read64(const void* ptr) { return ((const unalign*)ptr)->u64; } - -#else - -/* portable and safe solution. Generally efficient. - * see : http://stackoverflow.com/a/32095106/646947 - */ - static U32 XXH_read32(const void* memPtr) { - U32 val; - memcpy(&val, memPtr, sizeof(val)); - return val; + return MEM_read32(memPtr); } static U64 XXH_read64(const void* memPtr) { - U64 val; - memcpy(&val, memPtr, sizeof(val)); - return val; + return MEM_read64(memPtr); } -#endif /* XXH_FORCE_DIRECT_MEMORY_ACCESS */ - /* **************************************** * Compiler-specific Functions and Macros ******************************************/ -#define GCC_VERSION (__GNUC__ * 100 + __GNUC_MINOR__) +#define XXH_rotl32(x,r) ((x << r) | (x >> (32 - r))) +#define XXH_rotl64(x,r) ((x << r) | (x >> (64 - r))) -/* Note : although _rotl exists for minGW (GCC under windows), performance seems poor */ -#if defined(_MSC_VER) -# define XXH_rotl32(x,r) _rotl(x,r) -# define XXH_rotl64(x,r) _rotl64(x,r) -#else -# define XXH_rotl32(x,r) ((x << r) | (x >> (32 - r))) -# define XXH_rotl64(x,r) ((x << r) | (x >> (64 - r))) -#endif - -#if defined(_MSC_VER) /* Visual Studio */ -# define XXH_swap32 _byteswap_ulong -# define XXH_swap64 _byteswap_uint64 -#elif GCC_VERSION >= 403 -# define XXH_swap32 __builtin_bswap32 -# define XXH_swap64 __builtin_bswap64 -#else static U32 XXH_swap32 (U32 x) { - return ((x << 24) & 0xff000000 ) | - ((x << 8) & 0x00ff0000 ) | - ((x >> 8) & 0x0000ff00 ) | - ((x >> 24) & 0x000000ff ); + return MEM_swap32(x); } static U64 XXH_swap64 (U64 x) { - return ((x << 56) & 0xff00000000000000ULL) | - ((x << 40) & 0x00ff000000000000ULL) | - ((x << 24) & 0x0000ff0000000000ULL) | - ((x << 8) & 0x000000ff00000000ULL) | - ((x >> 8) & 0x00000000ff000000ULL) | - ((x >> 24) & 0x0000000000ff0000ULL) | - ((x >> 40) & 0x000000000000ff00ULL) | - ((x >> 56) & 0x00000000000000ffULL); + return MEM_swap64(x); } -#endif /* ************************************* diff --git a/contrib/linux-kernel/lib/xxhash.h b/contrib/linux-kernel/lib/xxhash.h index 82e24d17d..17a678dd4 100644 --- a/contrib/linux-kernel/lib/xxhash.h +++ b/contrib/linux-kernel/lib/xxhash.h @@ -64,10 +64,6 @@ XXH64 13.8 GB/s 1.9 GB/s XXH32 6.8 GB/s 6.0 GB/s */ -#if defined (__cplusplus) -extern "C" { -#endif - #ifndef XXHASH_H_5627135585666179 #define XXHASH_H_5627135585666179 1 @@ -91,22 +87,7 @@ typedef enum { XXH_OK=0, XXH_ERROR } XXH_errorcode; * `xxhash.c` is automatically included. * It's not useful to compile and link it as a separate module anymore. */ -#ifdef XXH_PRIVATE_API -# ifndef XXH_STATIC_LINKING_ONLY -# define XXH_STATIC_LINKING_ONLY -# endif -# if defined(__GNUC__) -# define XXH_PUBLIC_API static __inline __attribute__((unused)) -# elif defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) -# define XXH_PUBLIC_API static inline -# elif defined(_MSC_VER) -# define XXH_PUBLIC_API static __inline -# else -# define XXH_PUBLIC_API static /* this version may generate warnings for unused static functions; disable the relevant warning */ -# endif -#else -# define XXH_PUBLIC_API /* do nothing */ -#endif /* XXH_PRIVATE_API */ +#define XXH_PUBLIC_API /* do nothing */ /*!XXH_NAMESPACE, aka Namespace Emulation : @@ -119,29 +100,6 @@ with the value of XXH_NAMESPACE (so avoid to keep it NULL and avoid numeric valu Note that no change is required within the calling program as long as it includes `xxhash.h` : regular symbol name will be automatically translated by this header. */ -#ifdef XXH_NAMESPACE -# define XXH_CAT(A,B) A##B -# define XXH_NAME2(A,B) XXH_CAT(A,B) -# define XXH32 XXH_NAME2(XXH_NAMESPACE, XXH32) -# define XXH64 XXH_NAME2(XXH_NAMESPACE, XXH64) -# define XXH_versionNumber XXH_NAME2(XXH_NAMESPACE, XXH_versionNumber) -# define XXH32_createState XXH_NAME2(XXH_NAMESPACE, XXH32_createState) -# define XXH64_createState XXH_NAME2(XXH_NAMESPACE, XXH64_createState) -# define XXH32_freeState XXH_NAME2(XXH_NAMESPACE, XXH32_freeState) -# define XXH64_freeState XXH_NAME2(XXH_NAMESPACE, XXH64_freeState) -# define XXH32_reset XXH_NAME2(XXH_NAMESPACE, XXH32_reset) -# define XXH64_reset XXH_NAME2(XXH_NAMESPACE, XXH64_reset) -# define XXH32_update XXH_NAME2(XXH_NAMESPACE, XXH32_update) -# define XXH64_update XXH_NAME2(XXH_NAMESPACE, XXH64_update) -# define XXH32_digest XXH_NAME2(XXH_NAMESPACE, XXH32_digest) -# define XXH64_digest XXH_NAME2(XXH_NAMESPACE, XXH64_digest) -# define XXH32_copyState XXH_NAME2(XXH_NAMESPACE, XXH32_copyState) -# define XXH64_copyState XXH_NAME2(XXH_NAMESPACE, XXH64_copyState) -# define XXH32_canonicalFromHash XXH_NAME2(XXH_NAMESPACE, XXH32_canonicalFromHash) -# define XXH64_canonicalFromHash XXH_NAME2(XXH_NAMESPACE, XXH64_canonicalFromHash) -# define XXH32_hashFromCanonical XXH_NAME2(XXH_NAMESPACE, XXH32_hashFromCanonical) -# define XXH64_hashFromCanonical XXH_NAME2(XXH_NAMESPACE, XXH64_hashFromCanonical) -#endif /* ************************************* @@ -227,10 +185,6 @@ When done, free XXH state space if it was allocated dynamically. /* ************************** * Utils ****************************/ -#if !(defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L)) /* ! C99 */ -# define restrict /* disable restrict */ -#endif - XXH_PUBLIC_API void XXH32_copyState(XXH32_state_t* restrict dst_state, const XXH32_state_t* restrict src_state); XXH_PUBLIC_API void XXH64_copyState(XXH64_state_t* restrict dst_state, const XXH64_state_t* restrict src_state); @@ -292,14 +246,4 @@ XXH_PUBLIC_API XXH64_hash_t XXH64_hashFromCanonical(const XXH64_canonical_t* src unsigned reserved[2]; /* never read nor write, will be removed in a future version */ }; /* typedef'd to XXH64_state_t */ - -# ifdef XXH_PRIVATE_API -# include "xxhash.c" /* include xxhash functions as `static`, for inlining */ -# endif - #endif /* XXH_STATIC_LINKING_ONLY && XXH_STATIC_H_3543687687345 */ - - -#if defined (__cplusplus) -} -#endif diff --git a/contrib/linux-kernel/lib/zstd_decompress.c b/contrib/linux-kernel/lib/zstd_decompress.c index 0c0d34686..82386c2e8 100644 --- a/contrib/linux-kernel/lib/zstd_decompress.c +++ b/contrib/linux-kernel/lib/zstd_decompress.c @@ -11,23 +11,6 @@ /* *************************************************************** * Tuning parameters *****************************************************************/ -/*! - * HEAPMODE : - * Select how default decompression function ZSTD_decompress() will allocate memory, - * in memory stack (0), or in memory heap (1, requires malloc()) - */ -#ifndef ZSTD_HEAPMODE -# define ZSTD_HEAPMODE 1 -#endif - -/*! -* LEGACY_SUPPORT : -* if set to 1, ZSTD_decompress() can decode older formats (v0.1+) -*/ -#ifndef ZSTD_LEGACY_SUPPORT -# define ZSTD_LEGACY_SUPPORT 0 -#endif - /*! * MAXWINDOWSIZE_DEFAULT : * maximum window size accepted by DStream, by default. @@ -49,19 +32,7 @@ #include "huf.h" #include "zstd_internal.h" -#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1) -# include "zstd_legacy.h" -#endif - - -#if defined(_MSC_VER) -# include /* https://msdn.microsoft.com/fr-fr/library/84szxsww(v=vs.90).aspx */ -# define ZSTD_PREFETCH(ptr) _mm_prefetch((const char*)ptr, _MM_HINT_T0) -#elif defined(__GNUC__) -# define ZSTD_PREFETCH(ptr) __builtin_prefetch(ptr, 0, 0) -#else -# define ZSTD_PREFETCH(ptr) /* disabled */ -#endif +#define ZSTD_PREFETCH(ptr) __builtin_prefetch(ptr, 0, 0) /*-************************************* * Macros @@ -220,9 +191,6 @@ unsigned ZSTD_isFrame(const void* buffer, size_t size) if (magic == ZSTD_MAGICNUMBER) return 1; if ((magic & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) return 1; } -#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) - if (ZSTD_isLegacy(buffer, size)) return 1; -#endif return 0; } @@ -320,12 +288,6 @@ size_t ZSTD_getFrameParams(ZSTD_frameParams* fparamsPtr, const void* src, size_t * - ZSTD_CONTENTSIZE_ERROR if an error occurred (e.g. invalid magic number, srcSize too small) */ unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize) { -#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) - if (ZSTD_isLegacy(src, srcSize)) { - unsigned long long const ret = ZSTD_getDecompressedSize_legacy(src, srcSize); - return ret == 0 ? ZSTD_CONTENTSIZE_UNKNOWN : ret; - } -#endif { ZSTD_frameParams fParams; if (ZSTD_getFrameParams(&fParams, src, srcSize) != 0) return ZSTD_CONTENTSIZE_ERROR; @@ -1472,9 +1434,6 @@ size_t ZSTD_generateNxBytes(void* dst, size_t dstCapacity, BYTE byte, size_t len * @return : the compressed size of the frame starting at `src` */ size_t ZSTD_findFrameCompressedSize(const void *src, size_t srcSize) { -#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) - if (ZSTD_isLegacy(src, srcSize)) return ZSTD_findFrameCompressedSizeLegacy(src, srcSize); -#endif if (srcSize >= ZSTD_skippableHeaderSize && (MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { return ZSTD_skippableHeaderSize + MEM_readLE32((const BYTE*)src + 4); @@ -1618,24 +1577,6 @@ static size_t ZSTD_decompressMultiFrame(ZSTD_DCtx* dctx, while (srcSize >= ZSTD_frameHeaderSize_prefix) { U32 magicNumber; -#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) - if (ZSTD_isLegacy(src, srcSize)) { - size_t decodedSize; - size_t const frameSize = ZSTD_findFrameCompressedSizeLegacy(src, srcSize); - if (ZSTD_isError(frameSize)) return frameSize; - - decodedSize = ZSTD_decompressLegacy(dst, dstCapacity, src, frameSize, dict, dictSize); - - dst = (BYTE*)dst + decodedSize; - dstCapacity -= decodedSize; - - src = (const BYTE*)src + frameSize; - srcSize -= frameSize; - - continue; - } -#endif - magicNumber = MEM_readLE32(src); if (magicNumber != ZSTD_MAGICNUMBER) { if ((magicNumber & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) { @@ -1698,17 +1639,8 @@ size_t ZSTD_decompressDCtx(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const size_t ZSTD_decompress(void* dst, size_t dstCapacity, const void* src, size_t srcSize) { -#if defined(ZSTD_HEAPMODE) && (ZSTD_HEAPMODE==1) - size_t regenSize; - ZSTD_DCtx* const dctx = ZSTD_createDCtx(); - if (dctx==NULL) return ERROR(memory_allocation); - regenSize = ZSTD_decompressDCtx(dctx, dst, dstCapacity, src, srcSize); - ZSTD_freeDCtx(dctx); - return regenSize; -#else /* stack mode */ ZSTD_DCtx dctx; return ZSTD_decompressDCtx(&dctx, dst, dstCapacity, src, srcSize); -#endif } @@ -2216,10 +2148,6 @@ size_t ZSTD_freeDStream(ZSTD_DStream* zds) zds->inBuff = NULL; ZSTD_free(zds->outBuff, cMem); zds->outBuff = NULL; -#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT >= 1) - if (zds->legacyContext) - ZSTD_freeLegacyStreamContext(zds->legacyContext, zds->previousLegacyVersion); -#endif ZSTD_free(zds, cMem); return 0; } @@ -2306,11 +2234,6 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB char* op = ostart; U32 someMoreWork = 1; -#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1) - if (zds->legacyVersion) - return ZSTD_decompressLegacyStream(zds->legacyContext, zds->legacyVersion, output, input); -#endif - while (someMoreWork) { switch(zds->stage) { @@ -2321,21 +2244,7 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB case zdss_loadHeader : { size_t const hSize = ZSTD_getFrameParams(&zds->fParams, zds->headerBuffer, zds->lhSize); if (ZSTD_isError(hSize)) -#if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1) - { U32 const legacyVersion = ZSTD_isLegacy(istart, iend-istart); - if (legacyVersion) { - const void* const dict = zds->ddict ? zds->ddict->dictContent : NULL; - size_t const dictSize = zds->ddict ? zds->ddict->dictSize : 0; - CHECK_F(ZSTD_initLegacyStream(&zds->legacyContext, zds->previousLegacyVersion, legacyVersion, - dict, dictSize)); - zds->legacyVersion = zds->previousLegacyVersion = legacyVersion; - return ZSTD_decompressLegacyStream(zds->legacyContext, zds->legacyVersion, output, input); - } else { - return hSize; /* error */ - } } -#else return hSize; -#endif if (hSize != 0) { /* need more input */ size_t const toLoad = hSize - zds->lhSize; /* if hSize!=0, hSize > zds->lhSize */ if (toLoad > (size_t)(iend-ip)) { /* not enough input to load full header */ diff --git a/contrib/linux-kernel/lib/zstd_errors.h b/contrib/linux-kernel/lib/zstd_errors.h index 1cf0b0adb..8285df8e5 100644 --- a/contrib/linux-kernel/lib/zstd_errors.h +++ b/contrib/linux-kernel/lib/zstd_errors.h @@ -10,27 +10,12 @@ #ifndef ZSTD_ERRORS_H_398273423 #define ZSTD_ERRORS_H_398273423 -#if defined (__cplusplus) -extern "C" { -#endif - /*===== dependency =====*/ #include /* size_t */ /* ===== ZSTDERRORLIB_API : control library symbols visibility ===== */ -#if defined(__GNUC__) && (__GNUC__ >= 4) -# define ZSTDERRORLIB_VISIBILITY __attribute__ ((visibility ("default"))) -#else -# define ZSTDERRORLIB_VISIBILITY -#endif -#if defined(ZSTD_DLL_EXPORT) && (ZSTD_DLL_EXPORT==1) -# define ZSTDERRORLIB_API __declspec(dllexport) ZSTDERRORLIB_VISIBILITY -#elif defined(ZSTD_DLL_IMPORT) && (ZSTD_DLL_IMPORT==1) -# define ZSTDERRORLIB_API __declspec(dllimport) ZSTDERRORLIB_VISIBILITY /* It isn't required but allows to generate better code, saving a function pointer load from the IAT and an indirect jump.*/ -#else -# define ZSTDERRORLIB_API ZSTDERRORLIB_VISIBILITY -#endif +#define ZSTDERRORLIB_API /*-**************************************** * error codes list @@ -67,9 +52,4 @@ typedef enum { ZSTDERRORLIB_API ZSTD_ErrorCode ZSTD_getErrorCode(size_t functionResult); ZSTDERRORLIB_API const char* ZSTD_getErrorString(ZSTD_ErrorCode code); - -#if defined (__cplusplus) -} -#endif - #endif /* ZSTD_ERRORS_H_398273423 */ diff --git a/contrib/linux-kernel/lib/zstd_internal.h b/contrib/linux-kernel/lib/zstd_internal.h index 8b2c27f53..16a502a69 100644 --- a/contrib/linux-kernel/lib/zstd_internal.h +++ b/contrib/linux-kernel/lib/zstd_internal.h @@ -13,33 +13,8 @@ /*-******************************************************* * Compiler specifics *********************************************************/ -#ifdef _MSC_VER /* Visual Studio */ -# define FORCE_INLINE static __forceinline -# include /* For Visual 2005 */ -# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ -# pragma warning(disable : 4324) /* disable: C4324: padded structure */ -# pragma warning(disable : 4100) /* disable: C4100: unreferenced formal parameter */ -#else -# if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ -# ifdef __GNUC__ -# define FORCE_INLINE static inline __attribute__((always_inline)) -# else -# define FORCE_INLINE static inline -# endif -# else -# define FORCE_INLINE static -# endif /* __STDC_VERSION__ */ -#endif - -#ifdef _MSC_VER -# define FORCE_NOINLINE static __declspec(noinline) -#else -# ifdef __GNUC__ -# define FORCE_NOINLINE static __attribute__((__noinline__)) -# else -# define FORCE_NOINLINE static -# endif -#endif +#define FORCE_INLINE static __attribute__((always_inline)) +#define FORCE_NOINLINE static __attribute__((__noinline__)) /*-************************************* From be7da5d98b88f52eea93b4cfd41569ad9e9de247 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Thu, 30 Mar 2017 11:40:05 -0700 Subject: [PATCH 098/305] Change stdlib includes to linux includes --- contrib/linux-kernel/include/zstd.h | 2 +- contrib/linux-kernel/lib/error_private.h | 2 +- contrib/linux-kernel/lib/fse.h | 2 +- contrib/linux-kernel/lib/fse_compress.c | 4 +--- contrib/linux-kernel/lib/fse_decompress.c | 3 +-- contrib/linux-kernel/lib/huf.h | 2 +- contrib/linux-kernel/lib/huf_compress.c | 3 +-- contrib/linux-kernel/lib/mem.h | 5 ++--- contrib/linux-kernel/lib/xxhash.c | 3 +-- contrib/linux-kernel/lib/xxhash.h | 2 +- contrib/linux-kernel/lib/zstd_common.c | 1 - contrib/linux-kernel/lib/zstd_compress.c | 2 +- contrib/linux-kernel/lib/zstd_decompress.c | 2 +- contrib/linux-kernel/lib/zstd_errors.h | 2 +- 14 files changed, 14 insertions(+), 21 deletions(-) diff --git a/contrib/linux-kernel/include/zstd.h b/contrib/linux-kernel/include/zstd.h index ff7f86a6d..8e1f805d0 100644 --- a/contrib/linux-kernel/include/zstd.h +++ b/contrib/linux-kernel/include/zstd.h @@ -11,7 +11,7 @@ #define ZSTD_H_235446 /* ====== Dependency ======*/ -#include /* size_t */ +#include /* size_t */ /* ===== ZSTDLIB_API : control library symbols visibility ===== */ diff --git a/contrib/linux-kernel/lib/error_private.h b/contrib/linux-kernel/lib/error_private.h index e88f2fd17..7b2ce36d4 100644 --- a/contrib/linux-kernel/lib/error_private.h +++ b/contrib/linux-kernel/lib/error_private.h @@ -15,7 +15,7 @@ /* **************************************** * Dependencies ******************************************/ -#include /* size_t */ +#include /* size_t */ #include "zstd_errors.h" /* enum list */ diff --git a/contrib/linux-kernel/lib/fse.h b/contrib/linux-kernel/lib/fse.h index 538d52427..b833f6c2c 100644 --- a/contrib/linux-kernel/lib/fse.h +++ b/contrib/linux-kernel/lib/fse.h @@ -38,7 +38,7 @@ /*-***************************************** * Dependencies ******************************************/ -#include /* size_t, ptrdiff_t */ +#include /* size_t, ptrdiff_t */ /*-***************************************** diff --git a/contrib/linux-kernel/lib/fse_compress.c b/contrib/linux-kernel/lib/fse_compress.c index dbfa510c6..29071c483 100644 --- a/contrib/linux-kernel/lib/fse_compress.c +++ b/contrib/linux-kernel/lib/fse_compress.c @@ -41,9 +41,7 @@ /* ************************************************************** * Includes ****************************************************************/ -#include /* malloc, free, qsort */ -#include /* memcpy, memset */ -#include /* printf (debug) */ +#include /* memcpy, memset */ #include "bitstream.h" #define FSE_STATIC_LINKING_ONLY #include "fse.h" diff --git a/contrib/linux-kernel/lib/fse_decompress.c b/contrib/linux-kernel/lib/fse_decompress.c index e53d99fb4..958a068bc 100644 --- a/contrib/linux-kernel/lib/fse_decompress.c +++ b/contrib/linux-kernel/lib/fse_decompress.c @@ -42,8 +42,7 @@ /* ************************************************************** * Includes ****************************************************************/ -#include /* malloc, free, qsort */ -#include /* memcpy, memset */ +#include /* memcpy, memset */ #include "bitstream.h" #define FSE_STATIC_LINKING_ONLY #include "fse.h" diff --git a/contrib/linux-kernel/lib/huf.h b/contrib/linux-kernel/lib/huf.h index f22a7d4f1..63306b4bf 100644 --- a/contrib/linux-kernel/lib/huf.h +++ b/contrib/linux-kernel/lib/huf.h @@ -36,7 +36,7 @@ /* *** Dependencies *** */ -#include /* size_t */ +#include /* size_t */ /* *** simple functions *** */ diff --git a/contrib/linux-kernel/lib/huf_compress.c b/contrib/linux-kernel/lib/huf_compress.c index fd5dd0edc..af1a1e6c3 100644 --- a/contrib/linux-kernel/lib/huf_compress.c +++ b/contrib/linux-kernel/lib/huf_compress.c @@ -36,8 +36,7 @@ /* ************************************************************** * Includes ****************************************************************/ -#include /* memcpy, memset */ -#include /* printf (debug) */ +#include /* memcpy, memset */ #include "bitstream.h" #define FSE_STATIC_LINKING_ONLY /* FSE_optimalTableLog_internal */ #include "fse.h" /* header compression */ diff --git a/contrib/linux-kernel/lib/mem.h b/contrib/linux-kernel/lib/mem.h index b79fc27c8..1b8098e72 100644 --- a/contrib/linux-kernel/lib/mem.h +++ b/contrib/linux-kernel/lib/mem.h @@ -13,8 +13,8 @@ /*-**************************************** * Dependencies ******************************************/ -#include /* size_t, ptrdiff_t */ -#include /* memcpy */ +#include /* size_t, ptrdiff_t */ +#include /* memcpy */ /*-**************************************** @@ -30,7 +30,6 @@ MEM_STATIC void MEM_check(void) { MEM_STATIC_ASSERT((sizeof(size_t)==4) || (size /*-************************************************************** * Basic Types *****************************************************************/ -#include typedef uint8_t BYTE; typedef uint16_t U16; typedef int16_t S16; diff --git a/contrib/linux-kernel/lib/xxhash.c b/contrib/linux-kernel/lib/xxhash.c index 808a843ee..dccdc65c9 100644 --- a/contrib/linux-kernel/lib/xxhash.c +++ b/contrib/linux-kernel/lib/xxhash.c @@ -67,11 +67,10 @@ ***************************************/ /* Modify the local functions below should you wish to use some other memory routines */ /* for malloc(), free() */ -#include static void* XXH_malloc(size_t s) { return malloc(s); } static void XXH_free (void* p) { free(p); } /* for memcpy() */ -#include +#include static void* XXH_memcpy(void* dest, const void* src, size_t size) { return memcpy(dest,src,size); } #ifndef XXH_STATIC_LINKING_ONLY diff --git a/contrib/linux-kernel/lib/xxhash.h b/contrib/linux-kernel/lib/xxhash.h index 17a678dd4..b6a56707e 100644 --- a/contrib/linux-kernel/lib/xxhash.h +++ b/contrib/linux-kernel/lib/xxhash.h @@ -71,7 +71,7 @@ XXH32 6.8 GB/s 6.0 GB/s /* **************************** * Definitions ******************************/ -#include /* size_t */ +#include /* size_t */ typedef enum { XXH_OK=0, XXH_ERROR } XXH_errorcode; diff --git a/contrib/linux-kernel/lib/zstd_common.c b/contrib/linux-kernel/lib/zstd_common.c index 69199cb6f..3d24742ee 100644 --- a/contrib/linux-kernel/lib/zstd_common.c +++ b/contrib/linux-kernel/lib/zstd_common.c @@ -12,7 +12,6 @@ /*-************************************* * Dependencies ***************************************/ -#include /* malloc */ #include "error_private.h" #define ZSTD_STATIC_LINKING_ONLY #include "zstd.h" /* declaration of ZSTD_isError, ZSTD_getErrorName, ZSTD_getErrorCode, ZSTD_getErrorString, ZSTD_versionNumber */ diff --git a/contrib/linux-kernel/lib/zstd_compress.c b/contrib/linux-kernel/lib/zstd_compress.c index 84d898e57..d4b9a4751 100644 --- a/contrib/linux-kernel/lib/zstd_compress.c +++ b/contrib/linux-kernel/lib/zstd_compress.c @@ -11,7 +11,7 @@ /*-************************************* * Dependencies ***************************************/ -#include /* memset */ +#include /* memset */ #include "mem.h" #define FSE_STATIC_LINKING_ONLY /* FSE_encodeSymbol */ #include "fse.h" diff --git a/contrib/linux-kernel/lib/zstd_decompress.c b/contrib/linux-kernel/lib/zstd_decompress.c index 82386c2e8..bb72d6a4a 100644 --- a/contrib/linux-kernel/lib/zstd_decompress.c +++ b/contrib/linux-kernel/lib/zstd_decompress.c @@ -24,7 +24,7 @@ /*-******************************************************* * Dependencies *********************************************************/ -#include /* memcpy, memmove, memset */ +#include /* memcpy, memmove, memset */ #include "mem.h" /* low level memory routines */ #define FSE_STATIC_LINKING_ONLY #include "fse.h" diff --git a/contrib/linux-kernel/lib/zstd_errors.h b/contrib/linux-kernel/lib/zstd_errors.h index 8285df8e5..37e491b7e 100644 --- a/contrib/linux-kernel/lib/zstd_errors.h +++ b/contrib/linux-kernel/lib/zstd_errors.h @@ -11,7 +11,7 @@ #define ZSTD_ERRORS_H_398273423 /*===== dependency =====*/ -#include /* size_t */ +#include /* size_t */ /* ===== ZSTDERRORLIB_API : control library symbols visibility ===== */ From 6ce58897dabc45b1118c7db85883dfa736b6a8ce Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Thu, 30 Mar 2017 12:11:19 -0700 Subject: [PATCH 099/305] Use for unaligned memory access --- contrib/linux-kernel/lib/mem.h | 115 +++++++----------------------- contrib/linux-kernel/lib/xxhash.c | 35 +++------ 2 files changed, 33 insertions(+), 117 deletions(-) diff --git a/contrib/linux-kernel/lib/mem.h b/contrib/linux-kernel/lib/mem.h index 1b8098e72..6040aa5c7 100644 --- a/contrib/linux-kernel/lib/mem.h +++ b/contrib/linux-kernel/lib/mem.h @@ -13,6 +13,7 @@ /*-**************************************** * Dependencies ******************************************/ +#include #include /* size_t, ptrdiff_t */ #include /* memcpy */ @@ -47,104 +48,62 @@ typedef uintptr_t uPtrDiff; MEM_STATIC unsigned MEM_32bits(void) { return sizeof(size_t)==4; } MEM_STATIC unsigned MEM_64bits(void) { return sizeof(size_t)==8; } +#if defined(__LITTLE_ENDIAN) +# define MEM_LITTLE_ENDIAN 1 +#else +# define MEM_LITTLE_ENDIAN 0 +#endif + MEM_STATIC unsigned MEM_isLittleEndian(void) { - const union { U32 u; BYTE c[4]; } one = { 1 }; /* don't use static : performance detrimental */ - return one.c[0]; + return MEM_LITTLE_ENDIAN; } MEM_STATIC U16 MEM_read16(const void* memPtr) { - U16 val; memcpy(&val, memPtr, sizeof(val)); return val; + return get_unaligned((const U16*)memPtr); } MEM_STATIC U32 MEM_read32(const void* memPtr) { - U32 val; memcpy(&val, memPtr, sizeof(val)); return val; + return get_unaligned((const U32*)memPtr); } MEM_STATIC U64 MEM_read64(const void* memPtr) { - U64 val; memcpy(&val, memPtr, sizeof(val)); return val; + return get_unaligned((const U64*)memPtr); } MEM_STATIC size_t MEM_readST(const void* memPtr) { - size_t val; memcpy(&val, memPtr, sizeof(val)); return val; + return get_unaligned((const size_t*)memPtr); } MEM_STATIC void MEM_write16(void* memPtr, U16 value) { - memcpy(memPtr, &value, sizeof(value)); + put_unaligned(value, (U16*)memPtr); } MEM_STATIC void MEM_write32(void* memPtr, U32 value) { - memcpy(memPtr, &value, sizeof(value)); + put_unaligned(value, (U32*)memPtr); } MEM_STATIC void MEM_write64(void* memPtr, U64 value) { - memcpy(memPtr, &value, sizeof(value)); -} - -MEM_STATIC U32 MEM_swap32(U32 in) -{ -#if defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403) - return __builtin_bswap32(in); -#else - return ((in << 24) & 0xff000000 ) | - ((in << 8) & 0x00ff0000 ) | - ((in >> 8) & 0x0000ff00 ) | - ((in >> 24) & 0x000000ff ); -#endif -} - -MEM_STATIC U64 MEM_swap64(U64 in) -{ -#if defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403) - return __builtin_bswap64(in); -#else - return ((in << 56) & 0xff00000000000000ULL) | - ((in << 40) & 0x00ff000000000000ULL) | - ((in << 24) & 0x0000ff0000000000ULL) | - ((in << 8) & 0x000000ff00000000ULL) | - ((in >> 8) & 0x00000000ff000000ULL) | - ((in >> 24) & 0x0000000000ff0000ULL) | - ((in >> 40) & 0x000000000000ff00ULL) | - ((in >> 56) & 0x00000000000000ffULL); -#endif -} - -MEM_STATIC size_t MEM_swapST(size_t in) -{ - if (MEM_32bits()) - return (size_t)MEM_swap32((U32)in); - else - return (size_t)MEM_swap64((U64)in); + put_unaligned(value, (U64*)memPtr); } /*=== Little endian r/w ===*/ MEM_STATIC U16 MEM_readLE16(const void* memPtr) { - if (MEM_isLittleEndian()) - return MEM_read16(memPtr); - else { - const BYTE* p = (const BYTE*)memPtr; - return (U16)(p[0] + (p[1]<<8)); - } + return get_unaligned_le16(memPtr); } MEM_STATIC void MEM_writeLE16(void* memPtr, U16 val) { - if (MEM_isLittleEndian()) { - MEM_write16(memPtr, val); - } else { - BYTE* p = (BYTE*)memPtr; - p[0] = (BYTE)val; - p[1] = (BYTE)(val>>8); - } + put_unaligned_le16(val, memPtr); } MEM_STATIC U32 MEM_readLE24(const void* memPtr) @@ -160,34 +119,22 @@ MEM_STATIC void MEM_writeLE24(void* memPtr, U32 val) MEM_STATIC U32 MEM_readLE32(const void* memPtr) { - if (MEM_isLittleEndian()) - return MEM_read32(memPtr); - else - return MEM_swap32(MEM_read32(memPtr)); + return get_unaligned_le32(memPtr); } MEM_STATIC void MEM_writeLE32(void* memPtr, U32 val32) { - if (MEM_isLittleEndian()) - MEM_write32(memPtr, val32); - else - MEM_write32(memPtr, MEM_swap32(val32)); + put_unaligned_le32(val32, memPtr); } MEM_STATIC U64 MEM_readLE64(const void* memPtr) { - if (MEM_isLittleEndian()) - return MEM_read64(memPtr); - else - return MEM_swap64(MEM_read64(memPtr)); + return get_unaligned_le64(memPtr); } MEM_STATIC void MEM_writeLE64(void* memPtr, U64 val64) { - if (MEM_isLittleEndian()) - MEM_write64(memPtr, val64); - else - MEM_write64(memPtr, MEM_swap64(val64)); + put_unaligned_le64(val64, memPtr); } MEM_STATIC size_t MEM_readLEST(const void* memPtr) @@ -210,34 +157,22 @@ MEM_STATIC void MEM_writeLEST(void* memPtr, size_t val) MEM_STATIC U32 MEM_readBE32(const void* memPtr) { - if (MEM_isLittleEndian()) - return MEM_swap32(MEM_read32(memPtr)); - else - return MEM_read32(memPtr); + return get_unaligned_be32(memPtr); } MEM_STATIC void MEM_writeBE32(void* memPtr, U32 val32) { - if (MEM_isLittleEndian()) - MEM_write32(memPtr, MEM_swap32(val32)); - else - MEM_write32(memPtr, val32); + put_unaligned_be32(val32, memPtr); } MEM_STATIC U64 MEM_readBE64(const void* memPtr) { - if (MEM_isLittleEndian()) - return MEM_swap64(MEM_read64(memPtr)); - else - return MEM_read64(memPtr); + return get_unaligned_be64(memPtr); } MEM_STATIC void MEM_writeBE64(void* memPtr, U64 val64) { - if (MEM_isLittleEndian()) - MEM_write64(memPtr, MEM_swap64(val64)); - else - MEM_write64(memPtr, val64); + put_unaligned_be64(val64, memPtr); } MEM_STATIC size_t MEM_readBEST(const void* memPtr) diff --git a/contrib/linux-kernel/lib/xxhash.c b/contrib/linux-kernel/lib/xxhash.c index dccdc65c9..1918836d7 100644 --- a/contrib/linux-kernel/lib/xxhash.c +++ b/contrib/linux-kernel/lib/xxhash.c @@ -103,16 +103,6 @@ static U64 XXH_read64(const void* memPtr) #define XXH_rotl32(x,r) ((x << r) | (x >> (32 - r))) #define XXH_rotl64(x,r) ((x << r) | (x >> (64 - r))) -static U32 XXH_swap32 (U32 x) -{ - return MEM_swap32(x); -} -static U64 XXH_swap64 (U64 x) -{ - return MEM_swap64(x); -} - - /* ************************************* * Architecture Macros ***************************************/ @@ -120,8 +110,7 @@ typedef enum { XXH_bigEndian=0, XXH_littleEndian=1 } XXH_endianess; /* XXH_CPU_LITTLE_ENDIAN can be defined externally, for example on the compiler command line */ #ifndef XXH_CPU_LITTLE_ENDIAN - static const int g_one = 1; -# define XXH_CPU_LITTLE_ENDIAN (*(const char*)(&g_one)) +# define XXH_CPU_LITTLE_ENDIAN MEM_LITTLE_ENDIAN #endif @@ -130,12 +119,9 @@ typedef enum { XXH_bigEndian=0, XXH_littleEndian=1 } XXH_endianess; *****************************/ typedef enum { XXH_aligned, XXH_unaligned } XXH_alignment; -FORCE_INLINE U32 XXH_readLE32_align(const void* ptr, XXH_endianess endian, XXH_alignment align) +FORCE_INLINE U32 XXH_readLE32_align(const void* ptr, XXH_endianess, XXH_alignment) { - if (align==XXH_unaligned) - return endian==XXH_littleEndian ? XXH_read32(ptr) : XXH_swap32(XXH_read32(ptr)); - else - return endian==XXH_littleEndian ? *(const U32*)ptr : XXH_swap32(*(const U32*)ptr); + return MEM_readLE32(ptr); } FORCE_INLINE U32 XXH_readLE32(const void* ptr, XXH_endianess endian) @@ -145,15 +131,12 @@ FORCE_INLINE U32 XXH_readLE32(const void* ptr, XXH_endianess endian) static U32 XXH_readBE32(const void* ptr) { - return XXH_CPU_LITTLE_ENDIAN ? XXH_swap32(XXH_read32(ptr)) : XXH_read32(ptr); + return MEM_readBE32(ptr); } FORCE_INLINE U64 XXH_readLE64_align(const void* ptr, XXH_endianess endian, XXH_alignment align) { - if (align==XXH_unaligned) - return endian==XXH_littleEndian ? XXH_read64(ptr) : XXH_swap64(XXH_read64(ptr)); - else - return endian==XXH_littleEndian ? *(const U64*)ptr : XXH_swap64(*(const U64*)ptr); + return MEM_readLE64(ptr); } FORCE_INLINE U64 XXH_readLE64(const void* ptr, XXH_endianess endian) @@ -163,7 +146,7 @@ FORCE_INLINE U64 XXH_readLE64(const void* ptr, XXH_endianess endian) static U64 XXH_readBE64(const void* ptr) { - return XXH_CPU_LITTLE_ENDIAN ? XXH_swap64(XXH_read64(ptr)) : XXH_read64(ptr); + return MEM_readBE64(ptr); } @@ -729,15 +712,13 @@ XXH_PUBLIC_API unsigned long long XXH64_digest (const XXH64_state_t* state_in) XXH_PUBLIC_API void XXH32_canonicalFromHash(XXH32_canonical_t* dst, XXH32_hash_t hash) { XXH_STATIC_ASSERT(sizeof(XXH32_canonical_t) == sizeof(XXH32_hash_t)); - if (XXH_CPU_LITTLE_ENDIAN) hash = XXH_swap32(hash); - memcpy(dst, &hash, sizeof(*dst)); + MEM_writeBE32(dst, hash); } XXH_PUBLIC_API void XXH64_canonicalFromHash(XXH64_canonical_t* dst, XXH64_hash_t hash) { XXH_STATIC_ASSERT(sizeof(XXH64_canonical_t) == sizeof(XXH64_hash_t)); - if (XXH_CPU_LITTLE_ENDIAN) hash = XXH_swap64(hash); - memcpy(dst, &hash, sizeof(*dst)); + MEM_writeBE64(dst, hash); } XXH_PUBLIC_API XXH32_hash_t XXH32_hashFromCanonical(const XXH32_canonical_t* src) From e48b1355213d6242d89487e69e6aba832a044c98 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Thu, 30 Mar 2017 14:33:59 -0700 Subject: [PATCH 100/305] Remove STATIC_LINKING_ONLY defines --- contrib/linux-kernel/include/zstd.h | 7 +-- contrib/linux-kernel/lib/entropy_common.c | 2 - contrib/linux-kernel/lib/fse.h | 5 --- contrib/linux-kernel/lib/fse_compress.c | 1 - contrib/linux-kernel/lib/fse_decompress.c | 1 - contrib/linux-kernel/lib/huf.h | 4 -- contrib/linux-kernel/lib/huf_compress.c | 2 - contrib/linux-kernel/lib/huf_decompress.c | 1 - contrib/linux-kernel/lib/xxhash.c | 3 -- contrib/linux-kernel/lib/xxhash.h | 50 ++++++++++------------ contrib/linux-kernel/lib/zstd_common.c | 1 - contrib/linux-kernel/lib/zstd_compress.c | 2 - contrib/linux-kernel/lib/zstd_decompress.c | 2 - contrib/linux-kernel/lib/zstd_internal.h | 4 -- 14 files changed, 23 insertions(+), 62 deletions(-) diff --git a/contrib/linux-kernel/include/zstd.h b/contrib/linux-kernel/include/zstd.h index 8e1f805d0..8b1afa4ba 100644 --- a/contrib/linux-kernel/include/zstd.h +++ b/contrib/linux-kernel/include/zstd.h @@ -315,11 +315,6 @@ ZSTDLIB_API size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* outp ZSTDLIB_API size_t ZSTD_DStreamInSize(void); /*!< recommended size for input buffer */ ZSTDLIB_API size_t ZSTD_DStreamOutSize(void); /*!< recommended size for output buffer. Guarantee to successfully flush at least one complete block in all circumstances. */ -#endif /* ZSTD_H_235446 */ - - -#if defined(ZSTD_STATIC_LINKING_ONLY) && !defined(ZSTD_H_ZSTD_STATIC_LINKING_ONLY) -#define ZSTD_H_ZSTD_STATIC_LINKING_ONLY /**************************************************************************************** * START OF ADVANCED AND EXPERIMENTAL FUNCTIONS @@ -753,4 +748,4 @@ ZSTDLIB_API size_t ZSTD_decompressBlock(ZSTD_DCtx* dctx, void* dst, size_t dstCa ZSTDLIB_API size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize); /**< insert block into `dctx` history. Useful for uncompressed blocks */ -#endif /* ZSTD_H_ZSTD_STATIC_LINKING_ONLY */ +#endif /* ZSTD_H_235446 */ diff --git a/contrib/linux-kernel/lib/entropy_common.c b/contrib/linux-kernel/lib/entropy_common.c index c9d489bc7..f6a43662d 100644 --- a/contrib/linux-kernel/lib/entropy_common.c +++ b/contrib/linux-kernel/lib/entropy_common.c @@ -37,9 +37,7 @@ ***************************************/ #include "mem.h" #include "error_private.h" /* ERR_*, ERROR */ -#define FSE_STATIC_LINKING_ONLY /* FSE_MIN_TABLELOG */ #include "fse.h" -#define HUF_STATIC_LINKING_ONLY /* HUF_TABLELOG_ABSOLUTEMAX */ #include "huf.h" diff --git a/contrib/linux-kernel/lib/fse.h b/contrib/linux-kernel/lib/fse.h index b833f6c2c..a5c3980a7 100644 --- a/contrib/linux-kernel/lib/fse.h +++ b/contrib/linux-kernel/lib/fse.h @@ -286,8 +286,6 @@ If there is an error, the function will return an error code, which can be teste */ -#ifdef FSE_STATIC_LINKING_ONLY - /* *** Dependency *** */ #include "bitstream.h" @@ -672,7 +670,4 @@ MEM_STATIC unsigned FSE_endOfDState(const FSE_DState_t* DStatePtr) #define FSE_TABLESTEP(tableSize) ((tableSize>>1) + (tableSize>>3) + 3) -#endif /* FSE_STATIC_LINKING_ONLY */ - - #endif /* FSE_H */ diff --git a/contrib/linux-kernel/lib/fse_compress.c b/contrib/linux-kernel/lib/fse_compress.c index 29071c483..8b85b85f1 100644 --- a/contrib/linux-kernel/lib/fse_compress.c +++ b/contrib/linux-kernel/lib/fse_compress.c @@ -43,7 +43,6 @@ ****************************************************************/ #include /* memcpy, memset */ #include "bitstream.h" -#define FSE_STATIC_LINKING_ONLY #include "fse.h" diff --git a/contrib/linux-kernel/lib/fse_decompress.c b/contrib/linux-kernel/lib/fse_decompress.c index 958a068bc..ebc228daf 100644 --- a/contrib/linux-kernel/lib/fse_decompress.c +++ b/contrib/linux-kernel/lib/fse_decompress.c @@ -44,7 +44,6 @@ ****************************************************************/ #include /* memcpy, memset */ #include "bitstream.h" -#define FSE_STATIC_LINKING_ONLY #include "fse.h" diff --git a/contrib/linux-kernel/lib/huf.h b/contrib/linux-kernel/lib/huf.h index 63306b4bf..fd26fcf27 100644 --- a/contrib/linux-kernel/lib/huf.h +++ b/contrib/linux-kernel/lib/huf.h @@ -91,8 +91,6 @@ size_t HUF_compress4X_wksp (void* dst, size_t dstSize, const void* src, size_t s -#ifdef HUF_STATIC_LINKING_ONLY - /* *** Dependencies *** */ #include "mem.h" /* U32 */ @@ -246,6 +244,4 @@ size_t HUF_decompress1X_usingDTable(void* dst, size_t maxDstSize, const void* cS size_t HUF_decompress1X2_usingDTable(void* dst, size_t maxDstSize, const void* cSrc, size_t cSrcSize, const HUF_DTable* DTable); size_t HUF_decompress1X4_usingDTable(void* dst, size_t maxDstSize, const void* cSrc, size_t cSrcSize, const HUF_DTable* DTable); -#endif /* HUF_STATIC_LINKING_ONLY */ - #endif /* HUF_H_298734234 */ diff --git a/contrib/linux-kernel/lib/huf_compress.c b/contrib/linux-kernel/lib/huf_compress.c index af1a1e6c3..16576f579 100644 --- a/contrib/linux-kernel/lib/huf_compress.c +++ b/contrib/linux-kernel/lib/huf_compress.c @@ -38,9 +38,7 @@ ****************************************************************/ #include /* memcpy, memset */ #include "bitstream.h" -#define FSE_STATIC_LINKING_ONLY /* FSE_optimalTableLog_internal */ #include "fse.h" /* header compression */ -#define HUF_STATIC_LINKING_ONLY #include "huf.h" diff --git a/contrib/linux-kernel/lib/huf_decompress.c b/contrib/linux-kernel/lib/huf_decompress.c index 1e19666a7..4ac28bec9 100644 --- a/contrib/linux-kernel/lib/huf_decompress.c +++ b/contrib/linux-kernel/lib/huf_decompress.c @@ -44,7 +44,6 @@ #include /* memcpy, memset */ #include "bitstream.h" /* BIT_* */ #include "fse.h" /* header compression */ -#define HUF_STATIC_LINKING_ONLY #include "huf.h" diff --git a/contrib/linux-kernel/lib/xxhash.c b/contrib/linux-kernel/lib/xxhash.c index 1918836d7..44392265d 100644 --- a/contrib/linux-kernel/lib/xxhash.c +++ b/contrib/linux-kernel/lib/xxhash.c @@ -73,9 +73,6 @@ static void XXH_free (void* p) { free(p); } #include static void* XXH_memcpy(void* dest, const void* src, size_t size) { return memcpy(dest,src,size); } -#ifndef XXH_STATIC_LINKING_ONLY -# define XXH_STATIC_LINKING_ONLY -#endif #include "xxhash.h" #include "mem.h" diff --git a/contrib/linux-kernel/lib/xxhash.h b/contrib/linux-kernel/lib/xxhash.h index b6a56707e..6c4c79c42 100644 --- a/contrib/linux-kernel/lib/xxhash.h +++ b/contrib/linux-kernel/lib/xxhash.h @@ -206,9 +206,6 @@ XXH_PUBLIC_API void XXH64_canonicalFromHash(XXH64_canonical_t* dst, XXH64_hash_t XXH_PUBLIC_API XXH32_hash_t XXH32_hashFromCanonical(const XXH32_canonical_t* src); XXH_PUBLIC_API XXH64_hash_t XXH64_hashFromCanonical(const XXH64_canonical_t* src); -#endif /* XXHASH_H_5627135585666179 */ - - /* ================================================================================================ This section contains definitions which are not guaranteed to remain stable. @@ -216,34 +213,31 @@ XXH_PUBLIC_API XXH64_hash_t XXH64_hashFromCanonical(const XXH64_canonical_t* src They shall only be used with static linking. Never use these definitions in association with dynamic linking ! =================================================================================================== */ -#if defined(XXH_STATIC_LINKING_ONLY) && !defined(XXH_STATIC_H_3543687687345) -#define XXH_STATIC_H_3543687687345 - /* These definitions are only meant to allow allocation of XXH state statically, on stack, or in a struct for example. Do not use members directly. */ - struct XXH32_state_s { - unsigned total_len_32; - unsigned large_len; - unsigned v1; - unsigned v2; - unsigned v3; - unsigned v4; - unsigned mem32[4]; /* buffer defined as U32 for alignment */ - unsigned memsize; - unsigned reserved; /* never read nor write, will be removed in a future version */ - }; /* typedef'd to XXH32_state_t */ +struct XXH32_state_s { + unsigned total_len_32; + unsigned large_len; + unsigned v1; + unsigned v2; + unsigned v3; + unsigned v4; + unsigned mem32[4]; /* buffer defined as U32 for alignment */ + unsigned memsize; + unsigned reserved; /* never read nor write, will be removed in a future version */ +}; /* typedef'd to XXH32_state_t */ - struct XXH64_state_s { - unsigned long long total_len; - unsigned long long v1; - unsigned long long v2; - unsigned long long v3; - unsigned long long v4; - unsigned long long mem64[4]; /* buffer defined as U64 for alignment */ - unsigned memsize; - unsigned reserved[2]; /* never read nor write, will be removed in a future version */ - }; /* typedef'd to XXH64_state_t */ +struct XXH64_state_s { + unsigned long long total_len; + unsigned long long v1; + unsigned long long v2; + unsigned long long v3; + unsigned long long v4; + unsigned long long mem64[4]; /* buffer defined as U64 for alignment */ + unsigned memsize; + unsigned reserved[2]; /* never read nor write, will be removed in a future version */ +}; /* typedef'd to XXH64_state_t */ -#endif /* XXH_STATIC_LINKING_ONLY && XXH_STATIC_H_3543687687345 */ +#endif /* XXHASH_H_5627135585666179 */ diff --git a/contrib/linux-kernel/lib/zstd_common.c b/contrib/linux-kernel/lib/zstd_common.c index 3d24742ee..0788b6ef6 100644 --- a/contrib/linux-kernel/lib/zstd_common.c +++ b/contrib/linux-kernel/lib/zstd_common.c @@ -13,7 +13,6 @@ * Dependencies ***************************************/ #include "error_private.h" -#define ZSTD_STATIC_LINKING_ONLY #include "zstd.h" /* declaration of ZSTD_isError, ZSTD_getErrorName, ZSTD_getErrorCode, ZSTD_getErrorString, ZSTD_versionNumber */ diff --git a/contrib/linux-kernel/lib/zstd_compress.c b/contrib/linux-kernel/lib/zstd_compress.c index d4b9a4751..79b7699b8 100644 --- a/contrib/linux-kernel/lib/zstd_compress.c +++ b/contrib/linux-kernel/lib/zstd_compress.c @@ -13,9 +13,7 @@ ***************************************/ #include /* memset */ #include "mem.h" -#define FSE_STATIC_LINKING_ONLY /* FSE_encodeSymbol */ #include "fse.h" -#define HUF_STATIC_LINKING_ONLY #include "huf.h" #include "zstd_internal.h" /* includes zstd.h */ diff --git a/contrib/linux-kernel/lib/zstd_decompress.c b/contrib/linux-kernel/lib/zstd_decompress.c index bb72d6a4a..10700257b 100644 --- a/contrib/linux-kernel/lib/zstd_decompress.c +++ b/contrib/linux-kernel/lib/zstd_decompress.c @@ -26,9 +26,7 @@ *********************************************************/ #include /* memcpy, memmove, memset */ #include "mem.h" /* low level memory routines */ -#define FSE_STATIC_LINKING_ONLY #include "fse.h" -#define HUF_STATIC_LINKING_ONLY #include "huf.h" #include "zstd_internal.h" diff --git a/contrib/linux-kernel/lib/zstd_internal.h b/contrib/linux-kernel/lib/zstd_internal.h index 16a502a69..57ae77390 100644 --- a/contrib/linux-kernel/lib/zstd_internal.h +++ b/contrib/linux-kernel/lib/zstd_internal.h @@ -22,11 +22,7 @@ ***************************************/ #include "mem.h" #include "error_private.h" -#define ZSTD_STATIC_LINKING_ONLY #include "zstd.h" -#ifndef XXH_STATIC_LINKING_ONLY -# define XXH_STATIC_LINKING_ONLY /* XXH64_state_t */ -#endif #include "xxhash.h" /* XXH_reset, update, digest */ From 458e955c2341f6dd30df626ae2ff2c372d261f9f Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 30 Mar 2017 15:51:58 -0700 Subject: [PATCH 101/305] improved ZSTDMT_compress() Use a bit more threads by default. Uses overlap segments to boost compression ratio (like the streaming variant) --- lib/compress/zstdmt_compress.c | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/lib/compress/zstdmt_compress.c b/lib/compress/zstdmt_compress.c index 90f79509b..f9d722b11 100644 --- a/lib/compress/zstdmt_compress.c +++ b/lib/compress/zstdmt_compress.c @@ -228,7 +228,8 @@ void ZSTDMT_compressChunk(void* jobDescription) ZSTDMT_jobDescription* const job = (ZSTDMT_jobDescription*)jobDescription; const void* const src = (const char*)job->srcStart + job->dictSize; buffer_t const dstBuff = job->dstBuff; - DEBUGLOG(3, "job (first:%u) (last:%u) : dictSize %u, srcSize %u", job->firstChunk, job->lastChunk, (U32)job->dictSize, (U32)job->srcSize); + DEBUGLOG(3, "job (first:%u) (last:%u) : dictSize %u, srcSize %u", + job->firstChunk, job->lastChunk, (U32)job->dictSize, (U32)job->srcSize); if (job->cdict) { /* should only happen for first segment */ size_t const initError = ZSTD_compressBegin_usingCDict(job->cctx, job->cdict, job->fullFrameSize); if (job->cdict) DEBUGLOG(3, "using CDict "); @@ -250,7 +251,8 @@ void ZSTDMT_compressChunk(void* jobDescription) job->cSize = (job->lastChunk) ? ZSTD_compressEnd (job->cctx, dstBuff.start, dstBuff.size, src, job->srcSize) : ZSTD_compressContinue(job->cctx, dstBuff.start, dstBuff.size, src, job->srcSize); - DEBUGLOG(3, "compressed %u bytes into %u bytes (first:%u) (last:%u)", (unsigned)job->srcSize, (unsigned)job->cSize, job->firstChunk, job->lastChunk); + DEBUGLOG(3, "compressed %u bytes into %u bytes (first:%u) (last:%u)", + (unsigned)job->srcSize, (unsigned)job->cSize, job->firstChunk, job->lastChunk); _endJob: PTHREAD_MUTEX_LOCK(job->jobCompleted_mutex); @@ -388,8 +390,9 @@ size_t ZSTDMT_compressCCtx(ZSTDMT_CCtx* mtctx, int compressionLevel) { ZSTD_parameters params = ZSTD_getParams(compressionLevel, srcSize, 0); + size_t const overlapSize = (size_t)1 << (params.cParams.windowLog - 3); size_t const chunkTargetSize = (size_t)1 << (params.cParams.windowLog + 2); - unsigned const nbChunksMax = (unsigned)(srcSize / chunkTargetSize) + (srcSize < chunkTargetSize) /* min 1 */; + unsigned const nbChunksMax = (unsigned)(srcSize / chunkTargetSize) + 1; unsigned nbChunks = MIN(nbChunksMax, mtctx->nbThreads); size_t const proposedChunkSize = (srcSize + (nbChunks-1)) / nbChunks; size_t const avgChunkSize = ((proposedChunkSize & 0x1FFFF) < 0xFFFF) ? proposedChunkSize + 0xFFFF : proposedChunkSize; /* avoid too small last block */ @@ -413,6 +416,7 @@ size_t ZSTDMT_compressCCtx(ZSTDMT_CCtx* mtctx, buffer_t const dstAsBuffer = { dst, dstCapacity }; buffer_t const dstBuffer = u ? ZSTDMT_getBuffer(mtctx->buffPool, dstBufferCapacity) : dstAsBuffer; ZSTD_CCtx* const cctx = ZSTDMT_getCCtx(mtctx->cctxPool); + size_t dictSize = u ? overlapSize : 0; if ((cctx==NULL) || (dstBuffer.start==NULL)) { mtctx->jobs[u].cSize = ERROR(memory_allocation); /* job result */ @@ -421,7 +425,8 @@ size_t ZSTDMT_compressCCtx(ZSTDMT_CCtx* mtctx, break; /* let's wait for previous jobs to complete, but don't start new ones */ } - mtctx->jobs[u].srcStart = srcStart + frameStartPos; + mtctx->jobs[u].srcStart = srcStart + frameStartPos - dictSize; + mtctx->jobs[u].dictSize = dictSize; mtctx->jobs[u].srcSize = chunkSize; mtctx->jobs[u].fullFrameSize = srcSize; mtctx->jobs[u].params = params; From 34cc487d05513fe4981c2b8832cfdd7c9f455041 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 30 Mar 2017 16:23:22 -0700 Subject: [PATCH 102/305] overlap at full windowSize for max compression level as it provides max compression ratio --- lib/compress/zstdmt_compress.c | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/lib/compress/zstdmt_compress.c b/lib/compress/zstdmt_compress.c index f9d722b11..009d04a1f 100644 --- a/lib/compress/zstdmt_compress.c +++ b/lib/compress/zstdmt_compress.c @@ -390,7 +390,8 @@ size_t ZSTDMT_compressCCtx(ZSTDMT_CCtx* mtctx, int compressionLevel) { ZSTD_parameters params = ZSTD_getParams(compressionLevel, srcSize, 0); - size_t const overlapSize = (size_t)1 << (params.cParams.windowLog - 3); + U32 const overlapLog = (compressionLevel >= ZSTD_maxCLevel()) ? 0 : 3; + size_t const overlapSize = (size_t)1 << (params.cParams.windowLog - overlapLog); size_t const chunkTargetSize = (size_t)1 << (params.cParams.windowLog + 2); unsigned const nbChunksMax = (unsigned)(srcSize / chunkTargetSize) + 1; unsigned nbChunks = MIN(nbChunksMax, mtctx->nbThreads); From eea7858e2b17ae2ffa71d2d2e1e708b72d7323c6 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 30 Mar 2017 16:47:19 -0700 Subject: [PATCH 103/305] fixed minor warnings in debug code --- lib/compress/zstdmt_compress.c | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/lib/compress/zstdmt_compress.c b/lib/compress/zstdmt_compress.c index 009d04a1f..09a965478 100644 --- a/lib/compress/zstdmt_compress.c +++ b/lib/compress/zstdmt_compress.c @@ -44,26 +44,26 @@ DEBUGLOGRAW(l, " \n"); \ } -static unsigned long long GetCurrentClockTimeMicroseconds() +static unsigned long long GetCurrentClockTimeMicroseconds(void) { static clock_t _ticksPerSecond = 0; if (_ticksPerSecond <= 0) _ticksPerSecond = sysconf(_SC_CLK_TCK); - struct tms junk; clock_t newTicks = (clock_t) times(&junk); - return ((((unsigned long long)newTicks)*(1000000))/_ticksPerSecond); + { struct tms junk; clock_t newTicks = (clock_t) times(&junk); + return ((((unsigned long long)newTicks)*(1000000))/_ticksPerSecond); } } #define MUTEX_WAIT_TIME_DLEVEL 5 #define PTHREAD_MUTEX_LOCK(mutex) \ if (g_debugLevel>=MUTEX_WAIT_TIME_DLEVEL) { \ - unsigned long long beforeTime = GetCurrentClockTimeMicroseconds(); \ - pthread_mutex_lock(mutex); \ - unsigned long long afterTime = GetCurrentClockTimeMicroseconds(); \ - unsigned long long elapsedTime = (afterTime-beforeTime); \ - if (elapsedTime > 1000) { /* or whatever threshold you like; I'm using 1 millisecond here */ \ - DEBUGLOG(MUTEX_WAIT_TIME_DLEVEL, "Thread took %llu microseconds to acquire mutex %s \n", \ + unsigned long long const beforeTime = GetCurrentClockTimeMicroseconds(); \ + pthread_mutex_lock(mutex); \ + { unsigned long long const afterTime = GetCurrentClockTimeMicroseconds(); \ + unsigned long long const elapsedTime = (afterTime-beforeTime); \ + if (elapsedTime > 1000) { /* or whatever threshold you like; I'm using 1 millisecond here */ \ + DEBUGLOG(MUTEX_WAIT_TIME_DLEVEL, "Thread took %llu microseconds to acquire mutex %s \n", \ elapsedTime, #mutex); \ - } \ + } } \ } else pthread_mutex_lock(mutex); #else From 96fe545a18b4f69cae28749bfcc304c54705959a Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Fri, 31 Mar 2017 15:16:43 -0700 Subject: [PATCH 104/305] Turn off sparse mode for OS X by default --- programs/fileio.c | 3 +++ programs/platform.h | 8 ++++++++ programs/zstdcli.c | 4 ++++ 3 files changed, 15 insertions(+) diff --git a/programs/fileio.c b/programs/fileio.c index 572e9c9ac..87871ae01 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -227,6 +227,9 @@ static FILE* FIO_openDstFile(const char* dstFileName) DISPLAYLEVEL(4, "Sparse File Support is automatically disabled on stdout ; try --sparse \n"); } } else { + if (ZSTD_SPARSE_DEFAULT == 0 && g_sparseFileSupport == 1) { + g_sparseFileSupport = 0; + } if (!g_overwrite && strcmp (dstFileName, nulmark)) { /* Check if destination file already exists */ f = fopen( dstFileName, "rb" ); if (f != 0) { /* dest file exists, prompt for overwrite authorization */ diff --git a/programs/platform.h b/programs/platform.h index ee7819512..74412cde3 100644 --- a/programs/platform.h +++ b/programs/platform.h @@ -138,6 +138,14 @@ static __inline int IS_CONSOLE(FILE* stdStream) #endif +#ifndef ZSTD_SPARSE_DEFAULT +# if (defined(__APPLE__) && defined(__MACH__)) +# define ZSTD_SPARSE_DEFAULT 0 +# else +# define ZSTD_SPARSE_DEFAULT 1 +# endif +#endif + #if defined (__cplusplus) } diff --git a/programs/zstdcli.c b/programs/zstdcli.c index e2709ec84..0cf8cacd1 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -137,7 +137,11 @@ static int usage_advanced(const char* programName) #endif #ifndef ZSTD_NODECOMPRESS DISPLAY( "--test : test compressed file integrity \n"); +#if ZSTD_SPARSE_DEFAULT DISPLAY( "--[no-]sparse : sparse mode (default:enabled on file, disabled on stdout)\n"); +#else + DISPLAY( "--[no-]sparse : sparse mode (default:disabled)\n"); +#endif #endif DISPLAY( " -M# : Set a memory usage limit for decompression \n"); DISPLAY( "-- : All arguments after \"--\" are treated as files \n"); From eb38617175351bfaf933bc75a6604e070128ca79 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Fri, 31 Mar 2017 15:20:50 -0700 Subject: [PATCH 105/305] Clean up default sparse logic --- programs/fileio.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/programs/fileio.c b/programs/fileio.c index 87871ae01..78fb9a268 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -227,8 +227,8 @@ static FILE* FIO_openDstFile(const char* dstFileName) DISPLAYLEVEL(4, "Sparse File Support is automatically disabled on stdout ; try --sparse \n"); } } else { - if (ZSTD_SPARSE_DEFAULT == 0 && g_sparseFileSupport == 1) { - g_sparseFileSupport = 0; + if (g_sparseFileSupport == 1) { + g_sparseFileSupport = ZSTD_SPARSE_DEFAULT; } if (!g_overwrite && strcmp (dstFileName, nulmark)) { /* Check if destination file already exists */ f = fopen( dstFileName, "rb" ); From 14433ca1adfa9bb17e74fcbd7942cc6d6a91c80a Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 31 Mar 2017 10:54:45 -0700 Subject: [PATCH 106/305] numerous typos and clarifications in format specification fix limit values of Window_Size bump version to 0.2.5 --- doc/zstd_compression_format.md | 372 +++++++++++++++++---------------- 1 file changed, 197 insertions(+), 175 deletions(-) diff --git a/doc/zstd_compression_format.md b/doc/zstd_compression_format.md index d4b46548a..4bbdd57bc 100644 --- a/doc/zstd_compression_format.md +++ b/doc/zstd_compression_format.md @@ -16,7 +16,8 @@ Distribution of this document is unlimited. ### Version -0.2.4 (17/02/17) +0.2.5 (31/03/17) + Introduction ------------ @@ -109,7 +110,7 @@ The structure of a single Zstandard frame is following: __`Magic_Number`__ -4 Bytes, little-endian format. +4 Bytes, __little-endian__ format. Value : 0xFD2FB528 __`Frame_Header`__ @@ -127,7 +128,7 @@ An optional 32-bit checksum, only present if `Content_Checksum_flag` is set. The content checksum is the result of [xxh64() hash function](http://www.xxhash.org) digesting the original (decoded) data as input, and a seed of zero. -The low 4 bytes of the checksum are stored in little endian format. +The low 4 bytes of the checksum are stored in __little-endian__ format. ### `Frame_Header` @@ -154,41 +155,42 @@ Decoding this byte is enough to tell the size of `Frame_Header`. | 2 | `Content_Checksum_flag` | | 1-0 | `Dictionary_ID_flag` | -In this table, bit 7 the is highest bit, while bit 0 the is lowest. +In this table, bit 7 is the highest bit, while bit 0 is the lowest one. __`Frame_Content_Size_flag`__ This is a 2-bits flag (`= Frame_Header_Descriptor >> 6`), -specifying if decompressed data size is provided within the header. -The `Flag_Value` can be converted into `Field_Size`, +specifying if `Frame_Content_Size` (the decompressed data size) +is provided within the header. +`Flag_Value` provides `FCS_Field_Size`, which is the number of bytes used by `Frame_Content_Size` according to the following table: -|`Flag_Value`| 0 | 1 | 2 | 3 | -| ---------- | ------ | --- | --- | --- | -|`Field_Size`| 0 or 1 | 2 | 4 | 8 | +| `Flag_Value` | 0 | 1 | 2 | 3 | +| -------------- | ------ | --- | --- | --- | +|`FCS_Field_Size`| 0 or 1 | 2 | 4 | 8 | -When `Flag_Value` is `0`, `Field_Size` depends on `Single_Segment_flag` : +When `Flag_Value` is `0`, `FCS_Field_Size` depends on `Single_Segment_flag` : if `Single_Segment_flag` is set, `Field_Size` is 1. -Otherwise, `Field_Size` is 0 (content size not provided). +Otherwise, `Field_Size` is 0 : `Frame_Content_Size` is not provided. __`Single_Segment_flag`__ If this flag is set, data must be regenerated within a single continuous memory segment. -In this case, `Frame_Content_Size` is necessarily present, -but `Window_Descriptor` byte is skipped. +In this case, `Window_Descriptor` byte is skipped, +but `Frame_Content_Size` is necessarily present. As a consequence, the decoder must allocate a memory segment of size equal or bigger than `Frame_Content_Size`. In order to preserve the decoder from unreasonable memory requirements, -a decoder can reject a compressed frame +a decoder is allowed to reject a compressed frame which requests a memory size beyond decoder's authorized range. For broader compatibility, decoders are recommended to support memory sizes of at least 8 MB. -This is just a recommendation, +This is only a recommendation, each decoder is free to support higher or lower limits, depending on local limitations. @@ -224,37 +226,38 @@ It also specifies the size of this field as `Field_Size`. #### `Window_Descriptor` -Provides guarantees on maximum back-reference distance -that will be used within compressed data. +Provides guarantees on minimum memory buffer required to decompress a frame. This information is important for decoders to allocate enough memory. -The `Window_Descriptor` byte is optional. It is absent when `Single_Segment_flag` is set. -In this case, the maximum back-reference distance is the content size itself, -which can be any value from 1 to 2^64-1 bytes (16 EB). +The `Window_Descriptor` byte is optional. +When `Single_Segment_flag` is set, `Window_Descriptor` is not present. +In this case, the required buffer size is the frame content size itself, +which can be any value from 0 to 2^64-1 bytes (16 EB). | Bit numbers | 7-3 | 2-0 | | ----------- | ---------- | ---------- | | Field name | `Exponent` | `Mantissa` | -Maximum distance is given by the following formulas : +The minimum memory buffer size is called `Window_Size`. +It is described by the following formulas : ``` windowLog = 10 + Exponent; windowBase = 1 << windowLog; windowAdd = (windowBase / 8) * Mantissa; Window_Size = windowBase + windowAdd; ``` -The minimum window size is 1 KB. -The maximum size is `15*(1<<38)` bytes, which is 1.875 TB. +The minimum `Window_Size` is 1 KB. +The maximum `Window_Size` is `(1<<41) + 7*(1<<38)` bytes, which is 3.75 TB. To properly decode compressed data, a decoder will need to allocate a buffer of at least `Window_Size` bytes. In order to preserve decoder from unreasonable memory requirements, -a decoder can refuse a compressed frame +a decoder is allowed to reject a compressed frame which requests a memory size beyond decoder's authorized range. For improved interoperability, -decoders are recommended to be compatible with window sizes of 8 MB, +decoders are recommended to be compatible with `Window_Size >= 8 MB`, and encoders are recommended to not request more than 8 MB. It's merely a recommendation though, decoders are free to support larger or lower limits, @@ -264,47 +267,50 @@ depending on local limitations. This is a variable size field, which contains the ID of the dictionary required to properly decode the frame. -Note that this field is optional. When it's not present, +`Dictionary_ID` field is optional. When it's not present, it's up to the decoder to make sure it uses the correct dictionary. -Format is little-endian. Field size depends on `Dictionary_ID_flag`. 1 byte can represent an ID 0-255. 2 bytes can represent an ID 0-65535. 4 bytes can represent an ID 0-4294967295. +Format is __little-endian__. It's allowed to represent a small ID (for example `13`) -with a large 4-bytes dictionary ID, losing some compacity in the process. +with a large 4-bytes dictionary ID, even if it is less efficient. _Reserved ranges :_ If the frame is going to be distributed in a private environment, any dictionary ID can be used. However, for public distribution of compressed frames using a dictionary, -the following ranges are reserved for future use and should not be used : -- low range : 1 - 32767 -- high range : >= (2^31) - +the following ranges are reserved and shall not be used : +- low range : `<= 32767` +- high range : `>= (1 << 31)` #### `Frame_Content_Size` This is the original (uncompressed) size. This information is optional. -The `Field_Size` is provided according to value of `Frame_Content_Size_flag`. -The `Field_Size` can be equal to 0 (not present), 1, 2, 4 or 8 bytes. -Format is little-endian. +`Frame_Content_Size` uses a variable number of bytes, provided by `FCS_Field_Size`. +`FCS_Field_Size` is provided by the value of `Frame_Content_Size_flag`. +`FCS_Field_Size` can be equal to 0 (not present), 1, 2, 4 or 8 bytes. -| `Field_Size` | Range | -| ------------ | ---------- | -| 1 | 0 - 255 | -| 2 | 256 - 65791| -| 4 | 0 - 2^32-1 | -| 8 | 0 - 2^64-1 | +| `FCS_Field_Size` | Range | +| ---------------- | ---------- | +| 0 | unknown | +| 1 | 0 - 255 | +| 2 | 256 - 65791| +| 4 | 0 - 2^32-1 | +| 8 | 0 - 2^64-1 | -When `Field_Size` is 1, 4 or 8 bytes, the value is read directly. -When `Field_Size` is 2, _the offset of 256 is added_. +`Frame_Content_Size` format is __little-endian__. +When `FCS_Field_Size` is 1, 4 or 8 bytes, the value is read directly. +When `FCS_Field_Size` is 2, _the offset of 256 is added_. It's allowed to represent a small size (for example `18`) using any compatible variant. + Blocks ------- + After the magic number and header of each block, there are some number of blocks. Each frame must have at least one block but there is no upper limit @@ -312,64 +318,68 @@ on the number of blocks per frame. The structure of a block is as follows: -| `Last_Block` | `Block_Type` | `Block_Size` | `Block_Content` | -|:------------:|:------------:|:------------:|:---------------:| -| 1 bit | 2 bits | 21 bits | n bytes | +| `Block_Header` | `Block_Content` | +|:--------------:|:---------------:| +| 3 bytes | n bytes | -The block header (`Last_Block`, `Block_Type`, and `Block_Size`) uses 3-bytes. +`Block_Header` uses 3 bytes, written using __little-endian__ convention. +It contains 3 fields : + +| `Block_Size` | `Block_Type` | `Last_Block` | +|:------------:|:------------:|:------------:| +| bits 3-23 | bits 1-2 | bit 0 | __`Last_Block`__ The lowest bit signals if this block is the last one. -The frame will end after this one. +The frame will end after this last block. It may be followed by an optional `Content_Checksum` (see [Zstandard Frames](#zstandard-frames)). -__`Block_Type` and `Block_Size`__ - -The next 2 bits represent the `Block_Type`, -while the remaining 21 bits represent the `Block_Size`. -Format is __little-endian__. +__`Block_Type`__ +The next 2 bits represent the `Block_Type`. There are 4 block types : -| Value | 0 | 1 | 2 | 3 | +| Value | 0 | 1 | 2 | 3 | | ------------ | ----------- | ----------- | ------------------ | --------- | | `Block_Type` | `Raw_Block` | `RLE_Block` | `Compressed_Block` | `Reserved`| - `Raw_Block` - this is an uncompressed block. - `Block_Content` contains `Block_Size` bytes to read and copy - as decoded data. + `Block_Content` contains `Block_Size` bytes. -- `RLE_Block` - this is a single byte, repeated N times. - `Block_Content` consists of a single byte, - and `Block_Size` is the number of times this byte should be repeated. +- `RLE_Block` - this is a single byte, repeated `Block_Size` times. + `Block_Content` consists of a single byte. + On the decompression side, this byte must be repeated `Block_Size` times. - `Compressed_Block` - this is a [Zstandard compressed block](#compressed-blocks), explained later on. `Block_Size` is the length of `Block_Content`, the compressed data. - The decompressed size is unknown, + The decompressed size is not known, but its maximum possible value is guaranteed (see below) - `Reserved` - this is not a block. This value cannot be used with current version of this specification. +__`Block_Size`__ + +The upper 21 bits of `Block_Header` represent the `Block_Size`. + Block sizes must respect a few rules : -- In compressed mode, compressed size is always strictly less than decompressed size. -- Block decompressed size is always <= maximum back-reference distance. +- For `Compressed_Block`, `Block_Size` is always strictly less than decompressed size. +- Block decompressed size is always <= `Window_Size` - Block decompressed size is always <= 128 KB. -A data block is not necessarily "full" : -since an arbitrary “flush” may happen anytime, -block decompressed content can be any size (even empty), +A block can contain any number of bytes (even empty), up to `Block_Maximum_Decompressed_Size`, which is the smallest of : -- Maximum back-reference distance +- `Window_Size` - 128 KB + Compressed Blocks ----------------- -To decompress a compressed block, the compressed size must be provided from -`Block_Size` field in the block header. +To decompress a compressed block, the compressed size must be provided +from `Block_Size` field within `Block_Header`. A compressed block consists of 2 sections : - [Literals Section](#literals-section) @@ -381,36 +391,34 @@ data in [Sequence Execution](#sequence-execution) #### Prerequisites To decode a compressed block, the following elements are necessary : - Previous decoded data, up to a distance of `Window_Size`, - or all previous data when `Single_Segment_flag` is set. -- List of "recent offsets" from the previous compressed block. -- Decoding tables of the previous compressed block for each symbol type + or all previously decoded data when `Single_Segment_flag` is set. +- List of "recent offsets" from previous `Compressed_Block`. +- Decoding tables of previous `Compressed_Block` for each symbol type (literals, literals lengths, match lengths, offsets). Literals Section ---------------- -During sequence execution, symbols from the literals section -During sequence phase, literals will be entangled with match copy operations. All literals are regrouped in the first part of the block. -They can be decoded first, and then copied during sequence operations, -or they can be decoded on the flow, as needed by sequence commands. - -| `Literals_Section_Header` | [`Huffman_Tree_Description`] | Stream1 | [Stream2] | [Stream3] | [Stream4] | -| ------------------------- | ---------------------------- | ------- | --------- | --------- | --------- | +They can be decoded first, and then copied during [Sequence Execution], +or they can be decoded on the flow during [Sequence Execution]. Literals can be stored uncompressed or compressed using Huffman prefix codes. When compressed, an optional tree description can be present, followed by 1 or 4 streams. +| `Literals_Section_Header` | [`Huffman_Tree_Description`] | Stream1 | [Stream2] | [Stream3] | [Stream4] | +| ------------------------- | ---------------------------- | ------- | --------- | --------- | --------- | + #### `Literals_Section_Header` Header is in charge of describing how literals are packed. It's a byte-aligned variable-size bitfield, ranging from 1 to 5 bytes, -using little-endian convention. +using __little-endian__ convention. | `Literals_Block_Type` | `Size_Format` | `Regenerated_Size` | [`Compressed_Size`] | -| --------------------- | ------------- | ------------------ | ----------------- | -| 2 bits | 1 - 2 bits | 5 - 20 bits | 0 - 18 bits | +| --------------------- | ------------- | ------------------ | ------------------- | +| 2 bits | 1 - 2 bits | 5 - 20 bits | 0 - 18 bits | In this representation, bits on the left are the lowest bits. @@ -418,33 +426,38 @@ __`Literals_Block_Type`__ This field uses 2 lowest bits of first byte, describing 4 different block types : -| `Literals_Block_Type` | Value | -| ----------------------------- | ----- | -| `Raw_Literals_Block` | 0 | -| `RLE_Literals_Block` | 1 | -| `Compressed_Literals_Block` | 2 | -| `Repeat_Stats_Literals_Block` | 3 | +| `Literals_Block_Type` | Value | +| --------------------------- | ----- | +| `Raw_Literals_Block` | 0 | +| `RLE_Literals_Block` | 1 | +| `Compressed_Literals_Block` | 2 | +| `Treeless_Literals_Block` | 3 | - `Raw_Literals_Block` - Literals are stored uncompressed. -- `RLE_Literals_Block` - Literals consist of a single byte value repeated N times. +- `RLE_Literals_Block` - Literals consist of a single byte value + repeated `Regenerated_Size` times. - `Compressed_Literals_Block` - This is a standard Huffman-compressed block, starting with a Huffman tree description. See details below. -- `Repeat_Stats_Literals_Block` - This is a Huffman-compressed block, +- `Treeless_Literals_Block` - This is a Huffman-compressed block, using Huffman tree _from previous Huffman-compressed literals block_. - Huffman tree description will be skipped. - Note: If this mode is used without any previous Huffman-table in the frame - (or [dictionary](#dictionary-format)), this should be treated as corruption. + `Huffman_Tree_Description` will be skipped. + Note: If this mode is triggering without any previous Huffman-table in the frame + (or [dictionary](#dictionary-format)), this should be treated as data corruption. __`Size_Format`__ `Size_Format` is divided into 2 families : -- For `Raw_Literals_Block` and `RLE_Literals_Block` it's enough to decode `Regenerated_Size`. -- For `Compressed_Block`, its required to decode both `Compressed_Size` - and `Regenerated_Size` (the decompressed size). It will also decode the number of streams. +- For `Raw_Literals_Block` and `RLE_Literals_Block`, + it's only necessary to decode `Regenerated_Size`. + There is no `Compressed_Size` field. +- For `Compressed_Block` and `Treeless_Literals_Block`, + it's required to decode both `Compressed_Size` + and `Regenerated_Size` (the decompressed size). + It's also necessary to decode the number of streams (1 or 4). -For values spanning several bytes, convention is little-endian. +For values spanning several bytes, convention is __little-endian__. __`Size_Format` for `Raw_Literals_Block` and `RLE_Literals_Block`__ : @@ -463,9 +476,9 @@ __`Size_Format` for `Raw_Literals_Block` and `RLE_Literals_Block`__ : Only Stream1 is present for these cases. Note : it's allowed to represent a short value (for example `13`) -using a long format, accepting the increased compressed data size. +using a long format, even if it's less efficient. -__`Size_Format` for `Compressed_Literals_Block` and `Repeat_Stats_Literals_Block`__ : +__`Size_Format` for `Compressed_Literals_Block` and `Treeless_Literals_Block`__ : - Value 00 : _A single stream_. Both `Regenerated_Size` and `Compressed_Size` use 10 bits (0-1023). @@ -480,63 +493,64 @@ __`Size_Format` for `Compressed_Literals_Block` and `Repeat_Stats_Literals_Block Both `Regenerated_Size` and `Compressed_Size` use 18 bits (0-262143). `Literals_Section_Header` has 5 bytes. -Both `Compressed_Size` and `Regenerated_Size` fields follow little-endian convention. -Note: `Compressed_Size` __includes__ the size of the Huffman Tree description if it -is present. +Both `Compressed_Size` and `Regenerated_Size` fields follow __little-endian__ convention. +Note: `Compressed_Size` __includes__ the size of the Huffman Tree description +_when_ it is present. ### Raw Literals Block -The data in Stream1 is `Regenerated_Size` bytes long, and contains the raw literals data -to be used in sequence execution. +The data in Stream1 is `Regenerated_Size` bytes long, +it contains the raw literals data to be used during [Sequence Execution]. ### RLE Literals Block Stream1 consists of a single byte which should be repeated `Regenerated_Size` times to generate the decoded literals. -### Compressed Literals Block and Repeat Stats Literals Block -Both of these modes contain Huffman encoded data +### Compressed Literals Block and Treeless Literals Block +Both of these modes contain Huffman encoded data. +`Treeless_Literals_Block` does not have a `Huffman_Tree_Description`. #### `Huffman_Tree_Description` This section is only present when `Literals_Block_Type` type is `Compressed_Literals_Block` (`2`). The format of the Huffman tree description can be found at [Huffman Tree description](#huffman-tree-description). -The size Huffman Tree description will be determined during the decoding process, -and must be used to determine where the compressed Huffman streams begin. +The size of `Huffman_Tree_Description` is determined during decoding process, +it must be used to determine where streams begin. +`Total_Streams_Size = Compress_Size - Huffman_Tree_Description_Size`. -If repeat stats mode is used, the Huffman table used in the previous compressed block will -be used to decompress this block as well. +For `Treeless_Literals_Block`, +the Huffman table comes from previously compressed literals block. -Huffman compressed data consists either 1 or 4 Huffman-coded streams. +Huffman compressed data consists of either 1 or 4 Huffman-coded streams. If only one stream is present, it is a single bitstream occupying the entire -remaining portion of the literals block, encoded as described at +remaining portion of the literals block, encoded as described within [Huffman-Coded Streams](#huffman-coded-streams). If there are four streams, the literals section header only provides enough -information to know the regenerated and compressed sizes of all four streams combined. -The regenerated size of each stream is equal to `(totalSize+3)/4`, except for the last stream, -which may be up to 3 bytes smaller, to reach a total decompressed size match that described -in the literals header. +information to know the decompressed and compressed sizes of all four streams _combined_. +The decompressed size of each stream is equal to `(totalSize+3)/4`, +except for the last stream which may be up to 3 bytes smaller, +to reach a total decompressed size as specified in `Regenerated_Size`. -The compressed size of each stream is provided explicitly: the first 6 bytes of the compressed -data consist of three 2-byte little endian fields, describing the compressed sizes -of the first three streams. -The last streams size is computed from the total compressed size and the size of the other -three streams. +The compressed size of each stream is provided explicitly: +the first 6 bytes of the compressed data consist of three 2-byte __little-endian__ fields, +describing the compressed sizes of the first three streams. +Stream4 size is computed from total `Total_Streams_Size` minus sizes of other streams. -`stream4CSize = totalCSize - 6 - stream1CSize - stream2CSize - stream3CSize`. +`Stream4_Size = Total_Streams_Size - 6 - Stream1_Size - Stream2_Size - Stream3_Size`. -Note: remember that totalCSize may be smaller than the `Compressed_Size` found in the literals -block header as `Compressed_Size` also contains the size of the Huffman Tree description if it -is present. +Note: remember that `Total_Streams_Size` can be smaller than `Compressed_Size` in header, +because `Compressed_Size` also contains `Huffman_Tree_Description_Size` when it is present. Each of these 4 bitstreams is then decoded independently as a Huffman-Coded stream, as described at [Huffman-Coded Streams](#huffman-coded-streams) + Sequences Section ----------------- A compressed block is a succession of _sequences_ . A sequence is a literal copy command, followed by a match copy command. A literal copy command specifies a length. -It is the number of bytes to be copied (or extracted) from the literal section. +It is the number of bytes to be copied (or extracted) from the [Literals Section]. A match copy command specifies an offset and a length. When all _sequences_ are decoded, @@ -557,7 +571,7 @@ followed by the bitstream. | -------------------------- | ------------------------- | ---------------- | ---------------------- | --------- | To decode the `Sequences_Section`, it's required to know its size. -This size is deduced from `blockSize - literalSectionSize`. +This size is deduced from `Block_Size - Literals_Section_Size`. #### `Sequences_Section_Header` @@ -572,7 +586,7 @@ This is a variable size field using between 1 and 3 bytes. Let's call its first byte `byte0`. - `if (byte0 == 0)` : there are no sequences. The sequence section stops there. - Regenerated content is defined entirely by literals section. + Decompressed content is defined entirely as [Literals Section] content. - `if (byte0 < 128)` : `Number_of_Sequences = byte0` . Uses 1 byte. - `if (byte0 < 255)` : `Number_of_Sequences = ((byte0-128) << 8) + byte1` . Uses 2 bytes. - `if (byte0 == 255)`: `Number_of_Sequences = byte1 + (byte2<<8) + 0x7F00` . Uses 3 bytes. @@ -581,14 +595,14 @@ __Symbol compression modes__ This is a single byte, defining the compression mode of each symbol type. -|Bit number| 7-6 | 5-4 | 3-2 | 1-0 | +|Bit number| 7-6 | 5-4 | 3-2 | 1-0 | | -------- | ----------------------- | -------------- | -------------------- | ---------- | |Field name| `Literals_Lengths_Mode` | `Offsets_Mode` | `Match_Lengths_Mode` | `Reserved` | The last field, `Reserved`, must be all-zeroes. `Literals_Lengths_Mode`, `Offsets_Mode` and `Match_Lengths_Mode` define the `Compression_Mode` of -literals lengths, offsets, and match lengths respectively. +literals lengths, offsets, and match lengths symbols respectively. They follow the same enumeration : @@ -598,12 +612,12 @@ They follow the same enumeration : - `Predefined_Mode` : A predefined FSE distribution table is used, defined in [default distributions](#default-distributions). - The table takes no space in the compressed data. + No distribution table will be present. - `RLE_Mode` : The table description consists of a single byte. - This code will be repeated for every sequence. + This code will be repeated for all sequences. - `Repeat_Mode` : The table used in the previous compressed block will be used again. No distribution table will be present. - Note: this includes RLE mode, so if repeat_mode follows rle_mode the same symbol will be repeated. + Note: this includes RLE mode, so if `Repeat_Mode` follows `RLE_Mode`, the same symbol will be repeated. If this mode is used without any previous sequence table in the frame (or [dictionary](#dictionary-format)) to repeat, this should be treated as corruption. - `FSE_Compressed_Mode` : standard FSE compression. @@ -625,7 +639,7 @@ Literals length codes are values ranging from `0` to `35` included. They define lengths from 0 to 131071 bytes. The literals length is equal to the decoded `Baseline` plus the result of reading `Number_of_Bits` bits from the bitstream, -as a little-endian value. +as a __little-endian__ value. | `Literals_Length_Code` | 0-15 | | ---------------------- | ---------------------- | @@ -654,7 +668,7 @@ Match length codes are values ranging from `0` to `52` included. They define lengths from 3 to 131074 bytes. The match length is equal to the decoded `Baseline` plus the result of reading `Number_of_Bits` bits from the bitstream, -as a little-endian value. +as a __little-endian__ value. | `Match_Length_Code` | 0-31 | | ------------------- | ----------------------- | @@ -685,7 +699,7 @@ Recommendation is to support at least up to `22`. For information, at the time of this writing. the reference decoder supports a maximum `N` value of `28` in 64-bits mode. -An offset code is also the number of additional bits to read in little-endian fashion, +An offset code is also the number of additional bits to read in __little-endian__ fashion, and can be translated into an `Offset_Value` using the following formulas : ``` @@ -720,8 +734,8 @@ begins. FSE decoding requires a 'state' to be carried from symbol to symbol. For more explanation on FSE decoding, see the [FSE section](#fse). -For sequence decoding, a separate state must be kept track of for each of -literal lengths, offsets, and match lengths. +For sequence decoding, a separate state keeps track of each +literal lengths, offsets, and match lengths symbols. Some FSE primitives are also used. For more details on the operation of these primitives, see the [FSE section](#fse). @@ -753,8 +767,7 @@ See the [description of the codes] for how to determine these values. [description of the codes]: #the-codes-for-literals-lengths-match-lengths-and-offsets Decoding starts by reading the `Number_of_Bits` required to decode `Offset`. -It then does the same for `Match_Length`, -and then for `Literals_Length`. +It then does the same for `Match_Length`, and then for `Literals_Length`. This sequence is then used for [sequence execution](#sequence-execution). If it is not the last sequence in the block, @@ -807,6 +820,7 @@ short offsetCodes_defaultDistribution[29] = 1, 1, 1, 1, 1, 1, 1, 1,-1,-1,-1,-1,-1 }; ``` + Sequence Execution ------------------ Once literals and sequences have been decoded, @@ -826,7 +840,8 @@ in this case. The offset is defined as from the current position, so an offset of 6 and a match length of 3 means that 3 bytes should be copied from 6 bytes back. -Note that all offsets must be at most equal to the window size defined by the frame header. +Note that all offsets leading to previously decoded data +must be smaller than `Window_Size` defined in `Frame_Header_Descriptor`. #### Repeat offsets As seen in [Sequence Execution](#sequence-execution), @@ -842,11 +857,10 @@ so an `offset_value` of 1 means `Repeated_Offset2`, an `offset_value` of 2 means `Repeated_Offset3`, and an `offset_value` of 3 means `Repeated_Offset1 - 1_byte`. -In the first block, the offset history is populated with the following values : 1, 4 and 8 (in order). +For the first block, the starting offset history is populated with the following values : 1, 4 and 8 (in order). -Then each block gets its starting offset history from the ending values of the most recent compressed block. -Note that non-compressed blocks are skipped, -they do not contribute to offset history. +Then each block gets its starting offset history from the ending values of the most recent `Compressed_Block`. +Note that blocks which are not `Compressed_Block` are skipped, they do not contribute to offset history. [Offset Codes]: #offset-codes @@ -859,6 +873,7 @@ This means that when `Repeated_Offset1` (most recent) is used, history is unmodi When `Repeated_Offset2` is used, it's swapped with `Repeated_Offset1`. If any other offset is used, it becomes `Repeated_Offset1` and the rest are shift back by one. + Skippable Frames ---------------- @@ -878,7 +893,7 @@ Skippable frames defined in this specification are compatible with [LZ4] ones. __`Magic_Number`__ -4 Bytes, little-endian format. +4 Bytes, __little-endian__ format. Value : 0x184D2A5?, which means any value from 0x184D2A50 to 0x184D2A5F. All 16 values are valid to identify a skippable frame. @@ -886,13 +901,14 @@ __`Frame_Size`__ This is the size, in bytes, of the following `User_Data` (without including the magic number nor the size field itself). -This field is represented using 4 Bytes, little-endian format, unsigned 32-bits. +This field is represented using 4 Bytes, __little-endian__ format, unsigned 32-bits. This means `User_Data` can’t be bigger than (2^32-1) bytes. __`User_Data`__ The `User_Data` can be anything. Data will just be skipped by the decoder. + Entropy Encoding ---------------- Two types of entropy encoding are used by the Zstandard format: @@ -900,7 +916,7 @@ FSE, and Huffman coding. FSE --- -FSE, or FiniteStateEntropy is an entropy coding based on [ANS]. +FSE, short for Finite State Entropy, is an entropy codec based on [ANS]. FSE encoding/decoding involves a state that is carried over between symbols, so decoding must be done in the opposite direction as encoding. Therefore, all FSE bitstreams are read from end to beginning. @@ -909,15 +925,15 @@ For additional details on FSE, see [Finite State Entropy]. [Finite State Entropy]:https://github.com/Cyan4973/FiniteStateEntropy/ -FSE decoding involves a decoding table which has a power of 2 size and three elements: +FSE decoding involves a decoding table which has a power of 2 size, and contain three elements: `Symbol`, `Num_Bits`, and `Baseline`. The `log2` of the table size is its `Accuracy_Log`. The FSE state represents an index in this table. -The next symbol in the stream is the symbol indicated by the table value for that state. -To obtain the next state value, -the decoder should consume `Num_Bits` bits from the stream as a little endian value and add it to baseline. -To obtain the initial state value, consume `Accuracy_Log` bits from the stream as a little endian value. +To obtain the initial state value, consume `Accuracy_Log` bits from the stream as a __little-endian__ value. +The next symbol in the stream is the `Symbol` indicated in the table for that state. +To obtain the next state value, +the decoder should consume `Num_Bits` bits from the stream as a __little-endian__ value and add it to `Baseline`. [ANS]: https://en.wikipedia.org/wiki/Asymmetric_Numeral_Systems @@ -929,7 +945,7 @@ An FSE distribution table describes the probabilities of all symbols from `0` to the last present one (included) on a normalized scale of `1 << Accuracy_Log` . -It's a bitstream which is read forward, in little-endian fashion. +It's a bitstream which is read forward, in __little-endian__ fashion. It's not necessary to know its exact size, since it will be discovered and reported by the decoding process. @@ -1064,7 +1080,7 @@ Huffman Coding -------------- Zstandard Huffman-coded streams are read backwards, similar to the FSE bitstreams. -Therefore, to find the start of the bitstream it is therefore necessary to +Therefore, to find the start of the bitstream, it is therefore to know the offset of the last byte of the Huffman-coded stream. After writing the last bit containing information, the compressor @@ -1077,7 +1093,7 @@ byte to read. The decompressor needs to skip 0-7 initial `0`-bits and the first `1`-bit it occurs. Afterwards, the useful part of the bitstream begins. -The bitstream contains Huffman-coded symbols in little-endian order, +The bitstream contains Huffman-coded symbols in __little-endian__ order, with the codes defined by the method below. ### Huffman Tree Description @@ -1182,14 +1198,14 @@ The Huffman header compression uses 2 states, which share the same FSE distribution table. The first state (`State1`) encodes the even indexed symbols, and the second (`State2`) encodes the odd indexes. -State1 is initialized first, and then State2, and they take turns decoding -a single symbol and updating their state. +`State1` is initialized first, and then `State2`, and they take turns +decoding a single symbol and updating their state. For more details on these FSE operations, see the [FSE section](#fse). The number of symbols to decode is determined by tracking bitStream overflow condition: If updating state after decoding a symbol would require more bits than -remain in the stream, it is assumed the extra bits are 0. Then, +remain in the stream, it is assumed that extra bits are 0. Then, the symbols for each of the final states are decoded and the process is complete. ##### Conversion from weights to Huffman prefix codes @@ -1245,7 +1261,7 @@ it would be encoded as: |Encoding|`0000`|`0001`|`01`|`1`| `10000` | Starting from the end, -it's possible to read the bitstream in a little-endian fashion, +it's possible to read the bitstream in a __little-endian__ fashion, keeping track of already used bits. Since the bitstream is encoded in reverse order, by starting at the end the symbols can be read in forward order. @@ -1258,13 +1274,14 @@ If a bitstream is not entirely and exactly consumed, hence reaching exactly its beginning position with _all_ bits consumed, the decoding process is considered faulty. + Dictionary Format ----------------- -Zstandard is compatible with "raw content" dictionaries, free of any format restriction, -except that they must be at least 8 bytes. -These dictionaries function as if they were just the `Content` block of a formatted -dictionary. +Zstandard is compatible with "raw content" dictionaries, +free of any format restriction, except that they must be at least 8 bytes. +These dictionaries function as if they were just the `Content` part +of a formatted dictionary. But dictionaries created by `zstd --train` follow a format, described here. @@ -1274,9 +1291,9 @@ __Pre-requisites__ : a dictionary has a size, | `Magic_Number` | `Dictionary_ID` | `Entropy_Tables` | `Content` | | -------------- | --------------- | ---------------- | --------- | -__`Magic_Number`__ : 4 bytes ID, value 0xEC30A437, little-endian format +__`Magic_Number`__ : 4 bytes ID, value 0xEC30A437, __little-endian__ format -__`Dictionary_ID`__ : 4 bytes, stored in little-endian format. +__`Dictionary_ID`__ : 4 bytes, stored in __little-endian__ format. `Dictionary_ID` can be any value, except 0 (which means no `Dictionary_ID`). It's used by decoders to check if they use the correct dictionary. @@ -1284,9 +1301,9 @@ _Reserved ranges :_ If the frame is going to be distributed in a private environment, any `Dictionary_ID` can be used. However, for public distribution of compressed frames, - the following ranges are reserved for future use and should not be used : + the following ranges are reserved and shall not be used : - - low range : 1 - 32767 + - low range : <= 32767 - high range : >= (2^31) __`Entropy_Tables`__ : following the same format as the tables in compressed blocks. @@ -1298,26 +1315,30 @@ __`Entropy_Tables`__ : following the same format as the tables in compressed blo These tables populate the Repeat Stats literals mode and Repeat distribution mode for sequence decoding. It's finally followed by 3 offset values, populating recent offsets (instead of using `{1,4,8}`), - stored in order, 4-bytes little-endian each, for a total of 12 bytes. + stored in order, 4-bytes __little-endian__ each, for a total of 12 bytes. Each recent offset must have a value < dictionary size. __`Content`__ : The rest of the dictionary is its content. The content act as a "past" in front of data to compress or decompress, so it can be referenced in sequence commands. As long as the amount of data decoded from this frame is less than or - equal to the window-size, sequence commands may specify offsets longer - than the lenght of total decoded output so far to reference back to the - dictionary. After the total output has surpassed the window size however, + equal to `Window_Size`, sequence commands may specify offsets longer + than the total length of decoded output so far to reference back to the + dictionary. After the total output has surpassed `Window_Size` however, this is no longer allowed and the dictionary is no longer accessible. [compressed blocks]: #the-format-of-compressed_block + + Appendix A - Decoding tables for predefined codes ------------------------------------------------- -This appendix contains FSE decoding tables for the predefined literal length, match length, and offset -codes. The tables have been constructed using the algorithm as given above in the -"from normalized distribution to decoding tables" chapter. The tables here can be used as examples -to crosscheck that an implementation implements the decoding table generation algorithm correctly. +This appendix contains FSE decoding tables +for the predefined literal length, match length, and offset codes. +The tables have been constructed using the algorithm as given above in chapter +"from normalized distribution to decoding tables". +The tables here can be used as examples +to crosscheck that an implementation build its decoding tables correctly. #### Literal Length Code: @@ -1496,6 +1517,7 @@ to crosscheck that an implementation implements the decoding table generation al Version changes --------------- +- 0.2.5 : minor typos and clarifications - 0.2.4 : section restructuring, by Sean Purcell - 0.2.3 : clarified several details, by Sean Purcell - 0.2.2 : added predefined codes, by Johannes Rudolph From a935d67bf141aa2fa9fe59f4211e175b6b8d5b0f Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 31 Mar 2017 16:19:04 -0700 Subject: [PATCH 107/305] minor typo fixes in specification --- doc/zstd_compression_format.md | 35 +++++++++++++++++----------------- 1 file changed, 17 insertions(+), 18 deletions(-) diff --git a/doc/zstd_compression_format.md b/doc/zstd_compression_format.md index 4bbdd57bc..1f212fea2 100644 --- a/doc/zstd_compression_format.md +++ b/doc/zstd_compression_format.md @@ -231,8 +231,8 @@ This information is important for decoders to allocate enough memory. The `Window_Descriptor` byte is optional. When `Single_Segment_flag` is set, `Window_Descriptor` is not present. -In this case, the required buffer size is the frame content size itself, -which can be any value from 0 to 2^64-1 bytes (16 EB). +In this case, `Window_Size` is `Frame_Content_Size`, +which can be any value from 0 to 2^64-1 bytes (16 ExaBytes). | Bit numbers | 7-3 | 2-0 | | ----------- | ---------- | ---------- | @@ -284,7 +284,7 @@ If the frame is going to be distributed in a private environment, any dictionary ID can be used. However, for public distribution of compressed frames using a dictionary, the following ranges are reserved and shall not be used : -- low range : `<= 32767` +- low range : `<= 32767` - high range : `>= (1 << 31)` #### `Frame_Content_Size` @@ -311,23 +311,22 @@ It's allowed to represent a small size (for example `18`) using any compatible v Blocks ------- -After the magic number and header of each block, -there are some number of blocks. -Each frame must have at least one block but there is no upper limit -on the number of blocks per frame. +After `Magic_Number` and `Frame_Header`, there are some number of blocks. +Each frame must have at least one block, +but there is no upper limit on the number of blocks per frame. The structure of a block is as follows: | `Block_Header` | `Block_Content` | |:--------------:|:---------------:| -| 3 bytes | n bytes | +| 3 bytes | n bytes | `Block_Header` uses 3 bytes, written using __little-endian__ convention. It contains 3 fields : -| `Block_Size` | `Block_Type` | `Last_Block` | +| `Last_Block` | `Block_Type` | `Block_Size` | |:------------:|:------------:|:------------:| -| bits 3-23 | bits 1-2 | bit 0 | +| bit 0 | bits 1-2 | bits 3-23 | __`Last_Block`__ @@ -442,7 +441,7 @@ This field uses 2 lowest bits of first byte, describing 4 different block types - `Treeless_Literals_Block` - This is a Huffman-compressed block, using Huffman tree _from previous Huffman-compressed literals block_. `Huffman_Tree_Description` will be skipped. - Note: If this mode is triggering without any previous Huffman-table in the frame + Note: If this mode is triggered without any previous Huffman-table in the frame (or [dictionary](#dictionary-format)), this should be treated as data corruption. __`Size_Format`__ @@ -514,7 +513,7 @@ This section is only present when `Literals_Block_Type` type is `Compressed_Lite The format of the Huffman tree description can be found at [Huffman Tree description](#huffman-tree-description). The size of `Huffman_Tree_Description` is determined during decoding process, it must be used to determine where streams begin. -`Total_Streams_Size = Compress_Size - Huffman_Tree_Description_Size`. +`Total_Streams_Size = Compressed_Size - Huffman_Tree_Description_Size`. For `Treeless_Literals_Block`, the Huffman table comes from previously compressed literals block. @@ -527,14 +526,14 @@ remaining portion of the literals block, encoded as described within If there are four streams, the literals section header only provides enough information to know the decompressed and compressed sizes of all four streams _combined_. -The decompressed size of each stream is equal to `(totalSize+3)/4`, +The decompressed size of each stream is equal to `(Regenerated_Size+3)/4`, except for the last stream which may be up to 3 bytes smaller, to reach a total decompressed size as specified in `Regenerated_Size`. The compressed size of each stream is provided explicitly: the first 6 bytes of the compressed data consist of three 2-byte __little-endian__ fields, describing the compressed sizes of the first three streams. -Stream4 size is computed from total `Total_Streams_Size` minus sizes of other streams. +`Stream4_Size` is computed from total `Total_Streams_Size` minus sizes of other streams. `Stream4_Size = Total_Streams_Size - 6 - Stream1_Size - Stream2_Size - Stream3_Size`. @@ -550,11 +549,11 @@ Sequences Section A compressed block is a succession of _sequences_ . A sequence is a literal copy command, followed by a match copy command. A literal copy command specifies a length. -It is the number of bytes to be copied (or extracted) from the [Literals Section]. +It is the number of bytes to be copied (or extracted) from the Literals Section. A match copy command specifies an offset and a length. When all _sequences_ are decoded, -if there is are any literals left in the _literal section_, +if there are literals left in the _literal section_, these bytes are added at the end of the block. This is described in more detail in [Sequence Execution](#sequence-execution) @@ -586,7 +585,7 @@ This is a variable size field using between 1 and 3 bytes. Let's call its first byte `byte0`. - `if (byte0 == 0)` : there are no sequences. The sequence section stops there. - Decompressed content is defined entirely as [Literals Section] content. + Decompressed content is defined entirely as Literals Section content. - `if (byte0 < 128)` : `Number_of_Sequences = byte0` . Uses 1 byte. - `if (byte0 < 255)` : `Number_of_Sequences = ((byte0-128) << 8) + byte1` . Uses 2 bytes. - `if (byte0 == 255)`: `Number_of_Sequences = byte1 + (byte2<<8) + 0x7F00` . Uses 3 bytes. @@ -622,7 +621,7 @@ They follow the same enumeration : (or [dictionary](#dictionary-format)) to repeat, this should be treated as corruption. - `FSE_Compressed_Mode` : standard FSE compression. A distribution table will be present. - The format of this distribution table is described in (FSE Table Description)[#fse-table-description]. + The format of this distribution table is described in [FSE Table Description](#fse-table-description). Note that the maximum allowed accuracy log for literals length and match length tables is 9, and the maximum accuracy log for the offsets table is 8. From 3f75d5252766bf0e638cf3fb7f5d63da50baaefe Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 31 Mar 2017 17:11:38 -0700 Subject: [PATCH 108/305] Changed ZSTD_compressBound() required so that if Total = A+B compressBound(Total) <= compressBound(A) + compressBound(B) under condition of a minimum size for A and B Will help for ZSTDMT_compress() memory allocation --- lib/compress/zstd_compress.c | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index c31f8db91..cc69f1184 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -32,7 +32,10 @@ typedef enum { ZSTDcs_created=0, ZSTDcs_init, ZSTDcs_ongoing, ZSTDcs_ending } ZS * Helper functions ***************************************/ #define ZSTD_STATIC_ASSERT(c) { enum { ZSTD_static_assert = 1/(int)(!!(c)) }; } -size_t ZSTD_compressBound(size_t srcSize) { return FSE_compressBound(srcSize) + 12; } +size_t ZSTD_compressBound(size_t srcSize) { + size_t const margin = (srcSize < 512 KB) ? 16 : 0; + return srcSize + (srcSize >> 8) + margin; +} /*-************************************* From 21b6c53b9f153b97a0a632f18a041eac53a4b8fa Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Fri, 31 Mar 2017 15:01:41 -0700 Subject: [PATCH 109/305] Working on API --- contrib/linux-kernel/include/zstd.h | 317 ++++++--------------- contrib/linux-kernel/lib/error_private.c | 44 --- contrib/linux-kernel/lib/error_private.h | 2 +- contrib/linux-kernel/lib/fse.h | 66 ----- contrib/linux-kernel/lib/fse_compress.c | 47 +-- contrib/linux-kernel/lib/fse_decompress.c | 23 +- contrib/linux-kernel/lib/huf.h | 42 --- contrib/linux-kernel/lib/huf_compress.c | 21 -- contrib/linux-kernel/lib/huf_decompress.c | 43 +-- contrib/linux-kernel/lib/xxhash.c | 32 +-- contrib/linux-kernel/lib/xxhash.h | 8 - contrib/linux-kernel/lib/zstd_common.c | 58 ++-- contrib/linux-kernel/lib/zstd_compress.c | 165 ++++------- contrib/linux-kernel/lib/zstd_decompress.c | 124 +++----- contrib/linux-kernel/lib/zstd_errors.h | 55 ---- contrib/linux-kernel/lib/zstd_internal.h | 45 ++- 16 files changed, 246 insertions(+), 846 deletions(-) delete mode 100644 contrib/linux-kernel/lib/error_private.c delete mode 100644 contrib/linux-kernel/lib/zstd_errors.h diff --git a/contrib/linux-kernel/include/zstd.h b/contrib/linux-kernel/include/zstd.h index 8b1afa4ba..6594521d6 100644 --- a/contrib/linux-kernel/include/zstd.h +++ b/contrib/linux-kernel/include/zstd.h @@ -52,77 +52,92 @@ ZSTDLIB_API unsigned ZSTD_versionNumber(void); /**< library version number; to be used when checking dll version */ -/*************************************** -* Simple API -***************************************/ -/*! ZSTD_compress() : - Compresses `src` content as a single zstd compressed frame into already allocated `dst`. - Hint : compression runs faster if `dstCapacity` >= `ZSTD_compressBound(srcSize)`. - @return : compressed size written into `dst` (<= `dstCapacity), - or an error code if it fails (which can be tested using ZSTD_isError()). */ -ZSTDLIB_API size_t ZSTD_compress( void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - int compressionLevel); - -/*! ZSTD_decompress() : - `compressedSize` : must be the _exact_ size of some number of compressed and/or skippable frames. - `dstCapacity` is an upper bound of originalSize. - If user cannot imply a maximum upper bound, it's better to use streaming mode to decompress data. - @return : the number of bytes decompressed into `dst` (<= `dstCapacity`), - or an errorCode if it fails (which can be tested using ZSTD_isError()). */ -ZSTDLIB_API size_t ZSTD_decompress( void* dst, size_t dstCapacity, - const void* src, size_t compressedSize); - -/*! ZSTD_getDecompressedSize() : -* NOTE: This function is planned to be obsolete, in favour of ZSTD_getFrameContentSize. -* ZSTD_getFrameContentSize functions the same way, returning the decompressed size of a single -* frame, but distinguishes empty frames from frames with an unknown size, or errors. -* -* Additionally, ZSTD_findDecompressedSize can be used instead. It can handle multiple -* concatenated frames in one buffer, and so is more general. -* As a result however, it requires more computation and entire frames to be passed to it, -* as opposed to ZSTD_getFrameContentSize which requires only a single frame's header. -* -* 'src' is the start of a zstd compressed frame. -* @return : content size to be decompressed, as a 64-bits value _if known_, 0 otherwise. -* note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode. -* When `return==0`, data to decompress could be any size. -* In which case, it's necessary to use streaming mode to decompress data. -* Optionally, application can still use ZSTD_decompress() while relying on implied limits. -* (For example, data may be necessarily cut into blocks <= 16 KB). -* note 2 : decompressed size is always present when compression is done with ZSTD_compress() -* note 3 : decompressed size can be very large (64-bits value), -* potentially larger than what local system can handle as a single memory segment. -* In which case, it's necessary to use streaming mode to decompress data. -* note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified. -* Always ensure result fits within application's authorized limits. -* Each application can set its own limits. -* note 5 : when `return==0`, if precise failure cause is needed, use ZSTD_getFrameParams() to know more. */ -ZSTDLIB_API unsigned long long ZSTD_getDecompressedSize(const void* src, size_t srcSize); - - /*====== Helper functions ======*/ +typedef enum { + ZSTD_error_no_error, + ZSTD_error_GENERIC, + ZSTD_error_prefix_unknown, + ZSTD_error_version_unsupported, + ZSTD_error_parameter_unknown, + ZSTD_error_frameParameter_unsupported, + ZSTD_error_frameParameter_unsupportedBy32bits, + ZSTD_error_frameParameter_windowTooLarge, + ZSTD_error_compressionParameter_unsupported, + ZSTD_error_init_missing, + ZSTD_error_memory_allocation, + ZSTD_error_stage_wrong, + ZSTD_error_dstSize_tooSmall, + ZSTD_error_srcSize_wrong, + ZSTD_error_corruption_detected, + ZSTD_error_checksum_wrong, + ZSTD_error_tableLog_tooLarge, + ZSTD_error_maxSymbolValue_tooLarge, + ZSTD_error_maxSymbolValue_tooSmall, + ZSTD_error_dictionary_corrupted, + ZSTD_error_dictionary_wrong, + ZSTD_error_dictionaryCreation_failed, + ZSTD_error_maxCode +} ZSTD_ErrorCode; + ZSTDLIB_API int ZSTD_maxCLevel(void); /*!< maximum compression level available */ ZSTDLIB_API size_t ZSTD_compressBound(size_t srcSize); /*!< maximum compressed size in worst case scenario */ -ZSTDLIB_API unsigned ZSTD_isError(size_t code); /*!< tells if a `size_t` function result is an error code */ -ZSTDLIB_API const char* ZSTD_getErrorName(size_t code); /*!< provides readable string from an error code */ - +/*! ZSTD_isError() : +* tells if a `size_t` function result is an error code */ +ZSTDLIB_API static __attribute__((unused)) unsigned ZSTD_isError(size_t code) { + return code > (size_t)-ZSTD_error_maxCode; +} +/*! ZSTD_getErrorCode() : +* convert a `size_t` function result into a proper ZSTD_errorCode enum */ +ZSTDLIB_API static __attribute__((unused)) ZSTD_ErrorCode ZSTD_getErrorCode(size_t functionResult) { + if (!ZSTD_isError(functionResult)) { + return (ZSTD_ErrorCode)0; + } + return (ZSTD_ErrorCode)(0 - functionResult); +} /*************************************** * Explicit memory management ***************************************/ + +typedef enum { ZSTD_fast, ZSTD_dfast, ZSTD_greedy, ZSTD_lazy, ZSTD_lazy2, ZSTD_btlazy2, ZSTD_btopt, ZSTD_btopt2 } ZSTD_strategy; /* from faster to stronger */ + +typedef struct { + unsigned windowLog; /**< largest match distance : larger == more compression, more memory needed during decompression */ + unsigned chainLog; /**< fully searched segment : larger == more compression, slower, more memory (useless for fast) */ + unsigned hashLog; /**< dispatch table : larger == faster, more memory */ + unsigned searchLog; /**< nb of searches : larger == more compression, slower */ + unsigned searchLength; /**< match length searched : larger == faster decompression, sometimes less compression */ + unsigned targetLength; /**< acceptable match size for optimal parser (only) : larger == more compression, slower */ + ZSTD_strategy strategy; +} ZSTD_compressionParameters; + +typedef struct { + unsigned contentSizeFlag; /**< 1: content size will be in frame header (when known) */ + unsigned checksumFlag; /**< 1: generate a 32-bits checksum at end of frame, for error detection */ + unsigned noDictIDFlag; /**< 1: no dictID will be saved into frame header (if dictionary compression) */ +} ZSTD_frameParameters; + +typedef struct { + ZSTD_compressionParameters cParams; + ZSTD_frameParameters fParams; +} ZSTD_parameters; + +size_t ZSTD_CCtxWorkspaceBound(ZSTD_compressionParameters params); +size_t ZSTD_DCtxWorkspaceBound(void); + /*= Compression context * When compressing many times, * it is recommended to allocate a context just once, and re-use it for each successive compression operation. * This will make workload friendlier for system's memory. * Use one context per thread for parallel execution in multi-threaded environments. */ typedef struct ZSTD_CCtx_s ZSTD_CCtx; -ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx(void); -ZSTDLIB_API size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx); +ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx(void* workspace, size_t workspaceSize); /*! ZSTD_compressCCtx() : Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). */ -ZSTDLIB_API size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel); +ZSTDLIB_API size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, ZSTD_parameters params); + +ZSTDLIB_API size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void *dict, size_t dictSize, ZSTD_parameters params); /*= Decompression context * When decompressing many times, @@ -130,53 +145,28 @@ ZSTDLIB_API size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapaci * This will make workload friendlier for system's memory. * Use one context per thread for parallel execution in multi-threaded environments. */ typedef struct ZSTD_DCtx_s ZSTD_DCtx; -ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx(void); -ZSTDLIB_API size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx); +ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx(void* workspace, size_t workspaceSize); /*! ZSTD_decompressDCtx() : * Same as ZSTD_decompress(), requires an allocated ZSTD_DCtx (see ZSTD_createDCtx()). */ ZSTDLIB_API size_t ZSTD_decompressDCtx(ZSTD_DCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); - -/************************** -* Simple dictionary API -***************************/ -/*! ZSTD_compress_usingDict() : -* Compression using a predefined Dictionary (see dictBuilder/zdict.h). -* Note : This function loads the dictionary, resulting in significant startup delay. -* Note : When `dict == NULL || dictSize < 8` no dictionary is used. */ -ZSTDLIB_API size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict,size_t dictSize, - int compressionLevel); - -/*! ZSTD_decompress_usingDict() : -* Decompression using a predefined Dictionary (see dictBuilder/zdict.h). -* Dictionary must be identical to the one used during compression. -* Note : This function loads the dictionary, resulting in significant startup delay. -* Note : When `dict == NULL || dictSize < 8` no dictionary is used. */ -ZSTDLIB_API size_t ZSTD_decompress_usingDict(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict,size_t dictSize); - +ZSTDLIB_API size_t ZSTD_decompress_usingDict(ZSTD_DCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void *dict, size_t dictSize); /**************************** * Fast dictionary API ****************************/ +size_t ZSTD_CDictWorkspaceBound(ZSTD_compressionParameters params); +size_t ZSTD_DDictWorkspaceBound(void); + typedef struct ZSTD_CDict_s ZSTD_CDict; /*! ZSTD_createCDict() : * When compressing multiple messages / blocks with the same dictionary, it's recommended to load it just once. * ZSTD_createCDict() will create a digested dictionary, ready to start future compression operations without startup delay. * ZSTD_CDict can be created once and used by multiple threads concurrently, as its usage is read-only. -* `dictBuffer` can be released after ZSTD_CDict creation, as its content is copied within CDict */ -ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict(const void* dictBuffer, size_t dictSize, int compressionLevel); - -/*! ZSTD_freeCDict() : -* Function frees memory allocated by ZSTD_createCDict(). */ -ZSTDLIB_API size_t ZSTD_freeCDict(ZSTD_CDict* CDict); +* `dictBuffer` content is referenced, and it must remain accessible throughout the lifetime of the CDict */ +ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict(const void* dictBuffer, size_t dictSize, ZSTD_parameters params, void* workspace, size_t workspaceSize); /*! ZSTD_compress_usingCDict() : * Compression using a digested Dictionary. @@ -192,12 +182,8 @@ typedef struct ZSTD_DDict_s ZSTD_DDict; /*! ZSTD_createDDict() : * Create a digested dictionary, ready to start decompression operation without startup delay. -* dictBuffer can be released after DDict creation, as its content is copied inside DDict */ -ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict(const void* dictBuffer, size_t dictSize); - -/*! ZSTD_freeDDict() : -* Function frees memory allocated with ZSTD_createDDict() */ -ZSTDLIB_API size_t ZSTD_freeDDict(ZSTD_DDict* ddict); +* `dictBuffer` content is referenced, and it must remain accessible throughout the lifetime of the DDict */ +ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict(const void* dictBuffer, size_t dictSize, void* workspace, size_t workspaceSize); /*! ZSTD_decompress_usingDDict() : * Decompression using a digested Dictionary. @@ -265,13 +251,17 @@ typedef struct ZSTD_outBuffer_s { * * *******************************************************************/ +size_t ZSTD_CStreamWorkspaceBound(ZSTD_compressionParameters params); +size_t ZSTD_DStreamWorkspaceBound(size_t maxWindowSize); + typedef struct ZSTD_CStream_s ZSTD_CStream; /*===== ZSTD_CStream management functions =====*/ -ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream(void); -ZSTDLIB_API size_t ZSTD_freeCStream(ZSTD_CStream* zcs); +ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream(void* workspace, size_t workspaceSize); +ZSTDLIB_API size_t ZSTD_initCStream(ZSTD_CStream* zcs, ZSTD_parameters params, unsigned long long pledgedSrcSize); +ZSTDLIB_API size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, ZSTD_parameters params, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /*===== Streaming compression functions =====*/ -ZSTDLIB_API size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel); +ZSTDLIB_API size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize); /**< re-use compression parameters from previous init; skip dictionary loading stage; zcs must be init at least once before. note: pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ ZSTDLIB_API size_t ZSTD_compressStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input); ZSTDLIB_API size_t ZSTD_flushStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output); ZSTDLIB_API size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output); @@ -305,11 +295,12 @@ ZSTDLIB_API size_t ZSTD_CStreamOutSize(void); /**< recommended size for output typedef struct ZSTD_DStream_s ZSTD_DStream; /*===== ZSTD_DStream management functions =====*/ -ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream(void); -ZSTDLIB_API size_t ZSTD_freeDStream(ZSTD_DStream* zds); +ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream(void* workspace, size_t workspaceSize); +ZSTDLIB_API size_t ZSTD_initDStream(ZSTD_DStream* zds, size_t maxWindowSize); +ZSTDLIB_API size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, size_t maxWindowSize, const ZSTD_DDict* ddict); /*===== Streaming decompression functions =====*/ -ZSTDLIB_API size_t ZSTD_initDStream(ZSTD_DStream* zds); +ZSTDLIB_API size_t ZSTD_resetDStream(ZSTD_DStream* zds); /**< re-use decompression parameters from previous init; saves dictionary loading */ ZSTDLIB_API size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input); ZSTDLIB_API size_t ZSTD_DStreamInSize(void); /*!< recommended size for input buffer */ @@ -355,35 +346,6 @@ static const size_t ZSTD_frameHeaderSize_max = ZSTD_FRAMEHEADERSIZE_MAX; static const size_t ZSTD_skippableHeaderSize = 8; /* magic number + skippable frame length */ -/*--- Advanced types ---*/ -typedef enum { ZSTD_fast, ZSTD_dfast, ZSTD_greedy, ZSTD_lazy, ZSTD_lazy2, ZSTD_btlazy2, ZSTD_btopt, ZSTD_btopt2 } ZSTD_strategy; /* from faster to stronger */ - -typedef struct { - unsigned windowLog; /**< largest match distance : larger == more compression, more memory needed during decompression */ - unsigned chainLog; /**< fully searched segment : larger == more compression, slower, more memory (useless for fast) */ - unsigned hashLog; /**< dispatch table : larger == faster, more memory */ - unsigned searchLog; /**< nb of searches : larger == more compression, slower */ - unsigned searchLength; /**< match length searched : larger == faster decompression, sometimes less compression */ - unsigned targetLength; /**< acceptable match size for optimal parser (only) : larger == more compression, slower */ - ZSTD_strategy strategy; -} ZSTD_compressionParameters; - -typedef struct { - unsigned contentSizeFlag; /**< 1: content size will be in frame header (when known) */ - unsigned checksumFlag; /**< 1: generate a 32-bits checksum at end of frame, for error detection */ - unsigned noDictIDFlag; /**< 1: no dictID will be saved into frame header (if dictionary compression) */ -} ZSTD_frameParameters; - -typedef struct { - ZSTD_compressionParameters cParams; - ZSTD_frameParameters fParams; -} ZSTD_parameters; - -/*= Custom memory allocation functions */ -typedef void* (*ZSTD_allocFunction) (void* opaque, size_t size); -typedef void (*ZSTD_freeFunction) (void* opaque, void* address); -typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; void* opaque; } ZSTD_customMem; - /*************************************** * Compressed size functions ***************************************/ @@ -432,47 +394,9 @@ ZSTDLIB_API unsigned long long ZSTD_getFrameContentSize(const void *src, size_t * however it does mean that all frame data must be present and valid. */ ZSTDLIB_API unsigned long long ZSTD_findDecompressedSize(const void* src, size_t srcSize); - /*************************************** * Advanced compression functions ***************************************/ -/*! ZSTD_estimateCCtxSize() : - * Gives the amount of memory allocated for a ZSTD_CCtx given a set of compression parameters. - * `frameContentSize` is an optional parameter, provide `0` if unknown */ -ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams); - -/*! ZSTD_createCCtx_advanced() : - * Create a ZSTD compression context using external alloc and free functions */ -ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx_advanced(ZSTD_customMem customMem); - -/*! ZSTD_sizeofCCtx() : - * Gives the amount of memory used by a given ZSTD_CCtx */ -ZSTDLIB_API size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx); - -typedef enum { - ZSTD_p_forceWindow, /* Force back-references to remain < windowSize, even when referencing Dictionary content (default:0) */ - ZSTD_p_forceRawDict /* Force loading dictionary in "content-only" mode (no header analysis) */ -} ZSTD_CCtxParameter; -/*! ZSTD_setCCtxParameter() : - * Set advanced parameters, selected through enum ZSTD_CCtxParameter - * @result : 0, or an error code (which can be tested with ZSTD_isError()) */ -ZSTDLIB_API size_t ZSTD_setCCtxParameter(ZSTD_CCtx* cctx, ZSTD_CCtxParameter param, unsigned value); - -/*! ZSTD_createCDict_byReference() : - * Create a digested dictionary for compression - * Dictionary content is simply referenced, and therefore stays in dictBuffer. - * It is important that dictBuffer outlives CDict, it must remain read accessible throughout the lifetime of CDict */ -ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_byReference(const void* dictBuffer, size_t dictSize, int compressionLevel); - -/*! ZSTD_createCDict_advanced() : - * Create a ZSTD_CDict using external alloc and free, and customized compression parameters */ -ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_advanced(const void* dict, size_t dictSize, unsigned byReference, - ZSTD_parameters params, ZSTD_customMem customMem); - -/*! ZSTD_sizeof_CDict() : - * Gives the amount of memory used by a given ZSTD_sizeof_CDict */ -ZSTDLIB_API size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict); - /*! ZSTD_getCParams() : * @return ZSTD_compressionParameters structure for a selected compression level and estimated srcSize. * `estimatedSrcSize` value is optional, select 0 if not known */ @@ -492,15 +416,6 @@ ZSTDLIB_API size_t ZSTD_checkCParams(ZSTD_compressionParameters params); * both values are optional, select `0` if unknown. */ ZSTDLIB_API ZSTD_compressionParameters ZSTD_adjustCParams(ZSTD_compressionParameters cPar, unsigned long long srcSize, size_t dictSize); -/*! ZSTD_compress_advanced() : -* Same as ZSTD_compress_usingDict(), with fine-tune control of each compression parameter */ -ZSTDLIB_API size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict,size_t dictSize, - ZSTD_parameters params); - - /*--- Advanced decompression functions ---*/ /*! ZSTD_isFrame() : @@ -510,33 +425,6 @@ ZSTDLIB_API size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx, * Note 3 : Skippable Frame Identifiers are considered valid. */ ZSTDLIB_API unsigned ZSTD_isFrame(const void* buffer, size_t size); -/*! ZSTD_estimateDCtxSize() : - * Gives the potential amount of memory allocated to create a ZSTD_DCtx */ -ZSTDLIB_API size_t ZSTD_estimateDCtxSize(void); - -/*! ZSTD_createDCtx_advanced() : - * Create a ZSTD decompression context using external alloc and free functions */ -ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx_advanced(ZSTD_customMem customMem); - -/*! ZSTD_sizeof_DCtx() : - * Gives the amount of memory used by a given ZSTD_DCtx */ -ZSTDLIB_API size_t ZSTD_sizeof_DCtx(const ZSTD_DCtx* dctx); - -/*! ZSTD_createDDict_byReference() : - * Create a digested dictionary, ready to start decompression operation without startup delay. - * Dictionary content is simply referenced, and therefore stays in dictBuffer. - * It is important that dictBuffer outlives DDict, it must remain read accessible throughout the lifetime of DDict */ -ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_byReference(const void* dictBuffer, size_t dictSize); - -/*! ZSTD_createDDict_advanced() : - * Create a ZSTD_DDict using external alloc and free, optionally by reference */ -ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize, - unsigned byReference, ZSTD_customMem customMem); - -/*! ZSTD_sizeof_DDict() : - * Gives the amount of memory used by a given ZSTD_DDict */ -ZSTDLIB_API size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict); - /*! ZSTD_getDictID_fromDict() : * Provides the dictID stored within dictionary. * if @return == 0, the dictionary is not conformant with Zstandard specification. @@ -562,31 +450,6 @@ ZSTDLIB_API unsigned ZSTD_getDictID_fromDDict(const ZSTD_DDict* ddict); ZSTDLIB_API unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize); -/******************************************************************** -* Advanced streaming functions -********************************************************************/ - -/*===== Advanced Streaming compression functions =====*/ -ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem); -ZSTDLIB_API size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize); /**< pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ -ZSTDLIB_API size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */ -ZSTDLIB_API size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize, - ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */ -ZSTDLIB_API size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict); /**< note : cdict will just be referenced, and must outlive compression session */ -ZSTDLIB_API size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize); /**< re-use compression parameters from previous init; skip dictionary loading stage; zcs must be init at least once before. note: pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ -ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs); - - -/*===== Advanced Streaming decompression functions =====*/ -typedef enum { DStream_p_maxWindowSize } ZSTD_DStreamParameter_e; -ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem); -ZSTDLIB_API size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */ -ZSTDLIB_API size_t ZSTD_setDStreamParameter(ZSTD_DStream* zds, ZSTD_DStreamParameter_e paramType, unsigned paramValue); -ZSTDLIB_API size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, const ZSTD_DDict* ddict); /**< note : ddict will just be referenced, and must outlive decompression session */ -ZSTDLIB_API size_t ZSTD_resetDStream(ZSTD_DStream* zds); /**< re-use decompression parameters from previous init; saves dictionary loading */ -ZSTDLIB_API size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds); - - /********************************************************************* * Buffer-less and synchronous inner streaming functions * diff --git a/contrib/linux-kernel/lib/error_private.c b/contrib/linux-kernel/lib/error_private.c deleted file mode 100644 index 83e27cb49..000000000 --- a/contrib/linux-kernel/lib/error_private.c +++ /dev/null @@ -1,44 +0,0 @@ -/** - * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. - * All rights reserved. - * - * This source code is licensed under the BSD-style license found in the - * LICENSE file in the root directory of this source tree. An additional grant - * of patent rights can be found in the PATENTS file in the same directory. - */ - -/* The purpose of this file is to have a single list of error strings embedded in binary */ - -#include "error_private.h" - -const char* ERR_getErrorString(ERR_enum code) -{ - static const char* const notErrorCode = "Unspecified error code"; - switch( code ) - { - case PREFIX(no_error): return "No error detected"; - case PREFIX(GENERIC): return "Error (generic)"; - case PREFIX(prefix_unknown): return "Unknown frame descriptor"; - case PREFIX(version_unsupported): return "Version not supported"; - case PREFIX(parameter_unknown): return "Unknown parameter type"; - case PREFIX(frameParameter_unsupported): return "Unsupported frame parameter"; - case PREFIX(frameParameter_unsupportedBy32bits): return "Frame parameter unsupported in 32-bits mode"; - case PREFIX(frameParameter_windowTooLarge): return "Frame requires too much memory for decoding"; - case PREFIX(compressionParameter_unsupported): return "Compression parameter is out of bound"; - case PREFIX(init_missing): return "Context should be init first"; - case PREFIX(memory_allocation): return "Allocation error : not enough memory"; - case PREFIX(stage_wrong): return "Operation not authorized at current processing stage"; - case PREFIX(dstSize_tooSmall): return "Destination buffer is too small"; - case PREFIX(srcSize_wrong): return "Src size incorrect"; - case PREFIX(corruption_detected): return "Corrupted block detected"; - case PREFIX(checksum_wrong): return "Restored data doesn't match checksum"; - case PREFIX(tableLog_tooLarge): return "tableLog requires too much memory : unsupported"; - case PREFIX(maxSymbolValue_tooLarge): return "Unsupported max Symbol Value : too large"; - case PREFIX(maxSymbolValue_tooSmall): return "Specified maxSymbolValue is too small"; - case PREFIX(dictionary_corrupted): return "Dictionary is corrupted"; - case PREFIX(dictionary_wrong): return "Dictionary mismatch"; - case PREFIX(dictionaryCreation_failed): return "Cannot create Dictionary from provided samples"; - case PREFIX(maxCode): - default: return notErrorCode; - } -} diff --git a/contrib/linux-kernel/lib/error_private.h b/contrib/linux-kernel/lib/error_private.h index 7b2ce36d4..680eba112 100644 --- a/contrib/linux-kernel/lib/error_private.h +++ b/contrib/linux-kernel/lib/error_private.h @@ -16,7 +16,7 @@ * Dependencies ******************************************/ #include /* size_t */ -#include "zstd_errors.h" /* enum list */ +#include "zstd.h" /* enum list */ /* **************************************** diff --git a/contrib/linux-kernel/lib/fse.h b/contrib/linux-kernel/lib/fse.h index a5c3980a7..6df7ff935 100644 --- a/contrib/linux-kernel/lib/fse.h +++ b/contrib/linux-kernel/lib/fse.h @@ -59,34 +59,6 @@ #define FSE_VERSION_NUMBER (FSE_VERSION_MAJOR *100*100 + FSE_VERSION_MINOR *100 + FSE_VERSION_RELEASE) FSE_PUBLIC_API unsigned FSE_versionNumber(void); /**< library version number; to be used when checking dll version */ -/*-**************************************** -* FSE simple functions -******************************************/ -/*! FSE_compress() : - Compress content of buffer 'src', of size 'srcSize', into destination buffer 'dst'. - 'dst' buffer must be already allocated. Compression runs faster is dstCapacity >= FSE_compressBound(srcSize). - @return : size of compressed data (<= dstCapacity). - Special values : if return == 0, srcData is not compressible => Nothing is stored within dst !!! - if return == 1, srcData is a single byte symbol * srcSize times. Use RLE compression instead. - if FSE_isError(return), compression failed (more details using FSE_getErrorName()) -*/ -FSE_PUBLIC_API size_t FSE_compress(void* dst, size_t dstCapacity, - const void* src, size_t srcSize); - -/*! FSE_decompress(): - Decompress FSE data from buffer 'cSrc', of size 'cSrcSize', - into already allocated destination buffer 'dst', of size 'dstCapacity'. - @return : size of regenerated data (<= maxDstSize), - or an error code, which can be tested using FSE_isError() . - - ** Important ** : FSE_decompress() does not decompress non-compressible nor RLE data !!! - Why ? : making this distinction requires a header. - Header management is intentionally delegated to the user layer, which can better manage special cases. -*/ -FSE_PUBLIC_API size_t FSE_decompress(void* dst, size_t dstCapacity, - const void* cSrc, size_t cSrcSize); - - /*-***************************************** * Tool functions ******************************************/ @@ -97,20 +69,6 @@ FSE_PUBLIC_API unsigned FSE_isError(size_t code); /* tells if a return FSE_PUBLIC_API const char* FSE_getErrorName(size_t code); /* provides error code string (useful for debugging) */ -/*-***************************************** -* FSE advanced functions -******************************************/ -/*! FSE_compress2() : - Same as FSE_compress(), but allows the selection of 'maxSymbolValue' and 'tableLog' - Both parameters can be defined as '0' to mean : use default value - @return : size of compressed data - Special values : if return == 0, srcData is not compressible => Nothing is stored within cSrc !!! - if return == 1, srcData is a single byte symbol * srcSize times. Use RLE compression. - if FSE_isError(return), it's an error code. -*/ -FSE_PUBLIC_API size_t FSE_compress2 (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog); - - /*-***************************************** * FSE detailed API ******************************************/ @@ -133,16 +91,6 @@ or to save and provide normalized distribution using external method. */ /* *** COMPRESSION *** */ - -/*! FSE_count(): - Provides the precise count of each byte within a table 'count'. - 'count' is a table of unsigned int, of minimum size (*maxSymbolValuePtr+1). - *maxSymbolValuePtr will be updated if detected smaller than initial value. - @return : the count of the most frequent symbol (which is not identified). - if return == srcSize, there is only one symbol. - Can also return an error code, which can be tested with FSE_isError(). */ -FSE_PUBLIC_API size_t FSE_count(unsigned* count, unsigned* maxSymbolValuePtr, const void* src, size_t srcSize); - /*! FSE_optimalTableLog(): dynamically downsize 'tableLog' when conditions are met. It saves CPU time, by using smaller tables, while preserving or even improving compression ratio. @@ -171,13 +119,6 @@ FSE_PUBLIC_API size_t FSE_writeNCount (void* buffer, size_t bufferSize, const sh /*! Constructor and Destructor of FSE_CTable. Note that FSE_CTable size depends on 'tableLog' and 'maxSymbolValue' */ typedef unsigned FSE_CTable; /* don't allocate that. It's only meant to be more restrictive than void* */ -FSE_PUBLIC_API FSE_CTable* FSE_createCTable (unsigned tableLog, unsigned maxSymbolValue); -FSE_PUBLIC_API void FSE_freeCTable (FSE_CTable* ct); - -/*! FSE_buildCTable(): - Builds `ct`, which must be already allocated, using FSE_createCTable(). - @return : 0, or an errorCode, which can be tested using FSE_isError() */ -FSE_PUBLIC_API size_t FSE_buildCTable(FSE_CTable* ct, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog); /*! FSE_compress_usingCTable(): Compress `src` using `ct` into `dst` which must be already allocated. @@ -242,8 +183,6 @@ FSE_PUBLIC_API size_t FSE_readNCount (short* normalizedCounter, unsigned* maxSym /*! Constructor and Destructor of FSE_DTable. Note that its size depends on 'tableLog' */ typedef unsigned FSE_DTable; /* don't allocate that. It's just a way to be more restrictive than void* */ -FSE_PUBLIC_API FSE_DTable* FSE_createDTable(unsigned tableLog); -FSE_PUBLIC_API void FSE_freeDTable(FSE_DTable* dt); /*! FSE_buildDTable(): Builds 'dt', which must be already allocated, using FSE_createDTable(). @@ -313,11 +252,6 @@ If there is an error, the function will return an error code, which can be teste size_t FSE_count_wksp(unsigned* count, unsigned* maxSymbolValuePtr, const void* source, size_t sourceSize, unsigned* workSpace); -/** FSE_countFast() : - * same as FSE_count(), but blindly trusts that all byte values within src are <= *maxSymbolValuePtr - */ -size_t FSE_countFast(unsigned* count, unsigned* maxSymbolValuePtr, const void* src, size_t srcSize); - /* FSE_countFast_wksp() : * Same as FSE_countFast(), but using an externally provided scratch buffer. * `workSpace` must be a table of minimum `1024` unsigned diff --git a/contrib/linux-kernel/lib/fse_compress.c b/contrib/linux-kernel/lib/fse_compress.c index 8b85b85f1..7c13a4099 100644 --- a/contrib/linux-kernel/lib/fse_compress.c +++ b/contrib/linux-kernel/lib/fse_compress.c @@ -35,12 +35,13 @@ /* ************************************************************** * Compiler specifics ****************************************************************/ -#define FORCE_INLINE static __attribute__((always_inline)) +#define FORCE_INLINE static __always_inline /* ************************************************************** * Includes ****************************************************************/ +#include #include /* memcpy, memset */ #include "bitstream.h" #include "fse.h" @@ -164,14 +165,6 @@ size_t FSE_buildCTable_wksp(FSE_CTable* ct, const short* normalizedCounter, unsi } -size_t FSE_buildCTable(FSE_CTable* ct, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog) -{ - FSE_FUNCTION_TYPE tableSymbol[FSE_MAX_TABLESIZE]; /* memset() is not necessary, even if static analyzer complain about it */ - return FSE_buildCTable_wksp(ct, normalizedCounter, maxSymbolValue, tableLog, tableSymbol, sizeof(tableSymbol)); -} - - - #ifndef FSE_COMMONDEFS_ONLY /*-************************************************************** @@ -399,14 +392,6 @@ size_t FSE_countFast_wksp(unsigned* count, unsigned* maxSymbolValuePtr, return FSE_count_parallel_wksp(count, maxSymbolValuePtr, source, sourceSize, 0, workSpace); } -/* fast variant (unsafe : won't check if src contains values beyond count[] limit) */ -size_t FSE_countFast(unsigned* count, unsigned* maxSymbolValuePtr, - const void* source, size_t sourceSize) -{ - unsigned tmpCounters[1024]; - return FSE_countFast_wksp(count, maxSymbolValuePtr, source, sourceSize, tmpCounters); -} - /* FSE_count_wksp() : * Same as FSE_count(), but using an externally provided scratch buffer. * `workSpace` size must be table of >= `1024` unsigned */ @@ -445,16 +430,6 @@ size_t FSE_sizeof_CTable (unsigned maxSymbolValue, unsigned tableLog) return FSE_CTABLE_SIZE_U32 (tableLog, maxSymbolValue) * sizeof(U32); } -FSE_CTable* FSE_createCTable (unsigned maxSymbolValue, unsigned tableLog) -{ - size_t size; - if (tableLog > FSE_TABLELOG_ABSOLUTE_MAX) tableLog = FSE_TABLELOG_ABSOLUTE_MAX; - size = FSE_CTABLE_SIZE_U32 (tableLog, maxSymbolValue) * sizeof(U32); - return (FSE_CTable*)malloc(size); -} - -void FSE_freeCTable (FSE_CTable* ct) { free(ct); } - /* provides the minimum logSize to safely represent a distribution */ static unsigned FSE_minTableLog(size_t srcSize, unsigned maxSymbolValue) { @@ -817,23 +792,5 @@ size_t FSE_compress_wksp (void* dst, size_t dstSize, const void* src, size_t src return op-ostart; } -typedef struct { - FSE_CTable CTable_max[FSE_CTABLE_SIZE_U32(FSE_MAX_TABLELOG, FSE_MAX_SYMBOL_VALUE)]; - BYTE scratchBuffer[1 << FSE_MAX_TABLELOG]; -} fseWkspMax_t; - -size_t FSE_compress2 (void* dst, size_t dstCapacity, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog) -{ - fseWkspMax_t scratchBuffer; - FSE_STATIC_ASSERT(sizeof(scratchBuffer) >= FSE_WKSP_SIZE_U32(FSE_MAX_TABLELOG, FSE_MAX_SYMBOL_VALUE)); /* compilation failures here means scratchBuffer is not large enough */ - if (tableLog > FSE_MAX_TABLELOG) return ERROR(tableLog_tooLarge); - return FSE_compress_wksp(dst, dstCapacity, src, srcSize, maxSymbolValue, tableLog, &scratchBuffer, sizeof(scratchBuffer)); -} - -size_t FSE_compress (void* dst, size_t dstCapacity, const void* src, size_t srcSize) -{ - return FSE_compress2(dst, dstCapacity, src, srcSize, FSE_MAX_SYMBOL_VALUE, FSE_DEFAULT_TABLELOG); -} - #endif /* FSE_COMMONDEFS_ONLY */ diff --git a/contrib/linux-kernel/lib/fse_decompress.c b/contrib/linux-kernel/lib/fse_decompress.c index ebc228daf..2a35f1703 100644 --- a/contrib/linux-kernel/lib/fse_decompress.c +++ b/contrib/linux-kernel/lib/fse_decompress.c @@ -36,12 +36,13 @@ /* ************************************************************** * Compiler specifics ****************************************************************/ -#define FORCE_INLINE static __attribute__((always_inline)) +#define FORCE_INLINE static __always_inline /* ************************************************************** * Includes ****************************************************************/ +#include #include /* memcpy, memset */ #include "bitstream.h" #include "fse.h" @@ -81,16 +82,6 @@ /* Function templates */ -FSE_DTable* FSE_createDTable (unsigned tableLog) -{ - if (tableLog > FSE_TABLELOG_ABSOLUTE_MAX) tableLog = FSE_TABLELOG_ABSOLUTE_MAX; - return (FSE_DTable*)malloc( FSE_DTABLE_SIZE_U32(tableLog) * sizeof (U32) ); -} - -void FSE_freeDTable (FSE_DTable* dt) -{ - free(dt); -} size_t FSE_buildDTable(FSE_DTable* dt, const short* normalizedCounter, unsigned maxSymbolValue, unsigned tableLog) { @@ -298,14 +289,4 @@ size_t FSE_decompress_wksp(void* dst, size_t dstCapacity, const void* cSrc, size } -typedef FSE_DTable DTable_max_t[FSE_DTABLE_SIZE_U32(FSE_MAX_TABLELOG)]; - -size_t FSE_decompress(void* dst, size_t dstCapacity, const void* cSrc, size_t cSrcSize) -{ - DTable_max_t dt; /* Static analyzer seems unable to understand this table will be properly initialized later */ - return FSE_decompress_wksp(dst, dstCapacity, cSrc, cSrcSize, dt, FSE_MAX_TABLELOG); -} - - - #endif /* FSE_COMMONDEFS_ONLY */ diff --git a/contrib/linux-kernel/lib/huf.h b/contrib/linux-kernel/lib/huf.h index fd26fcf27..e0a30ce00 100644 --- a/contrib/linux-kernel/lib/huf.h +++ b/contrib/linux-kernel/lib/huf.h @@ -39,36 +39,6 @@ #include /* size_t */ -/* *** simple functions *** */ -/** -HUF_compress() : - Compress content from buffer 'src', of size 'srcSize', into buffer 'dst'. - 'dst' buffer must be already allocated. - Compression runs faster if `dstCapacity` >= HUF_compressBound(srcSize). - `srcSize` must be <= `HUF_BLOCKSIZE_MAX` == 128 KB. - @return : size of compressed data (<= `dstCapacity`). - Special values : if return == 0, srcData is not compressible => Nothing is stored within dst !!! - if return == 1, srcData is a single repeated byte symbol (RLE compression). - if HUF_isError(return), compression failed (more details using HUF_getErrorName()) -*/ -size_t HUF_compress(void* dst, size_t dstCapacity, - const void* src, size_t srcSize); - -/** -HUF_decompress() : - Decompress HUF data from buffer 'cSrc', of size 'cSrcSize', - into already allocated buffer 'dst', of minimum size 'dstSize'. - `originalSize` : **must** be the ***exact*** size of original (uncompressed) data. - Note : in contrast with FSE, HUF_decompress can regenerate - RLE (cSrcSize==1) and uncompressed (cSrcSize==dstSize) data, - because it knows size to regenerate. - @return : size of regenerated data (== originalSize), - or an error code, which can be tested using HUF_isError() -*/ -size_t HUF_decompress(void* dst, size_t originalSize, - const void* cSrc, size_t cSrcSize); - - /* *** Tool functions *** */ #define HUF_BLOCKSIZE_MAX (128 * 1024) /**< maximum input size for a single block compressed with HUF_compress */ size_t HUF_compressBound(size_t size); /**< maximum compressed size (worst case) */ @@ -80,11 +50,6 @@ const char* HUF_getErrorName(size_t code); /**< provides error code string (us /* *** Advanced function *** */ -/** HUF_compress2() : - * Same as HUF_compress(), but offers direct control over `maxSymbolValue` and `tableLog` . - * `tableLog` must be `<= HUF_TABLELOG_MAX` . */ -size_t HUF_compress2 (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog); - /** HUF_compress4X_wksp() : * Same as HUF_compress2(), but uses externally allocated `workSpace`, which must be a table of >= 1024 unsigned */ size_t HUF_compress4X_wksp (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize); /**< `workSpace` must be a table of at least HUF_WORKSPACE_SIZE_U32 unsigned */ @@ -136,9 +101,6 @@ typedef U32 HUF_DTable; /* **************************************** * Advanced decompression functions ******************************************/ -size_t HUF_decompress4X2 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< single-symbol decoder */ -size_t HUF_decompress4X4 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< double-symbols decoder */ - size_t HUF_decompress4X_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< decodes RLE and uncompressed */ size_t HUF_decompress4X_hufOnly(HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< considers RLE and uncompressed as errors */ size_t HUF_decompress4X2_DCtx(HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< single-symbol decoder */ @@ -223,7 +185,6 @@ size_t HUF_decompress4X4_usingDTable(void* dst, size_t maxDstSize, const void* c /* single stream variants */ -size_t HUF_compress1X (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog); size_t HUF_compress1X_wksp (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize); /**< `workSpace` must be a table of at least HUF_WORKSPACE_SIZE_U32 unsigned */ size_t HUF_compress1X_usingCTable(void* dst, size_t dstSize, const void* src, size_t srcSize, const HUF_CElt* CTable); /** HUF_compress1X_repeat() : @@ -233,9 +194,6 @@ size_t HUF_compress1X_usingCTable(void* dst, size_t dstSize, const void* src, si * If preferRepeat then the old table will always be used if valid. */ size_t HUF_compress1X_repeat(void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize, HUF_CElt* hufTable, HUF_repeat* repeat, int preferRepeat); /**< `workSpace` must be a table of at least HUF_WORKSPACE_SIZE_U32 unsigned */ -size_t HUF_decompress1X2 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /* single-symbol decoder */ -size_t HUF_decompress1X4 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /* double-symbol decoder */ - size_t HUF_decompress1X_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); size_t HUF_decompress1X2_DCtx(HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< single-symbol decoder */ size_t HUF_decompress1X4_DCtx(HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); /**< double-symbols decoder */ diff --git a/contrib/linux-kernel/lib/huf_compress.c b/contrib/linux-kernel/lib/huf_compress.c index 16576f579..b2e71405c 100644 --- a/contrib/linux-kernel/lib/huf_compress.c +++ b/contrib/linux-kernel/lib/huf_compress.c @@ -635,14 +635,6 @@ size_t HUF_compress1X_repeat (void* dst, size_t dstSize, return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 1 /* single stream */, workSpace, wkspSize, hufTable, repeat, preferRepeat); } -size_t HUF_compress1X (void* dst, size_t dstSize, - const void* src, size_t srcSize, - unsigned maxSymbolValue, unsigned huffLog) -{ - unsigned workSpace[1024]; - return HUF_compress1X_wksp(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, workSpace, sizeof(workSpace)); -} - size_t HUF_compress4X_wksp (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned huffLog, @@ -659,16 +651,3 @@ size_t HUF_compress4X_repeat (void* dst, size_t dstSize, { return HUF_compress_internal(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, 0 /* 4 streams */, workSpace, wkspSize, hufTable, repeat, preferRepeat); } - -size_t HUF_compress2 (void* dst, size_t dstSize, - const void* src, size_t srcSize, - unsigned maxSymbolValue, unsigned huffLog) -{ - unsigned workSpace[1024]; - return HUF_compress4X_wksp(dst, dstSize, src, srcSize, maxSymbolValue, huffLog, workSpace, sizeof(workSpace)); -} - -size_t HUF_compress (void* dst, size_t maxDstSize, const void* src, size_t srcSize) -{ - return HUF_compress2(dst, maxDstSize, src, (U32)srcSize, 255, HUF_TABLELOG_DEFAULT); -} diff --git a/contrib/linux-kernel/lib/huf_decompress.c b/contrib/linux-kernel/lib/huf_decompress.c index 4ac28bec9..f73223c4e 100644 --- a/contrib/linux-kernel/lib/huf_decompress.c +++ b/contrib/linux-kernel/lib/huf_decompress.c @@ -35,12 +35,13 @@ /* ************************************************************** * Compiler specifics ****************************************************************/ -#define FORCE_INLINE static __attribute__((always_inline)) +#define FORCE_INLINE static __always_inline /* ************************************************************** * Dependencies ****************************************************************/ +#include #include /* memcpy, memset */ #include "bitstream.h" /* BIT_* */ #include "fse.h" /* header compression */ @@ -210,12 +211,6 @@ size_t HUF_decompress1X2_DCtx (HUF_DTable* DCtx, void* dst, size_t dstSize, cons return HUF_decompress1X2_usingDTable_internal (dst, dstSize, ip, cSrcSize, DCtx); } -size_t HUF_decompress1X2 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) -{ - HUF_CREATE_STATIC_DTABLEX2(DTable, HUF_TABLELOG_MAX); - return HUF_decompress1X2_DCtx (DTable, dst, dstSize, cSrc, cSrcSize); -} - static size_t HUF_decompress4X2_usingDTable_internal( void* dst, size_t dstSize, @@ -333,13 +328,6 @@ size_t HUF_decompress4X2_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, cons return HUF_decompress4X2_usingDTable_internal (dst, dstSize, ip, cSrcSize, dctx); } -size_t HUF_decompress4X2 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) -{ - HUF_CREATE_STATIC_DTABLEX2(DTable, HUF_TABLELOG_MAX); - return HUF_decompress4X2_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); -} - - /* *************************/ /* double-symbols decoding */ /* *************************/ @@ -624,12 +612,6 @@ size_t HUF_decompress1X4_DCtx (HUF_DTable* DCtx, void* dst, size_t dstSize, cons return HUF_decompress1X4_usingDTable_internal (dst, dstSize, ip, cSrcSize, DCtx); } -size_t HUF_decompress1X4 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) -{ - HUF_CREATE_STATIC_DTABLEX4(DTable, HUF_TABLELOG_MAX); - return HUF_decompress1X4_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); -} - static size_t HUF_decompress4X4_usingDTable_internal( void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize, @@ -746,12 +728,6 @@ size_t HUF_decompress4X4_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, cons return HUF_decompress4X4_usingDTable_internal(dst, dstSize, ip, cSrcSize, dctx); } -size_t HUF_decompress4X4 (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) -{ - HUF_CREATE_STATIC_DTABLEX4(DTable, HUF_TABLELOG_MAX); - return HUF_decompress4X4_DCtx(DTable, dst, dstSize, cSrc, cSrcSize); -} - /* ********************************/ /* Generic decompression selector */ @@ -818,21 +794,6 @@ U32 HUF_selectDecoder (size_t dstSize, size_t cSrcSize) typedef size_t (*decompressionAlgo)(void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize); -size_t HUF_decompress (void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) -{ - static const decompressionAlgo decompress[2] = { HUF_decompress4X2, HUF_decompress4X4 }; - - /* validation checks */ - if (dstSize == 0) return ERROR(dstSize_tooSmall); - if (cSrcSize > dstSize) return ERROR(corruption_detected); /* invalid */ - if (cSrcSize == dstSize) { memcpy(dst, cSrc, dstSize); return dstSize; } /* not compressed */ - if (cSrcSize == 1) { memset(dst, *(const BYTE*)cSrc, dstSize); return dstSize; } /* RLE */ - - { U32 const algoNb = HUF_selectDecoder(dstSize, cSrcSize); - return decompress[algoNb](dst, dstSize, cSrc, cSrcSize); - } -} - size_t HUF_decompress4X_DCtx (HUF_DTable* dctx, void* dst, size_t dstSize, const void* cSrc, size_t cSrcSize) { /* validation checks */ diff --git a/contrib/linux-kernel/lib/xxhash.c b/contrib/linux-kernel/lib/xxhash.c index 44392265d..05ebedf7f 100644 --- a/contrib/linux-kernel/lib/xxhash.c +++ b/contrib/linux-kernel/lib/xxhash.c @@ -66,9 +66,6 @@ * Includes & Memory related functions ***************************************/ /* Modify the local functions below should you wish to use some other memory routines */ -/* for malloc(), free() */ -static void* XXH_malloc(size_t s) { return malloc(s); } -static void XXH_free (void* p) { free(p); } /* for memcpy() */ #include static void* XXH_memcpy(void* dest, const void* src, size_t size) { return memcpy(dest,src,size); } @@ -80,7 +77,8 @@ static void* XXH_memcpy(void* dest, const void* src, size_t size) { return memcp /* ************************************* * Compiler Specific Options ***************************************/ -#define FORCE_INLINE static __attribute__((always_inline)) +#include +#define FORCE_INLINE static __always_inline static U32 XXH_read32(const void* memPtr) @@ -116,8 +114,10 @@ typedef enum { XXH_bigEndian=0, XXH_littleEndian=1 } XXH_endianess; *****************************/ typedef enum { XXH_aligned, XXH_unaligned } XXH_alignment; -FORCE_INLINE U32 XXH_readLE32_align(const void* ptr, XXH_endianess, XXH_alignment) +FORCE_INLINE U32 XXH_readLE32_align(const void* ptr, XXH_endianess endian, XXH_alignment align) { + (void)endian; + (void)align; return MEM_readLE32(ptr); } @@ -133,6 +133,8 @@ static U32 XXH_readBE32(const void* ptr) FORCE_INLINE U64 XXH_readLE64_align(const void* ptr, XXH_endianess endian, XXH_alignment align) { + (void)endian; + (void)align; return MEM_readLE64(ptr); } @@ -397,26 +399,6 @@ XXH_PUBLIC_API unsigned long long XXH64 (const void* input, size_t len, unsigned * Advanced Hash Functions ****************************************************/ -XXH_PUBLIC_API XXH32_state_t* XXH32_createState(void) -{ - return (XXH32_state_t*)XXH_malloc(sizeof(XXH32_state_t)); -} -XXH_PUBLIC_API XXH_errorcode XXH32_freeState(XXH32_state_t* statePtr) -{ - XXH_free(statePtr); - return XXH_OK; -} - -XXH_PUBLIC_API XXH64_state_t* XXH64_createState(void) -{ - return (XXH64_state_t*)XXH_malloc(sizeof(XXH64_state_t)); -} -XXH_PUBLIC_API XXH_errorcode XXH64_freeState(XXH64_state_t* statePtr) -{ - XXH_free(statePtr); - return XXH_OK; -} - /*** Hash feed ***/ diff --git a/contrib/linux-kernel/lib/xxhash.h b/contrib/linux-kernel/lib/xxhash.h index 6c4c79c42..36d140eca 100644 --- a/contrib/linux-kernel/lib/xxhash.h +++ b/contrib/linux-kernel/lib/xxhash.h @@ -140,14 +140,6 @@ XXH64() : typedef struct XXH32_state_s XXH32_state_t; /* incomplete type */ typedef struct XXH64_state_s XXH64_state_t; /* incomplete type */ -/*! State allocation, compatible with dynamic libraries */ - -XXH_PUBLIC_API XXH32_state_t* XXH32_createState(void); -XXH_PUBLIC_API XXH_errorcode XXH32_freeState(XXH32_state_t* statePtr); - -XXH_PUBLIC_API XXH64_state_t* XXH64_createState(void); -XXH_PUBLIC_API XXH_errorcode XXH64_freeState(XXH64_state_t* statePtr); - /* hash streaming */ diff --git a/contrib/linux-kernel/lib/zstd_common.c b/contrib/linux-kernel/lib/zstd_common.c index 0788b6ef6..c62cbd365 100644 --- a/contrib/linux-kernel/lib/zstd_common.c +++ b/contrib/linux-kernel/lib/zstd_common.c @@ -13,7 +13,8 @@ * Dependencies ***************************************/ #include "error_private.h" -#include "zstd.h" /* declaration of ZSTD_isError, ZSTD_getErrorName, ZSTD_getErrorCode, ZSTD_getErrorString, ZSTD_versionNumber */ +#include "zstd_internal.h" /* declaration of ZSTD_isError, ZSTD_getErrorName, ZSTD_getErrorCode, ZSTD_getErrorString, ZSTD_versionNumber */ +#include /*-**************************************** @@ -22,41 +23,38 @@ unsigned ZSTD_versionNumber (void) { return ZSTD_VERSION_NUMBER; } -/*-**************************************** -* ZSTD Error Management -******************************************/ -/*! ZSTD_isError() : -* tells if a return value is an error code */ -unsigned ZSTD_isError(size_t code) { return ERR_isError(code); } - -/*! ZSTD_getErrorName() : -* provides error code string from function result (useful for debugging) */ -const char* ZSTD_getErrorName(size_t code) { return ERR_getErrorName(code); } - -/*! ZSTD_getError() : -* convert a `size_t` function result into a proper ZSTD_errorCode enum */ -ZSTD_ErrorCode ZSTD_getErrorCode(size_t code) { return ERR_getErrorCode(code); } - -/*! ZSTD_getErrorString() : -* provides error code string from enum */ -const char* ZSTD_getErrorString(ZSTD_ErrorCode code) { return ERR_getErrorString(code); } - - /*=************************************************************** * Custom allocator ****************************************************************/ -/* default uses stdlib */ -void* ZSTD_defaultAllocFunction(void* opaque, size_t size) -{ - void* address = malloc(size); - (void)opaque; - return address; + +#define stack_push(stack, size) ({ \ + void* const ptr = ZSTD_PTR_ALIGN((stack)->ptr); \ + (stack)->ptr = (char*)ptr + (size); \ + (stack)->ptr <= (stack)->end ? ptr : NULL; \ + }) + +ZSTD_customMem ZSTD_initStack(void* workspace, size_t workspaceSize) { + ZSTD_customMem stackMem = { ZSTD_stackAlloc, ZSTD_stackFree, workspace }; + ZSTD_stack* stack = (ZSTD_stack*) workspace; + /* Verify preconditions */ + if (!workspace || workspaceSize < sizeof(ZSTD_stack) || workspace != ZSTD_PTR_ALIGN(workspace)) { + ZSTD_customMem error = {NULL, NULL, NULL}; + return error; + } + /* Initialize the stack */ + stack->ptr = workspace; + stack->end = (char*)workspace + workspaceSize; + stack_push(stack, sizeof(ZSTD_stack)); + return stackMem; } -void ZSTD_defaultFreeFunction(void* opaque, void* address) -{ +void* ZSTD_stackAlloc(void* opaque, size_t size) { + ZSTD_stack* stack = (ZSTD_stack*)opaque; + return stack_push(stack, size); +} +void ZSTD_stackFree(void* opaque, void* address) { (void)opaque; - free(address); + (void)address; } void* ZSTD_malloc(size_t size, ZSTD_customMem customMem) diff --git a/contrib/linux-kernel/lib/zstd_compress.c b/contrib/linux-kernel/lib/zstd_compress.c index 79b7699b8..df258c636 100644 --- a/contrib/linux-kernel/lib/zstd_compress.c +++ b/contrib/linux-kernel/lib/zstd_compress.c @@ -11,6 +11,7 @@ /*-************************************* * Dependencies ***************************************/ +#include #include /* memset */ #include "mem.h" #include "fse.h" @@ -84,18 +85,26 @@ struct ZSTD_CCtx_s { unsigned tmpCounters[HUF_WORKSPACE_SIZE_U32]; }; -ZSTD_CCtx* ZSTD_createCCtx(void) -{ - return ZSTD_createCCtx_advanced(defaultCustomMem); +size_t ZSTD_CCtxWorkspaceBound(ZSTD_compressionParameters cParams) { + size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << cParams.windowLog); + U32 const divider = (cParams.searchLength==3) ? 3 : 4; + size_t const maxNbSeq = blockSize / divider; + size_t const tokenSpace = blockSize + 11*maxNbSeq; + size_t const chainSize = (cParams.strategy == ZSTD_fast) ? 0 : (1 << cParams.chainLog); + size_t const hSize = ((size_t)1) << cParams.hashLog; + U32 const hashLog3 = (cParams.searchLength>3) ? 0 : MIN(ZSTD_HASHLOG3_MAX, cParams.windowLog); + size_t const h3Size = ((size_t)1) << hashLog3; + size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); + size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<workSpaceSize; -} - -size_t ZSTD_setCCtxParameter(ZSTD_CCtx* cctx, ZSTD_CCtxParameter param, unsigned value) -{ - switch(param) - { - case ZSTD_p_forceWindow : cctx->forceWindow = value>0; cctx->loadedDictEnd = 0; return 0; - case ZSTD_p_forceRawDict : cctx->forceRawDict = value>0; return 0; - default: return ERROR(parameter_unknown); - } -} - const seqStore_t* ZSTD_getSeqStore(const ZSTD_CCtx* ctx) /* hidden interface */ { return &(ctx->seqStore); @@ -191,28 +190,6 @@ ZSTD_compressionParameters ZSTD_adjustCParams(ZSTD_compressionParameters cPar, u } -size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams) -{ - size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << cParams.windowLog); - U32 const divider = (cParams.searchLength==3) ? 3 : 4; - size_t const maxNbSeq = blockSize / divider; - size_t const tokenSpace = blockSize + 11*maxNbSeq; - - size_t const chainSize = (cParams.strategy == ZSTD_fast) ? 0 : (1 << cParams.chainLog); - size_t const hSize = ((size_t)1) << cParams.hashLog; - U32 const hashLog3 = (cParams.searchLength>3) ? 0 : MIN(ZSTD_HASHLOG3_MAX, cParams.windowLog); - size_t const h3Size = ((size_t)1) << hashLog3; - size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32); - - size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1< /* fprintf */ U32 g_startDebug = 0; const BYTE* g_start = NULL; #endif @@ -2770,37 +2746,15 @@ static size_t ZSTD_compress_internal (ZSTD_CCtx* cctx, return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize); } -size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict,size_t dictSize, - ZSTD_parameters params) +size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void* dict, size_t dictSize, ZSTD_parameters params) { - CHECK_F(ZSTD_checkCParams(params.cParams)); return ZSTD_compress_internal(ctx, dst, dstCapacity, src, srcSize, dict, dictSize, params); } -size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void* dict, size_t dictSize, int compressionLevel) -{ - ZSTD_parameters params = ZSTD_getParams(compressionLevel, srcSize, dict ? dictSize : 0); - params.fParams.contentSizeFlag = 1; - return ZSTD_compress_internal(ctx, dst, dstCapacity, src, srcSize, dict, dictSize, params); -} -size_t ZSTD_compressCCtx (ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel) +size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, ZSTD_parameters params) { - return ZSTD_compress_usingDict(ctx, dst, dstCapacity, src, srcSize, NULL, 0, compressionLevel); -} - -size_t ZSTD_compress(void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel) -{ - size_t result; - ZSTD_CCtx ctxBody; - memset(&ctxBody, 0, sizeof(ctxBody)); - memcpy(&ctxBody.customMem, &defaultCustomMem, sizeof(ZSTD_customMem)); - result = ZSTD_compressCCtx(&ctxBody, dst, dstCapacity, src, srcSize, compressionLevel); - ZSTD_free(ctxBody.workSpace, defaultCustomMem); /* can't free ctxBody itself, as it's on stack; free only heap content */ - return result; + return ZSTD_compress_internal(ctx, dst, dstCapacity, src, srcSize, NULL, 0, params); } @@ -2813,16 +2767,14 @@ struct ZSTD_CDict_s { ZSTD_CCtx* refContext; }; /* typedef'd tp ZSTD_CDict within "zstd.h" */ -size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict) +size_t ZSTD_CDictWorkspaceBound(ZSTD_compressionParameters cParams) { - if (cdict==NULL) return 0; /* support sizeof on NULL */ - return ZSTD_sizeof_CCtx(cdict->refContext) + (cdict->dictBuffer ? cdict->dictContentSize : 0) + sizeof(*cdict); + return ZSTD_CCtxWorkspaceBound(cParams) + ZSTD_ALIGN(sizeof(ZSTD_CDict)); } -ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, unsigned byReference, - ZSTD_parameters params, ZSTD_customMem customMem) +static ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, unsigned byReference, + ZSTD_parameters params, ZSTD_customMem customMem) { - if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; if (!customMem.customAlloc || !customMem.customFree) return NULL; { ZSTD_CDict* const cdict = (ZSTD_CDict*) ZSTD_malloc(sizeof(ZSTD_CDict), customMem); @@ -2859,20 +2811,10 @@ ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, u } } -ZSTD_CDict* ZSTD_createCDict(const void* dict, size_t dictSize, int compressionLevel) +ZSTD_CDict* ZSTD_createCDict(const void* dict, size_t dictSize, ZSTD_parameters params, void* workspace, size_t workspaceSize) { - ZSTD_customMem const allocator = { NULL, NULL, NULL }; - ZSTD_parameters params = ZSTD_getParams(compressionLevel, 0, dictSize); - params.fParams.contentSizeFlag = 1; - return ZSTD_createCDict_advanced(dict, dictSize, 0, params, allocator); -} - -ZSTD_CDict* ZSTD_createCDict_byReference(const void* dict, size_t dictSize, int compressionLevel) -{ - ZSTD_customMem const allocator = { NULL, NULL, NULL }; - ZSTD_parameters params = ZSTD_getParams(compressionLevel, 0, dictSize); - params.fParams.contentSizeFlag = 1; - return ZSTD_createCDict_advanced(dict, dictSize, 1, params, allocator); + ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize); + return ZSTD_createCDict_advanced(dict, dictSize, 1, params, stackMem); } size_t ZSTD_freeCDict(ZSTD_CDict* cdict) @@ -2951,16 +2893,19 @@ struct ZSTD_CStream_s { ZSTD_customMem customMem; }; /* typedef'd to ZSTD_CStream within "zstd.h" */ -ZSTD_CStream* ZSTD_createCStream(void) +size_t ZSTD_CStreamWorkspaceBound(ZSTD_compressionParameters cParams) { - return ZSTD_createCStream_advanced(defaultCustomMem); + size_t const inBuffSize = (size_t)1 << cParams.windowLog; + size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, inBuffSize); + size_t const outBuffSize = ZSTD_compressBound(blockSize) + 1; + + return ZSTD_CCtxWorkspaceBound(cParams) + ZSTD_ALIGN(sizeof(ZSTD_CStream)) + ZSTD_ALIGN(inBuffSize) + ZSTD_ALIGN(outBuffSize); } ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem) { ZSTD_CStream* zcs; - if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; if (!customMem.customAlloc || !customMem.customFree) return NULL; zcs = (ZSTD_CStream*)ZSTD_malloc(sizeof(ZSTD_CStream), customMem); @@ -2972,6 +2917,12 @@ ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem) return zcs; } +ZSTD_CStream* ZSTD_createCStream(void* workspace, size_t workspaceSize) +{ + ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize); + return ZSTD_createCStream_advanced(stackMem); +} + size_t ZSTD_freeCStream(ZSTD_CStream* zcs) { if (zcs==NULL) return 0; /* support free on NULL */ @@ -3021,7 +2972,7 @@ size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize) return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } -size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, +static size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize) { @@ -3056,39 +3007,19 @@ size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, } /* note : cdict must outlive compression session */ -size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) +size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, ZSTD_parameters params, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize) { - ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict); - size_t const initError = ZSTD_initCStream_advanced(zcs, NULL, 0, params, 0); + size_t const initError = ZSTD_initCStream_advanced(zcs, NULL, 0, params, pledgedSrcSize); zcs->cdict = cdict; zcs->cctx->dictID = params.fParams.noDictIDFlag ? 0 : cdict->refContext->dictID; return initError; } -size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel) +size_t ZSTD_initCStream(ZSTD_CStream* zcs, ZSTD_parameters params, unsigned long long pledgedSrcSize) { - ZSTD_parameters const params = ZSTD_getParams(compressionLevel, 0, dictSize); - return ZSTD_initCStream_advanced(zcs, dict, dictSize, params, 0); -} - -size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize) -{ - ZSTD_parameters params = ZSTD_getParams(compressionLevel, pledgedSrcSize, 0); - if (pledgedSrcSize) params.fParams.contentSizeFlag = 1; return ZSTD_initCStream_advanced(zcs, NULL, 0, params, pledgedSrcSize); } -size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel) -{ - return ZSTD_initCStream_usingDict(zcs, NULL, 0, compressionLevel); -} - -size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs) -{ - if (zcs==NULL) return 0; /* support sizeof on NULL */ - return sizeof(*zcs) + ZSTD_sizeof_CCtx(zcs->cctx) + ZSTD_sizeof_CDict(zcs->cdictLocal) + zcs->outBuffSize + zcs->inBuffSize; -} - /*====== Compression ======*/ typedef enum { zsf_gather, zsf_flush, zsf_end } ZSTD_flush_e; diff --git a/contrib/linux-kernel/lib/zstd_decompress.c b/contrib/linux-kernel/lib/zstd_decompress.c index 10700257b..0d512e8f6 100644 --- a/contrib/linux-kernel/lib/zstd_decompress.c +++ b/contrib/linux-kernel/lib/zstd_decompress.c @@ -90,9 +90,10 @@ struct ZSTD_DCtx_s BYTE headerBuffer[ZSTD_FRAMEHEADERSIZE_MAX]; }; /* typedef'd to ZSTD_DCtx within "zstd.h" */ -size_t ZSTD_sizeof_DCtx (const ZSTD_DCtx* dctx) { return (dctx==NULL) ? 0 : sizeof(ZSTD_DCtx); } - -size_t ZSTD_estimateDCtxSize(void) { return sizeof(ZSTD_DCtx); } +size_t ZSTD_DCtxWorkspaceBound(void) +{ + return ZSTD_ALIGN(sizeof(ZSTD_stack)) + ZSTD_ALIGN(sizeof(ZSTD_DCtx)); +} size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx) { @@ -118,7 +119,6 @@ ZSTD_DCtx* ZSTD_createDCtx_advanced(ZSTD_customMem customMem) { ZSTD_DCtx* dctx; - if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; if (!customMem.customAlloc || !customMem.customFree) return NULL; dctx = (ZSTD_DCtx*)ZSTD_malloc(sizeof(ZSTD_DCtx), customMem); @@ -128,9 +128,10 @@ ZSTD_DCtx* ZSTD_createDCtx_advanced(ZSTD_customMem customMem) return dctx; } -ZSTD_DCtx* ZSTD_createDCtx(void) +ZSTD_DCtx* ZSTD_createDCtx(void* workspace, size_t workspaceSize) { - return ZSTD_createDCtx_advanced(defaultCustomMem); + ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize); + return ZSTD_createDCtx_advanced(stackMem); } size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx) @@ -354,20 +355,6 @@ unsigned long long ZSTD_findDecompressedSize(const void* src, size_t srcSize) } } -/** ZSTD_getDecompressedSize() : -* compatible with legacy mode -* @return : decompressed size if known, 0 otherwise - note : 0 can mean any of the following : - - decompressed size is not present within frame header - - frame header unknown / not supported - - frame header not complete (`srcSize` too small) */ -unsigned long long ZSTD_getDecompressedSize(const void* src, size_t srcSize) -{ - unsigned long long const ret = ZSTD_getFrameContentSize(src, srcSize); - return ret >= ZSTD_CONTENTSIZE_ERROR ? 0 : ret; -} - - /** ZSTD_decodeFrameHeader() : * `headerSize` must be the size provided by ZSTD_frameHeaderSize(). * @return : 0 if success, or an error code, which can be tested using ZSTD_isError() */ @@ -1620,10 +1607,7 @@ static size_t ZSTD_decompressMultiFrame(ZSTD_DCtx* dctx, return (BYTE*)dst - (BYTE*)dststart; } -size_t ZSTD_decompress_usingDict(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const void* dict, size_t dictSize) +size_t ZSTD_decompress_usingDict(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void *dict, size_t dictSize) { return ZSTD_decompressMultiFrame(dctx, dst, dstCapacity, src, srcSize, dict, dictSize, NULL); } @@ -1635,13 +1619,6 @@ size_t ZSTD_decompressDCtx(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const } -size_t ZSTD_decompress(void* dst, size_t dstCapacity, const void* src, size_t srcSize) -{ - ZSTD_DCtx dctx; - return ZSTD_decompressDCtx(&dctx, dst, dstCapacity, src, srcSize); -} - - /*-************************************** * Advanced Streaming Decompression API * Bufferless and synchronous @@ -1900,6 +1877,11 @@ struct ZSTD_DDict_s { ZSTD_customMem cMem; }; /* typedef'd to ZSTD_DDict within "zstd.h" */ +size_t ZSTD_DDictWorkspaceBound(void) +{ + return ZSTD_ALIGN(sizeof(ZSTD_stack)) + ZSTD_ALIGN(sizeof(ZSTD_DDict)); +} + static const void* ZSTD_DDictDictContent(const ZSTD_DDict* ddict) { return ddict->dictContent; @@ -1953,9 +1935,8 @@ static size_t ZSTD_loadEntropy_inDDict(ZSTD_DDict* ddict) } -ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize, unsigned byReference, ZSTD_customMem customMem) +static ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize, unsigned byReference, ZSTD_customMem customMem) { - if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; if (!customMem.customAlloc || !customMem.customFree) return NULL; { ZSTD_DDict* const ddict = (ZSTD_DDict*) ZSTD_malloc(sizeof(ZSTD_DDict), customMem); @@ -1989,21 +1970,10 @@ ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize, unsigne * Create a digested dictionary, to start decompression without startup delay. * `dict` content is copied inside DDict. * Consequently, `dict` can be released after `ZSTD_DDict` creation */ -ZSTD_DDict* ZSTD_createDDict(const void* dict, size_t dictSize) +ZSTD_DDict* ZSTD_createDDict(const void* dict, size_t dictSize, void* workspace, size_t workspaceSize) { - ZSTD_customMem const allocator = { NULL, NULL, NULL }; - return ZSTD_createDDict_advanced(dict, dictSize, 0, allocator); -} - - -/*! ZSTD_createDDict_byReference() : - * Create a digested dictionary, to start decompression without startup delay. - * Dictionary content is simply referenced, it will be accessed during decompression. - * Warning : dictBuffer must outlive DDict (DDict must be freed before dictBuffer) */ -ZSTD_DDict* ZSTD_createDDict_byReference(const void* dictBuffer, size_t dictSize) -{ - ZSTD_customMem const allocator = { NULL, NULL, NULL }; - return ZSTD_createDDict_advanced(dictBuffer, dictSize, 1, allocator); + ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize); + return ZSTD_createDDict_advanced(dict, dictSize, 1, stackMem); } @@ -2017,12 +1987,6 @@ size_t ZSTD_freeDDict(ZSTD_DDict* ddict) } } -size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict) -{ - if (ddict==NULL) return 0; /* support sizeof on NULL */ - return sizeof(*ddict) + (ddict->dictBuffer ? ddict->dictSize : 0) ; -} - /*! ZSTD_getDictID_fromDict() : * Provides the dictID stored within dictionary. * if @return == 0, the dictionary is not conformant with Zstandard specification. @@ -2110,17 +2074,17 @@ struct ZSTD_DStream_s { U32 hostageByte; }; /* typedef'd to ZSTD_DStream within "zstd.h" */ - -ZSTD_DStream* ZSTD_createDStream(void) -{ - return ZSTD_createDStream_advanced(defaultCustomMem); +size_t ZSTD_DStreamWorkspaceBound(size_t maxWindowSize) { + size_t const blockSize = MIN(maxWindowSize, ZSTD_BLOCKSIZE_ABSOLUTEMAX); + size_t const inBuffSize = blockSize; + size_t const outBuffSize = maxWindowSize + blockSize + WILDCOPY_OVERLENGTH * 2; + return ZSTD_DCtxWorkspaceBound() + ZSTD_ALIGN(sizeof(ZSTD_DStream)) + ZSTD_ALIGN(inBuffSize) + ZSTD_ALIGN(outBuffSize); } -ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem) +static ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem) { ZSTD_DStream* zds; - if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; if (!customMem.customAlloc || !customMem.customFree) return NULL; zds = (ZSTD_DStream*) ZSTD_malloc(sizeof(ZSTD_DStream), customMem); @@ -2134,6 +2098,12 @@ ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem) return zds; } +ZSTD_DStream* ZSTD_createDStream(void* workspace, size_t workspaceSize) +{ + ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize); + return ZSTD_createDStream_advanced(stackMem); +} + size_t ZSTD_freeDStream(ZSTD_DStream* zds) { if (zds==NULL) return 0; /* support free on null */ @@ -2157,29 +2127,22 @@ size_t ZSTD_freeDStream(ZSTD_DStream* zds) size_t ZSTD_DStreamInSize(void) { return ZSTD_BLOCKSIZE_ABSOLUTEMAX + ZSTD_blockHeaderSize; } size_t ZSTD_DStreamOutSize(void) { return ZSTD_BLOCKSIZE_ABSOLUTEMAX; } -size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize) +size_t ZSTD_initDStream(ZSTD_DStream* zds, size_t maxWindowSize) { + zds->maxWindowSize = maxWindowSize; zds->stage = zdss_loadHeader; zds->lhSize = zds->inPos = zds->outStart = zds->outEnd = 0; ZSTD_freeDDict(zds->ddictLocal); - if (dict && dictSize >= 8) { - zds->ddictLocal = ZSTD_createDDict(dict, dictSize); - if (zds->ddictLocal == NULL) return ERROR(memory_allocation); - } else zds->ddictLocal = NULL; + zds->ddictLocal = NULL; zds->ddict = zds->ddictLocal; zds->legacyVersion = 0; zds->hostageByte = 0; return ZSTD_frameHeaderSize_prefix; } -size_t ZSTD_initDStream(ZSTD_DStream* zds) +size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, size_t maxWindowSize, const ZSTD_DDict* ddict) /**< note : ddict will just be referenced, and must outlive decompression session */ { - return ZSTD_initDStream_usingDict(zds, NULL, 0); -} - -size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, const ZSTD_DDict* ddict) /**< note : ddict will just be referenced, and must outlive decompression session */ -{ - size_t const initResult = ZSTD_initDStream(zds); + size_t const initResult = ZSTD_initDStream(zds, maxWindowSize); zds->ddict = ddict; return initResult; } @@ -2193,25 +2156,6 @@ size_t ZSTD_resetDStream(ZSTD_DStream* zds) return ZSTD_frameHeaderSize_prefix; } -size_t ZSTD_setDStreamParameter(ZSTD_DStream* zds, - ZSTD_DStreamParameter_e paramType, unsigned paramValue) -{ - switch(paramType) - { - default : return ERROR(parameter_unknown); - case DStream_p_maxWindowSize : zds->maxWindowSize = paramValue ? paramValue : (U32)(-1); break; - } - return 0; -} - - -size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds) -{ - if (zds==NULL) return 0; /* support sizeof on NULL */ - return sizeof(*zds) + ZSTD_sizeof_DCtx(zds->dctx) + ZSTD_sizeof_DDict(zds->ddictLocal) + zds->inBuffSize + zds->outBuffSize; -} - - /* ***** Decompression ***** */ MEM_STATIC size_t ZSTD_limitCopy(void* dst, size_t dstCapacity, const void* src, size_t srcSize) diff --git a/contrib/linux-kernel/lib/zstd_errors.h b/contrib/linux-kernel/lib/zstd_errors.h deleted file mode 100644 index 37e491b7e..000000000 --- a/contrib/linux-kernel/lib/zstd_errors.h +++ /dev/null @@ -1,55 +0,0 @@ -/** - * Copyright (c) 2016-present, Yann Collet, Facebook, Inc. - * All rights reserved. - * - * This source code is licensed under the BSD-style license found in the - * LICENSE file in the root directory of this source tree. An additional grant - * of patent rights can be found in the PATENTS file in the same directory. - */ - -#ifndef ZSTD_ERRORS_H_398273423 -#define ZSTD_ERRORS_H_398273423 - -/*===== dependency =====*/ -#include /* size_t */ - - -/* ===== ZSTDERRORLIB_API : control library symbols visibility ===== */ -#define ZSTDERRORLIB_API - -/*-**************************************** -* error codes list -******************************************/ -typedef enum { - ZSTD_error_no_error, - ZSTD_error_GENERIC, - ZSTD_error_prefix_unknown, - ZSTD_error_version_unsupported, - ZSTD_error_parameter_unknown, - ZSTD_error_frameParameter_unsupported, - ZSTD_error_frameParameter_unsupportedBy32bits, - ZSTD_error_frameParameter_windowTooLarge, - ZSTD_error_compressionParameter_unsupported, - ZSTD_error_init_missing, - ZSTD_error_memory_allocation, - ZSTD_error_stage_wrong, - ZSTD_error_dstSize_tooSmall, - ZSTD_error_srcSize_wrong, - ZSTD_error_corruption_detected, - ZSTD_error_checksum_wrong, - ZSTD_error_tableLog_tooLarge, - ZSTD_error_maxSymbolValue_tooLarge, - ZSTD_error_maxSymbolValue_tooSmall, - ZSTD_error_dictionary_corrupted, - ZSTD_error_dictionary_wrong, - ZSTD_error_dictionaryCreation_failed, - ZSTD_error_maxCode -} ZSTD_ErrorCode; - -/*! ZSTD_getErrorCode() : - convert a `size_t` function result into a `ZSTD_ErrorCode` enum type, - which can be used to compare directly with enum list published into "error_public.h" */ -ZSTDERRORLIB_API ZSTD_ErrorCode ZSTD_getErrorCode(size_t functionResult); -ZSTDERRORLIB_API const char* ZSTD_getErrorString(ZSTD_ErrorCode code); - -#endif /* ZSTD_ERRORS_H_398273423 */ diff --git a/contrib/linux-kernel/lib/zstd_internal.h b/contrib/linux-kernel/lib/zstd_internal.h index 57ae77390..b7dcc03e0 100644 --- a/contrib/linux-kernel/lib/zstd_internal.h +++ b/contrib/linux-kernel/lib/zstd_internal.h @@ -13,13 +13,15 @@ /*-******************************************************* * Compiler specifics *********************************************************/ -#define FORCE_INLINE static __attribute__((always_inline)) -#define FORCE_NOINLINE static __attribute__((__noinline__)) +#define FORCE_INLINE static __always_inline +#define FORCE_NOINLINE static noinline /*-************************************* * Dependencies ***************************************/ +#include +#include #include "mem.h" #include "error_private.h" #include "zstd.h" @@ -207,25 +209,35 @@ const seqStore_t* ZSTD_getSeqStore(const ZSTD_CCtx* ctx); void ZSTD_seqToCodes(const seqStore_t* seqStorePtr); int ZSTD_isSkipFrame(ZSTD_DCtx* dctx); -/* custom memory allocation functions */ -void* ZSTD_defaultAllocFunction(void* opaque, size_t size); -void ZSTD_defaultFreeFunction(void* opaque, void* address); -#ifndef ZSTD_DLL_IMPORT -static const ZSTD_customMem defaultCustomMem = { ZSTD_defaultAllocFunction, ZSTD_defaultFreeFunction, NULL }; -#endif +/*= Custom memory allocation functions */ +typedef void* (*ZSTD_allocFunction) (void* opaque, size_t size); +typedef void (*ZSTD_freeFunction) (void* opaque, void* address); +typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; void* opaque; } ZSTD_customMem; + void* ZSTD_malloc(size_t size, ZSTD_customMem customMem); void ZSTD_free(void* ptr, ZSTD_customMem customMem); +/*====== stack allocation ======*/ + +typedef struct { + void* ptr; + const void* end; +} ZSTD_stack; + +#define ZSTD_ALIGN(x) ALIGN(x, sizeof(size_t)) +#define ZSTD_PTR_ALIGN(p) PTR_ALIGN(p, sizeof(size_t)) + +ZSTD_customMem ZSTD_initStack(void* workspace, size_t workspaceSize); + +void* ZSTD_stackAlloc(void* opaque, size_t size); +void ZSTD_stackFree(void* opaque, void* address); + /*====== common function ======*/ MEM_STATIC U32 ZSTD_highbit32(U32 val) { -# if defined(_MSC_VER) /* Visual */ - unsigned long r=0; - _BitScanReverse(&r, val); - return (unsigned)r; -# elif defined(__GNUC__) && (__GNUC__ >= 3) /* GCC Intrinsic */ +# if defined(__GNUC__) && (__GNUC__ >= 3) /* GCC Intrinsic */ return 31 - __builtin_clz(val); # else /* Software version */ static const int DeBruijnClz[32] = { 0, 9, 1, 10, 13, 21, 2, 29, 11, 14, 16, 18, 22, 25, 3, 30, 8, 12, 20, 28, 15, 17, 24, 7, 19, 27, 23, 6, 26, 5, 4, 31 }; @@ -250,5 +262,12 @@ MEM_STATIC U32 ZSTD_highbit32(U32 val) * do not use with extDict variant ! */ void ZSTD_invalidateRepCodes(ZSTD_CCtx* cctx); +size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx); +size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx); +size_t ZSTD_freeCDict(ZSTD_CDict* cdict); +size_t ZSTD_freeDDict(ZSTD_DDict* cdict); +size_t ZSTD_freeCStream(ZSTD_CStream* zcs); +size_t ZSTD_freeDStream(ZSTD_DStream* zds); + #endif /* ZSTD_CCOMMON_H_MODULE */ From 30c76989704975b3fe7f3561d751a5ba24d838e8 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 31 Mar 2017 18:27:03 -0700 Subject: [PATCH 110/305] optimize ZSTDMT_compress() memory usage does no longer allocate temporary buffers when there is enough room in dstBuffer to decompress directly there. (previous method would skip that for 1st chunk only). Also : fix ZSTD_compressBound() for small srcSize --- lib/compress/zstd_compress.c | 3 ++- lib/compress/zstdmt_compress.c | 19 ++++++++++++------- tests/zstreamtest.c | 5 +++-- 3 files changed, 17 insertions(+), 10 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index cc69f1184..e1c734927 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -33,7 +33,8 @@ typedef enum { ZSTDcs_created=0, ZSTDcs_init, ZSTDcs_ongoing, ZSTDcs_ending } ZS ***************************************/ #define ZSTD_STATIC_ASSERT(c) { enum { ZSTD_static_assert = 1/(int)(!!(c)) }; } size_t ZSTD_compressBound(size_t srcSize) { - size_t const margin = (srcSize < 512 KB) ? 16 : 0; + size_t const lowLimit = 256 KB; + size_t const margin = (srcSize < lowLimit) ? (lowLimit-srcSize) >> 12 : 0; /* from 64 to 0 */ return srcSize + (srcSize >> 8) + margin; } diff --git a/lib/compress/zstdmt_compress.c b/lib/compress/zstdmt_compress.c index 09a965478..c9c3e8533 100644 --- a/lib/compress/zstdmt_compress.c +++ b/lib/compress/zstdmt_compress.c @@ -33,7 +33,7 @@ # include # include # include - static unsigned g_debugLevel = 2; + static unsigned g_debugLevel = 5; # define DEBUGLOGRAW(l, ...) if (l<=g_debugLevel) { fprintf(stderr, __VA_ARGS__); } # define DEBUGLOG(l, ...) if (l<=g_debugLevel) { fprintf(stderr, __FILE__ ": "); fprintf(stderr, __VA_ARGS__); fprintf(stderr, " \n"); } @@ -253,6 +253,7 @@ void ZSTDMT_compressChunk(void* jobDescription) ZSTD_compressContinue(job->cctx, dstBuff.start, dstBuff.size, src, job->srcSize); DEBUGLOG(3, "compressed %u bytes into %u bytes (first:%u) (last:%u)", (unsigned)job->srcSize, (unsigned)job->cSize, job->firstChunk, job->lastChunk); + DEBUGLOG(5, "dstBuff.size : %u ; => %s", (U32)dstBuff.size, ZSTD_getErrorName(job->cSize)); _endJob: PTHREAD_MUTEX_LOCK(job->jobCompleted_mutex); @@ -399,7 +400,8 @@ size_t ZSTDMT_compressCCtx(ZSTDMT_CCtx* mtctx, size_t const avgChunkSize = ((proposedChunkSize & 0x1FFFF) < 0xFFFF) ? proposedChunkSize + 0xFFFF : proposedChunkSize; /* avoid too small last block */ size_t remainingSrcSize = srcSize; const char* const srcStart = (const char*)src; - size_t frameStartPos = 0; + unsigned const compressWithinDst = (dstCapacity >= ZSTD_compressBound(srcSize)) ? nbChunks : (unsigned)(dstCapacity / ZSTD_compressBound(avgChunkSize)); /* presumes avgChunkSize >= 256 KB, which should be the case */ + size_t frameStartPos = 0, dstBufferPos = 0; DEBUGLOG(3, "windowLog : %2u => chunkTargetSize : %u bytes ", params.cParams.windowLog, (U32)chunkTargetSize); DEBUGLOG(2, "nbChunks : %2u (chunkSize : %u bytes) ", nbChunks, (U32)avgChunkSize); @@ -413,9 +415,9 @@ size_t ZSTDMT_compressCCtx(ZSTDMT_CCtx* mtctx, { unsigned u; for (u=0; ubuffPool, dstBufferCapacity) : dstAsBuffer; + size_t const dstBufferCapacity = ZSTD_compressBound(chunkSize); + buffer_t const dstAsBuffer = { (char*)dst + dstBufferPos, dstBufferCapacity }; + buffer_t const dstBuffer = u < compressWithinDst ? dstAsBuffer : ZSTDMT_getBuffer(mtctx->buffPool, dstBufferCapacity); ZSTD_CCtx* const cctx = ZSTDMT_getCCtx(mtctx->cctxPool); size_t dictSize = u ? overlapSize : 0; @@ -444,6 +446,7 @@ size_t ZSTDMT_compressCCtx(ZSTDMT_CCtx* mtctx, POOL_add(mtctx->factory, ZSTDMT_compressChunk, &mtctx->jobs[u]); frameStartPos += chunkSize; + dstBufferPos += dstBufferCapacity; remainingSrcSize -= chunkSize; } } /* note : since nbChunks <= nbThreads, all jobs should be running immediately in parallel */ @@ -467,8 +470,10 @@ size_t ZSTDMT_compressCCtx(ZSTDMT_CCtx* mtctx, if (ZSTD_isError(cSize)) error = cSize; if ((!error) && (dstPos + cSize > dstCapacity)) error = ERROR(dstSize_tooSmall); if (chunkID) { /* note : chunk 0 is already written directly into dst */ - if (!error) memcpy((char*)dst + dstPos, mtctx->jobs[chunkID].dstBuff.start, cSize); - ZSTDMT_releaseBuffer(mtctx->buffPool, mtctx->jobs[chunkID].dstBuff); + if (!error) + memmove((char*)dst + dstPos, mtctx->jobs[chunkID].dstBuff.start, cSize); /* may overlap if chunk decompressed within dst */ + if (chunkID >= compressWithinDst) /* otherwise, it decompresses within dst */ + ZSTDMT_releaseBuffer(mtctx->buffPool, mtctx->jobs[chunkID].dstBuff); mtctx->jobs[chunkID].dstBuff = g_nullBuffer; } dstPos += cSize ; diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index 370859ca9..10bcbe0cd 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -856,6 +856,7 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp /* some issues can only happen when reusing states */ if ((FUZ_rand(&lseed) & 0xFF) == 131) { U32 const nbThreads = (FUZ_rand(&lseed) % 6) + 1; + DISPLAYLEVEL(5, "Creating new context with %u threads \n", nbThreads); ZSTDMT_freeCCtx(zc); zc = ZSTDMT_createCCtx(nbThreads); resetAllowed=0; @@ -946,7 +947,7 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp outBuff.size = outBuff.pos + adjustedDstSize; DISPLAYLEVEL(5, "Flushing into dst buffer of size %u \n", (U32)adjustedDstSize); { size_t const flushError = ZSTDMT_flushStream(zc, &outBuff); - CHECK (ZSTD_isError(flushError), "flush error : %s", ZSTD_getErrorName(flushError)); + CHECK (ZSTD_isError(flushError), "ZSTDMT_flushStream error : %s", ZSTD_getErrorName(flushError)); } } } /* final frame epilogue */ @@ -957,7 +958,7 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp outBuff.size = outBuff.pos + adjustedDstSize; DISPLAYLEVEL(5, "Ending into dst buffer of size %u \n", (U32)adjustedDstSize); remainingToFlush = ZSTDMT_endStream(zc, &outBuff); - CHECK (ZSTD_isError(remainingToFlush), "flush error : %s", ZSTD_getErrorName(remainingToFlush)); + CHECK (ZSTD_isError(remainingToFlush), "ZSTDMT_endStream error : %s", ZSTD_getErrorName(remainingToFlush)); DISPLAYLEVEL(5, "endStream : remainingToFlush : %u \n", (U32)remainingToFlush); } } DISPLAYLEVEL(5, "Frame completed \n"); From 805c5a3efbc22ef7735f009730aff8c3f95b6b86 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Sat, 1 Apr 2017 00:36:31 -0700 Subject: [PATCH 111/305] updated documentation on multithreading modes --- programs/zstd.1 | 20 +++++++++++++++++--- programs/zstd.1.md | 30 ++++++++++++++++++++++++++++-- programs/zstdcli.c | 16 ++++++++-------- 3 files changed, 53 insertions(+), 13 deletions(-) diff --git a/programs/zstd.1 b/programs/zstd.1 index 8b418a0ee..3b0064993 100644 --- a/programs/zstd.1 +++ b/programs/zstd.1 @@ -1,5 +1,5 @@ . -.TH "ZSTD" "1" "March 2017" "zstd 1.1.5" "User Commands" +.TH "ZSTD" "1" "April 2017" "zstd 1.1.5" "User Commands" . .SH "NAME" \fBzstd\fR \- zstd, unzstd, zstdcat \- Compress or decompress \.zst files @@ -97,6 +97,10 @@ Use FILEs as a training set to create a dictionary\. The training set should con unlocks high compression levels 20+ (maximum 22), using a lot more memory\. Note that decompression will also require more memory when using these levels\. . .TP +\fB\-T#\fR +Compress using # threads (default: 1)\. This modifier is only available if \fBzstd\fR was compiled with multithreading support\. +. +.TP \fB\-D file\fR use \fBfile\fR as Dictionary to compress or decompress FILE(s) . @@ -126,7 +130,7 @@ remove source file(s) after successful compression or decompression . .TP \fB\-k\fR, \fB\-\-keep\fR -keep source file(s) after successful compression or decompression\. This is the default behaviour\. +keep source file(s) after successful compression or decompression\. This is the default behavior\. . .TP \fB\-r\fR @@ -150,7 +154,7 @@ suppress warnings, interactivity, and notifications\. specify twice to suppress . .TP \fB\-C\fR, \fB\-\-[no\-]check\fR -add integrety check computed from uncompressed data (default : enabled) +add integrity check computed from uncompressed data (default : enabled) . .TP \fB\-\-\fR @@ -226,6 +230,9 @@ set process priority to real\-time . .SH "ADVANCED COMPRESSION OPTIONS" . +.SS "\-B#:" +Select the size of each compression job\. This parameter is available only when multi\-threading is enabled\. Default value is \fB4 * windowSize\fR, which means it varies depending on compression level\. \fB\-B#\fR makes it possible to select a custom value\. Note that job size must respect a minimum value which is enforced transparently\. This minimum is either 1 MB, or \fBoverlapSize\fR, whichever is largest\. +. .SS "\-\-zstd[=options]:" \fBzstd\fR provides 22 predefined compression levels\. The selected or default predefined compression level can be changed with advanced compression options\. The \fIoptions\fR are provided as a comma\-separated list\. You may specify only the options you want to change and the rest will be taken from the selected or default compression level\. The list of available \fIoptions\fR: . @@ -293,6 +300,13 @@ A larger minimum match length usually improves compression ratio but decreases c .IP The minimum \fItlen\fR is 4 and the maximum is 999\. . +.TP +\fBoverlapLog\fR=\fIovlog\fR, \fBovlog\fR=\fIovlog\fR +Select the amount of data reloaded from previous job into next one\. Reloading more data improves compression ratio, but decreases speed\. This parameter is only available if multithreading is enabled\. +. +.IP +The minimum \fIovlog\fR is 0, and the maximum is 9\. 0 means "no overlap", hence completely independent jobs\. 9 means "full overlap", meaning up to \fBwindowSize\fR is reloaded from previous job\. Reducing \fIovlog\fR by 1 reduces the amount of reload by a factor 2\. Default \fIovlog\fR is 6, which means "reload \fBwindowSize / 8\fR"\. Exception : the maximum compression level (22) has a default \fIovlog\fR of 9\. +. .SS "Example" The following parameters sets advanced compression options to those of predefined level 19 for files bigger than 256 KB: . diff --git a/programs/zstd.1.md b/programs/zstd.1.md index d1161a522..f8ef513ea 100644 --- a/programs/zstd.1.md +++ b/programs/zstd.1.md @@ -99,6 +99,9 @@ the last one takes effect. * `--ultra`: unlocks high compression levels 20+ (maximum 22), using a lot more memory. Note that decompression will also require more memory when using these levels. +* `-T#`: + Compress using # threads (default: 1). + This modifier is only available if `zstd` was compiled with multithreading support. * `-D file`: use `file` as Dictionary to compress or decompress FILE(s) * `--nodictID`: @@ -123,7 +126,7 @@ the last one takes effect. remove source file(s) after successful compression or decompression * `-k`, `--keep`: keep source file(s) after successful compression or decompression. - This is the default behaviour. + This is the default behavior. * `-r`: operate recursively on dictionaries * `-h`/`-H`, `--help`: @@ -136,10 +139,11 @@ the last one takes effect. suppress warnings, interactivity, and notifications. specify twice to suppress errors too. * `-C`, `--[no-]check`: - add integrety check computed from uncompressed data (default : enabled) + add integrity check computed from uncompressed data (default : enabled) * `--`: All arguments after `--` are treated as files + DICTIONARY BUILDER ------------------ `zstd` offers _dictionary_ compression, @@ -218,8 +222,17 @@ BENCHMARK * `--priority=rt`: set process priority to real-time + ADVANCED COMPRESSION OPTIONS ---------------------------- +### -B#: +Select the size of each compression job. +This parameter is available only when multi-threading is enabled. +Default value is `4 * windowSize`, which means it varies depending on compression level. +`-B#` makes it possible to select a custom value. +Note that job size must respect a minimum value which is enforced transparently. +This minimum is either 1 MB, or `overlapSize`, whichever is largest. + ### --zstd[=options]: `zstd` provides 22 predefined compression levels. The selected or default predefined compression level can be changed with @@ -290,6 +303,19 @@ The list of available _options_: The minimum _tlen_ is 4 and the maximum is 999. +- `overlapLog`=_ovlog_, `ovlog`=_ovlog_: + Select the amount of data reloaded from previous job into next one. + Reloading more data improves compression ratio, but decreases speed. + This parameter is only available if multithreading is enabled. + + The minimum _ovlog_ is 0, and the maximum is 9. + 0 means "no overlap", hence completely independent jobs. + 9 means "full overlap", meaning up to `windowSize` is reloaded from previous job. + Reducing _ovlog_ by 1 reduces the amount of reload by a factor 2. + Default _ovlog_ is 6, which means "reload `windowSize / 8`". + Exception : the maximum compression level (22) has a default _ovlog_ of 9. + + ### Example The following parameters sets advanced compression options to those of predefined level 19 for files bigger than 256 KB: diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 0cf8cacd1..18e259c74 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -113,19 +113,20 @@ static int usage_advanced(const char* programName) DISPLAY( "\n"); DISPLAY( "Advanced arguments :\n"); DISPLAY( " -V : display Version number and exit\n"); - DISPLAY( " -v : verbose mode; specify multiple times to increase log level (default:%d)\n", DEFAULT_DISPLAY_LEVEL); + DISPLAY( " -v : verbose mode; specify multiple times to increase verbosity\n"); DISPLAY( " -q : suppress warnings; specify twice to suppress errors too\n"); DISPLAY( " -c : force write to standard output, even if it is the console\n"); -#ifdef UTIL_HAS_CREATEFILELIST - DISPLAY( " -r : operate recursively on directories \n"); -#endif #ifndef ZSTD_NOCOMPRESS DISPLAY( "--ultra : enable levels beyond %i, up to %i (requires more memory)\n", ZSTDCLI_CLEVEL_MAX, ZSTD_maxCLevel()); - DISPLAY( "--no-dictID : don't write dictID into header (dictionary compression)\n"); - DISPLAY( "--[no-]check : integrity check (default:enabled) \n"); #ifdef ZSTD_MULTITHREAD DISPLAY( " -T# : use # threads for compression (default:1) \n"); - DISPLAY( " -B# : select size of independent sections (default:0==automatic) \n"); + DISPLAY( " -B# : select size of each job (default:0==automatic) \n"); +#endif + DISPLAY( "--no-dictID : don't write dictID into header (dictionary compression)\n"); + DISPLAY( "--[no-]check : integrity check (default:enabled) \n"); +#endif +#ifdef UTIL_HAS_CREATEFILELIST + DISPLAY( " -r : operate recursively on directories \n"); #endif #ifdef ZSTD_GZCOMPRESS DISPLAY( "--format=gzip : compress files to the .gz format \n"); @@ -134,7 +135,6 @@ static int usage_advanced(const char* programName) DISPLAY( "--format=xz : compress files to the .xz format \n"); DISPLAY( "--format=lzma : compress files to the .lzma format \n"); #endif -#endif #ifndef ZSTD_NODECOMPRESS DISPLAY( "--test : test compressed file integrity \n"); #if ZSTD_SPARSE_DEFAULT From a36330a27dc966533cf221b3dc701c4243eed5eb Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Mon, 3 Apr 2017 14:57:24 -0700 Subject: [PATCH 112/305] minor man update, for `overlapSize` --- programs/zstd.1 | 2 +- programs/zstd.1.md | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/programs/zstd.1 b/programs/zstd.1 index 3b0064993..c6b2c2742 100644 --- a/programs/zstd.1 +++ b/programs/zstd.1 @@ -302,7 +302,7 @@ The minimum \fItlen\fR is 4 and the maximum is 999\. . .TP \fBoverlapLog\fR=\fIovlog\fR, \fBovlog\fR=\fIovlog\fR -Select the amount of data reloaded from previous job into next one\. Reloading more data improves compression ratio, but decreases speed\. This parameter is only available if multithreading is enabled\. +Determine \fBoverlapSize\fR, amount of data reloaded from previous job\. This parameter is only available when multithreading is enabled\. Reloading more data improves compression ratio, but decreases speed\. . .IP The minimum \fIovlog\fR is 0, and the maximum is 9\. 0 means "no overlap", hence completely independent jobs\. 9 means "full overlap", meaning up to \fBwindowSize\fR is reloaded from previous job\. Reducing \fIovlog\fR by 1 reduces the amount of reload by a factor 2\. Default \fIovlog\fR is 6, which means "reload \fBwindowSize / 8\fR"\. Exception : the maximum compression level (22) has a default \fIovlog\fR of 9\. diff --git a/programs/zstd.1.md b/programs/zstd.1.md index f8ef513ea..07c36f683 100644 --- a/programs/zstd.1.md +++ b/programs/zstd.1.md @@ -304,9 +304,9 @@ The list of available _options_: The minimum _tlen_ is 4 and the maximum is 999. - `overlapLog`=_ovlog_, `ovlog`=_ovlog_: - Select the amount of data reloaded from previous job into next one. + Determine `overlapSize`, amount of data reloaded from previous job. + This parameter is only available when multithreading is enabled. Reloading more data improves compression ratio, but decreases speed. - This parameter is only available if multithreading is enabled. The minimum _ovlog_ is 0, and the maximum is 9. 0 means "no overlap", hence completely independent jobs. From 59aadc85dc868cd918b3eba94dba5c65b1a4f7bc Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Mon, 3 Apr 2017 15:23:09 -0700 Subject: [PATCH 113/305] Preallocate workSpace for cctx and combine create and init for cstream --- contrib/linux-kernel/include/zstd.h | 10 ++--- contrib/linux-kernel/lib/zstd_common.c | 6 +++ contrib/linux-kernel/lib/zstd_compress.c | 35 +++++++++-------- contrib/linux-kernel/lib/zstd_decompress.c | 44 +++++++++++----------- contrib/linux-kernel/lib/zstd_internal.h | 1 + 5 files changed, 53 insertions(+), 43 deletions(-) diff --git a/contrib/linux-kernel/include/zstd.h b/contrib/linux-kernel/include/zstd.h index 6594521d6..fde751271 100644 --- a/contrib/linux-kernel/include/zstd.h +++ b/contrib/linux-kernel/include/zstd.h @@ -256,9 +256,8 @@ size_t ZSTD_DStreamWorkspaceBound(size_t maxWindowSize); typedef struct ZSTD_CStream_s ZSTD_CStream; /*===== ZSTD_CStream management functions =====*/ -ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream(void* workspace, size_t workspaceSize); -ZSTDLIB_API size_t ZSTD_initCStream(ZSTD_CStream* zcs, ZSTD_parameters params, unsigned long long pledgedSrcSize); -ZSTDLIB_API size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, ZSTD_parameters params, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); +ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream(ZSTD_parameters params, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize); +ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_usingCDict(ZSTD_parameters params, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize); /*===== Streaming compression functions =====*/ ZSTDLIB_API size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize); /**< re-use compression parameters from previous init; skip dictionary loading stage; zcs must be init at least once before. note: pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ @@ -295,9 +294,8 @@ ZSTDLIB_API size_t ZSTD_CStreamOutSize(void); /**< recommended size for output typedef struct ZSTD_DStream_s ZSTD_DStream; /*===== ZSTD_DStream management functions =====*/ -ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream(void* workspace, size_t workspaceSize); -ZSTDLIB_API size_t ZSTD_initDStream(ZSTD_DStream* zds, size_t maxWindowSize); -ZSTDLIB_API size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, size_t maxWindowSize, const ZSTD_DDict* ddict); +ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream(size_t maxWindowSize, void* workspace, size_t workspaceSize); +ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream_usingDDict(size_t maxWindowSize, const ZSTD_DDict* ddict, void* workspace, size_t workspaceSize); /*===== Streaming decompression functions =====*/ ZSTDLIB_API size_t ZSTD_resetDStream(ZSTD_DStream* zds); /**< re-use decompression parameters from previous init; saves dictionary loading */ diff --git a/contrib/linux-kernel/lib/zstd_common.c b/contrib/linux-kernel/lib/zstd_common.c index c62cbd365..ca9382c67 100644 --- a/contrib/linux-kernel/lib/zstd_common.c +++ b/contrib/linux-kernel/lib/zstd_common.c @@ -48,6 +48,12 @@ ZSTD_customMem ZSTD_initStack(void* workspace, size_t workspaceSize) { return stackMem; } +void* ZSTD_stackAllocAll(void* opaque, size_t* size) { + ZSTD_stack* stack = (ZSTD_stack*)opaque; + *size = stack->end - ZSTD_PTR_ALIGN(stack->ptr); + return stack_push(stack, *size); +} + void* ZSTD_stackAlloc(void* opaque, size_t size) { ZSTD_stack* stack = (ZSTD_stack*)opaque; return stack_push(stack, size); diff --git a/contrib/linux-kernel/lib/zstd_compress.c b/contrib/linux-kernel/lib/zstd_compress.c index df258c636..f91f9f016 100644 --- a/contrib/linux-kernel/lib/zstd_compress.c +++ b/contrib/linux-kernel/lib/zstd_compress.c @@ -115,7 +115,11 @@ static ZSTD_CCtx* ZSTD_createCCtx_advanced(ZSTD_customMem customMem) ZSTD_CCtx* ZSTD_createCCtx(void* workspace, size_t workspaceSize) { ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize); - return ZSTD_createCCtx_advanced(stackMem); + ZSTD_CCtx* cctx = ZSTD_createCCtx_advanced(stackMem); + if (cctx) { + cctx->workSpace = ZSTD_stackAllocAll(cctx->customMem.opaque, &cctx->workSpaceSize); + } + return cctx; } size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx) @@ -2917,12 +2921,6 @@ ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem) return zcs; } -ZSTD_CStream* ZSTD_createCStream(void* workspace, size_t workspaceSize) -{ - ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize); - return ZSTD_createCStream_advanced(stackMem); -} - size_t ZSTD_freeCStream(ZSTD_CStream* zcs) { if (zcs==NULL) return 0; /* support free on NULL */ @@ -3006,18 +3004,25 @@ static size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } -/* note : cdict must outlive compression session */ -size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, ZSTD_parameters params, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize) +ZSTD_CStream* ZSTD_createCStream(ZSTD_parameters params, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize) { - size_t const initError = ZSTD_initCStream_advanced(zcs, NULL, 0, params, pledgedSrcSize); - zcs->cdict = cdict; - zcs->cctx->dictID = params.fParams.noDictIDFlag ? 0 : cdict->refContext->dictID; - return initError; + ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize); + ZSTD_CStream* const zcs = ZSTD_createCStream_advanced(stackMem); + if (zcs) { + size_t const code = ZSTD_initCStream_advanced(zcs, NULL, 0, params, pledgedSrcSize); + if (ZSTD_isError(code)) { return NULL; } + } + return zcs; } -size_t ZSTD_initCStream(ZSTD_CStream* zcs, ZSTD_parameters params, unsigned long long pledgedSrcSize) +ZSTD_CStream* ZSTD_createCStream_usingCDict(ZSTD_parameters params, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize) { - return ZSTD_initCStream_advanced(zcs, NULL, 0, params, pledgedSrcSize); + ZSTD_CStream* const zcs = ZSTD_createCStream(params, pledgedSrcSize, workspace, workspaceSize); + if (zcs) { + zcs->cdict = cdict; + zcs->cctx->dictID = params.fParams.noDictIDFlag ? 0 : cdict->refContext->dictID; + } + return zcs; } /*====== Compression ======*/ diff --git a/contrib/linux-kernel/lib/zstd_decompress.c b/contrib/linux-kernel/lib/zstd_decompress.c index 0d512e8f6..379806e63 100644 --- a/contrib/linux-kernel/lib/zstd_decompress.c +++ b/contrib/linux-kernel/lib/zstd_decompress.c @@ -2098,10 +2098,30 @@ static ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem) return zds; } -ZSTD_DStream* ZSTD_createDStream(void* workspace, size_t workspaceSize) +ZSTD_DStream* ZSTD_createDStream(size_t maxWindowSize, void* workspace, size_t workspaceSize) { ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize); - return ZSTD_createDStream_advanced(stackMem); + ZSTD_DStream* zds = ZSTD_createDStream_advanced(stackMem); + if (!zds) { return NULL; } + + zds->maxWindowSize = maxWindowSize; + zds->stage = zdss_loadHeader; + zds->lhSize = zds->inPos = zds->outStart = zds->outEnd = 0; + ZSTD_freeDDict(zds->ddictLocal); + zds->ddictLocal = NULL; + zds->ddict = zds->ddictLocal; + zds->legacyVersion = 0; + zds->hostageByte = 0; + return zds; +} + +ZSTD_DStream* ZSTD_createDStream_usingDDict(size_t maxWindowSize, const ZSTD_DDict* ddict, void* workspace, size_t workspaceSize) +{ + ZSTD_DStream* zds = ZSTD_createDStream(maxWindowSize, workspace, workspaceSize); + if (zds) { + zds->ddict = ddict; + } + return zds; } size_t ZSTD_freeDStream(ZSTD_DStream* zds) @@ -2127,26 +2147,6 @@ size_t ZSTD_freeDStream(ZSTD_DStream* zds) size_t ZSTD_DStreamInSize(void) { return ZSTD_BLOCKSIZE_ABSOLUTEMAX + ZSTD_blockHeaderSize; } size_t ZSTD_DStreamOutSize(void) { return ZSTD_BLOCKSIZE_ABSOLUTEMAX; } -size_t ZSTD_initDStream(ZSTD_DStream* zds, size_t maxWindowSize) -{ - zds->maxWindowSize = maxWindowSize; - zds->stage = zdss_loadHeader; - zds->lhSize = zds->inPos = zds->outStart = zds->outEnd = 0; - ZSTD_freeDDict(zds->ddictLocal); - zds->ddictLocal = NULL; - zds->ddict = zds->ddictLocal; - zds->legacyVersion = 0; - zds->hostageByte = 0; - return ZSTD_frameHeaderSize_prefix; -} - -size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, size_t maxWindowSize, const ZSTD_DDict* ddict) /**< note : ddict will just be referenced, and must outlive decompression session */ -{ - size_t const initResult = ZSTD_initDStream(zds, maxWindowSize); - zds->ddict = ddict; - return initResult; -} - size_t ZSTD_resetDStream(ZSTD_DStream* zds) { zds->stage = zdss_loadHeader; diff --git a/contrib/linux-kernel/lib/zstd_internal.h b/contrib/linux-kernel/lib/zstd_internal.h index b7dcc03e0..66f318af6 100644 --- a/contrib/linux-kernel/lib/zstd_internal.h +++ b/contrib/linux-kernel/lib/zstd_internal.h @@ -229,6 +229,7 @@ typedef struct { ZSTD_customMem ZSTD_initStack(void* workspace, size_t workspaceSize); +void* ZSTD_stackAllocAll(void* opaque, size_t* size); void* ZSTD_stackAlloc(void* opaque, size_t size); void ZSTD_stackFree(void* opaque, void* address); From 87cec8fd56b01147ea229482b1e492985e050dab Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Mon, 3 Apr 2017 16:08:20 -0700 Subject: [PATCH 114/305] Fix compilation errors --- contrib/linux-kernel/lib/entropy_common.c | 2 -- contrib/linux-kernel/lib/error_private.h | 12 ------------ contrib/linux-kernel/lib/fse.h | 1 - contrib/linux-kernel/lib/huf.h | 1 - 4 files changed, 16 deletions(-) diff --git a/contrib/linux-kernel/lib/entropy_common.c b/contrib/linux-kernel/lib/entropy_common.c index f6a43662d..68d880827 100644 --- a/contrib/linux-kernel/lib/entropy_common.c +++ b/contrib/linux-kernel/lib/entropy_common.c @@ -47,10 +47,8 @@ unsigned FSE_versionNumber(void) { return FSE_VERSION_NUMBER; } /*=== Error Management ===*/ unsigned FSE_isError(size_t code) { return ERR_isError(code); } -const char* FSE_getErrorName(size_t code) { return ERR_getErrorName(code); } unsigned HUF_isError(size_t code) { return ERR_isError(code); } -const char* HUF_getErrorName(size_t code) { return ERR_getErrorName(code); } /*-************************************************************** diff --git a/contrib/linux-kernel/lib/error_private.h b/contrib/linux-kernel/lib/error_private.h index 680eba112..9bfbe4b5c 100644 --- a/contrib/linux-kernel/lib/error_private.h +++ b/contrib/linux-kernel/lib/error_private.h @@ -41,16 +41,4 @@ ERR_STATIC unsigned ERR_isError(size_t code) { return (code > ERROR(maxCode)); } ERR_STATIC ERR_enum ERR_getErrorCode(size_t code) { if (!ERR_isError(code)) return (ERR_enum)0; return (ERR_enum) (0-code); } - -/*-**************************************** -* Error Strings -******************************************/ - -const char* ERR_getErrorString(ERR_enum code); /* error_private.c */ - -ERR_STATIC const char* ERR_getErrorName(size_t code) -{ - return ERR_getErrorString(ERR_getErrorCode(code)); -} - #endif /* ERROR_H_MODULE */ diff --git a/contrib/linux-kernel/lib/fse.h b/contrib/linux-kernel/lib/fse.h index 6df7ff935..fbd97fae3 100644 --- a/contrib/linux-kernel/lib/fse.h +++ b/contrib/linux-kernel/lib/fse.h @@ -66,7 +66,6 @@ FSE_PUBLIC_API size_t FSE_compressBound(size_t size); /* maximum compresse /* Error Management */ FSE_PUBLIC_API unsigned FSE_isError(size_t code); /* tells if a return value is an error code */ -FSE_PUBLIC_API const char* FSE_getErrorName(size_t code); /* provides error code string (useful for debugging) */ /*-***************************************** diff --git a/contrib/linux-kernel/lib/huf.h b/contrib/linux-kernel/lib/huf.h index e0a30ce00..f135226dc 100644 --- a/contrib/linux-kernel/lib/huf.h +++ b/contrib/linux-kernel/lib/huf.h @@ -45,7 +45,6 @@ size_t HUF_compressBound(size_t size); /**< maximum compressed size (worst /* Error Management */ unsigned HUF_isError(size_t code); /**< tells if a return value is an error code */ -const char* HUF_getErrorName(size_t code); /**< provides error code string (useful for debugging) */ /* *** Advanced function *** */ From 62ecad3819e6d03b1eed8141ad8dd7abea841df3 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Mon, 3 Apr 2017 20:56:39 -0700 Subject: [PATCH 115/305] Fix ZSTD_initCStream_usingCDict() to use dictionary --- lib/compress/zstd_compress.c | 3 ++- tests/zstreamtest.c | 28 ++++++++++++++++++++++++++++ 2 files changed, 30 insertions(+), 1 deletion(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index e1c734927..03b9758e8 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -3068,7 +3068,8 @@ size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) size_t const initError = ZSTD_initCStream_advanced(zcs, NULL, 0, params, 0); zcs->cdict = cdict; zcs->cctx->dictID = params.fParams.noDictIDFlag ? 0 : cdict->refContext->dictID; - return initError; + if (ZSTD_isError(initError)) return initError; + return ZSTD_resetCStream_internal(zcs, 0); } size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel) diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index 10bcbe0cd..e1e68b01c 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -438,6 +438,34 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo if (!ZSTD_isError(r)) goto _output_error; /* must fail : frame requires > 100 bytes */ DISPLAYLEVEL(3, "OK (%s)\n", ZSTD_getErrorName(r)); } + DISPLAYLEVEL(3, "test%3i : check dictionary used : ", testNb++); + { ZSTD_parameters params = ZSTD_getParams(1, CNBufferSize, dictionary.filled); + ZSTD_CDict* cdict; + params.fParams.noDictIDFlag = 1; + cdict = ZSTD_createCDict_advanced(dictionary.start, dictionary.filled, 1, params, customMem); + { size_t const initError = ZSTD_initCStream_usingCDict(zc, cdict); + if (ZSTD_isError(initError)) goto _output_error; } + cSize = 0; + outBuff.dst = compressedBuffer; + outBuff.size = compressedBufferSize; + outBuff.pos = 0; + inBuff.src = CNBuffer; + inBuff.size = CNBufferSize; + inBuff.pos = 0; + { size_t const r = ZSTD_compressStream(zc, &outBuff, &inBuff); + if (ZSTD_isError(r)) goto _output_error; } + if (inBuff.pos != inBuff.size) goto _output_error; /* entire input should be consumed */ + { size_t const r = ZSTD_endStream(zc, &outBuff); + if (r != 0) goto _output_error; } /* error, or some data not flushed */ + cSize = outBuff.pos; + ZSTD_freeCDict(cdict); + DISPLAYLEVEL(3, "OK (%u bytes : %.2f%%)\n", (U32)cSize, (double)cSize/CNBufferSize*100); + + { size_t const r = ZSTD_decompress(decodedBuffer, CNBufferSize, compressedBuffer, cSize); + if (!ZSTD_isError(r)) goto _output_error; /* must fail : dictionary not used */ + DISPLAYLEVEL(3, "OK (%s)\n", ZSTD_getErrorName(r)); } + } + /* Unknown srcSize */ DISPLAYLEVEL(3, "test%3i : pledgedSrcSize == 0 behaves properly : ", testNb++); { ZSTD_parameters params = ZSTD_getParams(5, 0, 0); From 39a6cc5172ef32e8afc115fa2144d9cd23837df3 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Mon, 3 Apr 2017 21:00:44 -0700 Subject: [PATCH 116/305] Make ZSTD_compress_usingCDict() respect contentSizeFlag --- lib/compress/zstd_compress.c | 2 ++ tests/fuzzer.c | 33 +++++++++++++++++++++++++++++++++ 2 files changed, 35 insertions(+) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 03b9758e8..21f31b756 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -2921,6 +2921,8 @@ size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, if (cdict->refContext->params.fParams.contentSizeFlag==1) { cctx->params.fParams.contentSizeFlag = 1; cctx->frameContentSize = srcSize; + } else { + cctx->params.fParams.contentSizeFlag = 0; } return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize); diff --git a/tests/fuzzer.c b/tests/fuzzer.c index fed875843..342c953e9 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -404,6 +404,39 @@ static int basicUnitTests(U32 seed, double compressibility) if (r != CNBuffSize) goto _output_error); DISPLAYLEVEL(4, "OK \n"); + DISPLAYLEVEL(4, "test%3i : compress with preprocessed dictionary : ", testNb++); + { ZSTD_parameters params = ZSTD_getParams(1, CNBuffSize, dictSize); + params.fParams.contentSizeFlag = 0; + { ZSTD_customMem customMem = { NULL, NULL, NULL }; + ZSTD_CDict* cdict = ZSTD_createCDict_advanced(dictBuffer, dictSize, 1, params, customMem); + cSize = ZSTD_compress_usingCDict(cctx, compressedBuffer, ZSTD_compressBound(CNBuffSize), + CNBuffer, CNBuffSize, cdict); + ZSTD_freeCDict(cdict); + if (ZSTD_isError(cSize)) goto _output_error; + } + } + DISPLAYLEVEL(4, "OK (%u bytes : %.2f%%)\n", (U32)cSize, (double)cSize/CNBuffSize*100); + + DISPLAYLEVEL(4, "test%3i : retrieve dictID from frame : ", testNb++); + { U32 const did = ZSTD_getDictID_fromFrame(compressedBuffer, cSize); + if (did != dictID) goto _output_error; /* non-conformant (content-only) dictionary */ + } + DISPLAYLEVEL(4, "OK \n"); + + DISPLAYLEVEL(4, "test%3i : frame should not have content size : ", testNb++); + { unsigned long long const contentSize = ZSTD_findDecompressedSize(compressedBuffer, cSize); + if (contentSize != ZSTD_CONTENTSIZE_UNKNOWN) goto _output_error; /* cdict contentSizeFlag not used */ + } + DISPLAYLEVEL(4, "OK \n"); + + DISPLAYLEVEL(4, "test%3i : frame built with dictionary should be decompressible : ", testNb++); + CHECKPLUS(r, ZSTD_decompress_usingDict(dctx, + decodedBuffer, CNBuffSize, + compressedBuffer, cSize, + dictBuffer, dictSize), + if (r != CNBuffSize) goto _output_error); + DISPLAYLEVEL(4, "OK \n"); + DISPLAYLEVEL(4, "test%3i : compress without dictID : ", testNb++); { ZSTD_parameters p = ZSTD_getParams(3, CNBuffSize, dictSize); p.fParams.noDictIDFlag = 1; From 26b046a7c42c13af544c354db3741b4f3418a352 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Mon, 3 Apr 2017 21:46:28 -0700 Subject: [PATCH 117/305] Remove unnecessary dictID store --- lib/compress/zstd_compress.c | 1 - 1 file changed, 1 deletion(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 21f31b756..8318843be 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -3069,7 +3069,6 @@ size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict); size_t const initError = ZSTD_initCStream_advanced(zcs, NULL, 0, params, 0); zcs->cdict = cdict; - zcs->cctx->dictID = params.fParams.noDictIDFlag ? 0 : cdict->refContext->dictID; if (ZSTD_isError(initError)) return initError; return ZSTD_resetCStream_internal(zcs, 0); } From 33fc0ad56a624bbeaf54796e82aaa356d1e13929 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 11:52:49 -0700 Subject: [PATCH 118/305] Fix ZSTD_createCStream_usingCDict() and ZSTD_compress_usingCDict() --- contrib/linux-kernel/include/zstd.h | 2 +- contrib/linux-kernel/lib/zstd_compress.c | 9 +++++++-- 2 files changed, 8 insertions(+), 3 deletions(-) diff --git a/contrib/linux-kernel/include/zstd.h b/contrib/linux-kernel/include/zstd.h index fde751271..9d8fc8e0b 100644 --- a/contrib/linux-kernel/include/zstd.h +++ b/contrib/linux-kernel/include/zstd.h @@ -257,7 +257,7 @@ size_t ZSTD_DStreamWorkspaceBound(size_t maxWindowSize); typedef struct ZSTD_CStream_s ZSTD_CStream; /*===== ZSTD_CStream management functions =====*/ ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream(ZSTD_parameters params, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize); -ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_usingCDict(ZSTD_parameters params, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize); +ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_usingCDict(const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize); /*===== Streaming compression functions =====*/ ZSTDLIB_API size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize); /**< re-use compression parameters from previous init; skip dictionary loading stage; zcs must be init at least once before. note: pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ diff --git a/contrib/linux-kernel/lib/zstd_compress.c b/contrib/linux-kernel/lib/zstd_compress.c index f91f9f016..e57cd6baa 100644 --- a/contrib/linux-kernel/lib/zstd_compress.c +++ b/contrib/linux-kernel/lib/zstd_compress.c @@ -2861,6 +2861,8 @@ size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, if (cdict->refContext->params.fParams.contentSizeFlag==1) { cctx->params.fParams.contentSizeFlag = 1; cctx->frameContentSize = srcSize; + } else { + cctx->params.fParams.contentSizeFlag = 0; } return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize); @@ -3015,12 +3017,15 @@ ZSTD_CStream* ZSTD_createCStream(ZSTD_parameters params, unsigned long long pled return zcs; } -ZSTD_CStream* ZSTD_createCStream_usingCDict(ZSTD_parameters params, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize) +ZSTD_CStream* ZSTD_createCStream_usingCDict(const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize) { + ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict); ZSTD_CStream* const zcs = ZSTD_createCStream(params, pledgedSrcSize, workspace, workspaceSize); if (zcs) { zcs->cdict = cdict; - zcs->cctx->dictID = params.fParams.noDictIDFlag ? 0 : cdict->refContext->dictID; + if (ZSTD_isError(ZSTD_resetCStream_internal(zcs, pledgedSrcSize))) { + return NULL; + } } return zcs; } From 79298bf187181e9edc37c5b766ae4020d9972d9f Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 11:53:22 -0700 Subject: [PATCH 119/305] Add userland test with mock kernel headers --- contrib/linux-kernel/test/.gitignore | 1 + contrib/linux-kernel/test/Makefile | 27 + contrib/linux-kernel/test/UserlandTest.cpp | 479 ++++++++++++++++++ .../linux-kernel/test/include/asm/unaligned.h | 177 +++++++ .../test/include/linux/compiler.h | 12 + .../linux-kernel/test/include/linux/kernel.h | 14 + .../linux-kernel/test/include/linux/string.h | 1 + .../linux-kernel/test/include/linux/types.h | 2 + 8 files changed, 713 insertions(+) create mode 100644 contrib/linux-kernel/test/.gitignore create mode 100644 contrib/linux-kernel/test/Makefile create mode 100644 contrib/linux-kernel/test/UserlandTest.cpp create mode 100644 contrib/linux-kernel/test/include/asm/unaligned.h create mode 100644 contrib/linux-kernel/test/include/linux/compiler.h create mode 100644 contrib/linux-kernel/test/include/linux/kernel.h create mode 100644 contrib/linux-kernel/test/include/linux/string.h create mode 100644 contrib/linux-kernel/test/include/linux/types.h diff --git a/contrib/linux-kernel/test/.gitignore b/contrib/linux-kernel/test/.gitignore new file mode 100644 index 000000000..4fc10228d --- /dev/null +++ b/contrib/linux-kernel/test/.gitignore @@ -0,0 +1 @@ +*Test diff --git a/contrib/linux-kernel/test/Makefile b/contrib/linux-kernel/test/Makefile new file mode 100644 index 000000000..b16964f84 --- /dev/null +++ b/contrib/linux-kernel/test/Makefile @@ -0,0 +1,27 @@ + +IFLAGS := -isystem include/ -I ../include/ -I ../lib/ -isystem googletest/googletest/include + +SOURCES := $(wildcard ../lib/*.c) +OBJECTS := $(patsubst %.c,%.o,$(SOURCES)) + +ARFLAGS := rcs +CXXFLAGS += -std=c++11 +CFLAGS += -g -O0 +CPPFLAGS += $(IFLAGS) + +../lib/libzstd.a: $(OBJECTS) + $(AR) $(ARFLAGS) $@ $^ + +UserlandTest: UserlandTest.cpp ../lib/libzstd.a + $(CXX) $(CXXFLAGS) $(CFLAGS) $(CPPFLAGS) $^ googletest/build/googlemock/gtest/libgtest.a googletest/build/googlemock/gtest/libgtest_main.a -o $@ + +# Install googletest +.PHONY: googletest +googletest: + @$(RM) -rf googletest + @git clone https://github.com/google/googletest + @mkdir -p googletest/build + @cd googletest/build && cmake .. && $(MAKE) + +clean: + $(RM) -f *.{o,a} ../lib/*.{o,a} diff --git a/contrib/linux-kernel/test/UserlandTest.cpp b/contrib/linux-kernel/test/UserlandTest.cpp new file mode 100644 index 000000000..7bdbd1bf7 --- /dev/null +++ b/contrib/linux-kernel/test/UserlandTest.cpp @@ -0,0 +1,479 @@ +extern "C" { +#include +} +#include +#include +#include +#include + +using namespace std; + +namespace { +struct WorkspaceDeleter { + void *memory; + + template void operator()(T const *) { free(memory); } +}; + +std::unique_ptr +createCCtx(ZSTD_compressionParameters cParams) { + size_t const workspaceSize = ZSTD_CCtxWorkspaceBound(cParams); + void *workspace = malloc(workspaceSize); + std::unique_ptr cctx{ + ZSTD_createCCtx(workspace, workspaceSize), WorkspaceDeleter{workspace}}; + if (!cctx) { + throw std::runtime_error{"Bad cctx"}; + } + return cctx; +} + +std::unique_ptr +createCCtx(int level, unsigned long long estimatedSrcSize = 0, + size_t dictSize = 0) { + auto const cParams = ZSTD_getCParams(level, estimatedSrcSize, dictSize); + return createCCtx(cParams); +} + +std::unique_ptr +createDCtx() { + size_t const workspaceSize = ZSTD_DCtxWorkspaceBound(); + void *workspace = malloc(workspaceSize); + std::unique_ptr dctx{ + ZSTD_createDCtx(workspace, workspaceSize), WorkspaceDeleter{workspace}}; + if (!dctx) { + throw std::runtime_error{"Bad dctx"}; + } + return dctx; +} + +std::unique_ptr +createCDict(std::string const& dict, ZSTD_parameters params) { + size_t const workspaceSize = ZSTD_CDictWorkspaceBound(params.cParams); + void *workspace = malloc(workspaceSize); + std::unique_ptr cdict{ + ZSTD_createCDict(dict.data(), dict.size(), params, workspace, + workspaceSize), + WorkspaceDeleter{workspace}}; + if (!cdict) { + throw std::runtime_error{"Bad cdict"}; + } + return cdict; +} + +std::unique_ptr +createCDict(std::string const& dict, int level) { + auto const params = ZSTD_getParams(level, 0, dict.size()); + return createCDict(dict, params); +} + +std::unique_ptr +createDDict(std::string const& dict) { + size_t const workspaceSize = ZSTD_DDictWorkspaceBound(); + void *workspace = malloc(workspaceSize); + std::unique_ptr ddict{ + ZSTD_createDDict(dict.data(), dict.size(), workspace, workspaceSize), + WorkspaceDeleter{workspace}}; + if (!ddict) { + throw std::runtime_error{"Bad ddict"}; + } + return ddict; +} + +std::unique_ptr +createCStream(ZSTD_parameters params, unsigned long long pledgedSrcSize = 0) { + size_t const workspaceSize = ZSTD_CStreamWorkspaceBound(params.cParams); + void *workspace = malloc(workspaceSize); + std::unique_ptr zcs{ + ZSTD_createCStream(params, pledgedSrcSize, workspace, workspaceSize)}; + if (!zcs) { + throw std::runtime_error{"bad cstream"}; + } + return zcs; +} + +std::unique_ptr +createCStream(ZSTD_compressionParameters cParams, ZSTD_CDict const &cdict, + unsigned long long pledgedSrcSize = 0) { + size_t const workspaceSize = ZSTD_CStreamWorkspaceBound(cParams); + void *workspace = malloc(workspaceSize); + std::unique_ptr zcs{ + ZSTD_createCStream_usingCDict(&cdict, pledgedSrcSize, workspace, + workspaceSize)}; + if (!zcs) { + throw std::runtime_error{"bad cstream"}; + } + return zcs; +} + +std::unique_ptr +createCStream(int level, unsigned long long pledgedSrcSize = 0) { + auto const params = ZSTD_getParams(level, pledgedSrcSize, 0); + return createCStream(params, pledgedSrcSize); +} + +std::unique_ptr +createDStream(size_t maxWindowSize = (1ULL << ZSTD_WINDOWLOG_MAX), + ZSTD_DDict const *ddict = nullptr) { + size_t const workspaceSize = ZSTD_DStreamWorkspaceBound(maxWindowSize); + void *workspace = malloc(workspaceSize); + std::unique_ptr zds{ + ddict == nullptr + ? ZSTD_createDStream(maxWindowSize, workspace, workspaceSize) + : ZSTD_createDStream_usingDDict(maxWindowSize, ddict, workspace, + workspaceSize)}; + if (!zds) { + throw std::runtime_error{"bad dstream"}; + } + return zds; +} + +std::string compress(ZSTD_CCtx &cctx, std::string const &data, + ZSTD_parameters params, std::string const &dict = "") { + std::string compressed; + compressed.resize(ZSTD_compressBound(data.size())); + size_t const rc = + dict.empty() + ? ZSTD_compressCCtx(&cctx, &compressed[0], compressed.size(), + data.data(), data.size(), params) + : ZSTD_compress_usingDict(&cctx, &compressed[0], compressed.size(), + data.data(), data.size(), dict.data(), + dict.size(), params); + if (ZSTD_isError(rc)) { + throw std::runtime_error{"compression error"}; + } + compressed.resize(rc); + return compressed; +} + +std::string compress(ZSTD_CCtx& cctx, std::string const& data, int level, std::string const& dict = "") { + auto const params = ZSTD_getParams(level, 0, dict.size()); + return compress(cctx, data, params, dict); +} + +std::string decompress(ZSTD_DCtx& dctx, std::string const& compressed, size_t decompressedSize, std::string const& dict = "") { + std::string decompressed; + decompressed.resize(decompressedSize); + size_t const rc = + dict.empty() + ? ZSTD_decompressDCtx(&dctx, &decompressed[0], decompressed.size(), + compressed.data(), compressed.size()) + : ZSTD_decompress_usingDict( + &dctx, &decompressed[0], decompressed.size(), compressed.data(), + compressed.size(), dict.data(), dict.size()); + if (ZSTD_isError(rc)) { + throw std::runtime_error{"decompression error"}; + } + decompressed.resize(rc); + return decompressed; +} + +std::string compress(ZSTD_CCtx& cctx, std::string const& data, ZSTD_CDict& cdict) { + std::string compressed; + compressed.resize(ZSTD_compressBound(data.size())); + size_t const rc = + ZSTD_compress_usingCDict(&cctx, &compressed[0], compressed.size(), + data.data(), data.size(), &cdict); + if (ZSTD_isError(rc)) { + throw std::runtime_error{"compression error"}; + } + compressed.resize(rc); + return compressed; +} + +std::string decompress(ZSTD_DCtx& dctx, std::string const& compressed, size_t decompressedSize, ZSTD_DDict& ddict) { + std::string decompressed; + decompressed.resize(decompressedSize); + size_t const rc = + ZSTD_decompress_usingDDict(&dctx, &decompressed[0], decompressed.size(), + compressed.data(), compressed.size(), &ddict); + if (ZSTD_isError(rc)) { + throw std::runtime_error{"decompression error"}; + } + decompressed.resize(rc); + return decompressed; +} + +std::string compress(ZSTD_CStream& zcs, std::string const& data) { + std::string compressed; + compressed.resize(ZSTD_compressBound(data.size())); + ZSTD_inBuffer in = {data.data(), data.size(), 0}; + ZSTD_outBuffer out = {&compressed[0], compressed.size(), 0}; + while (in.pos != in.size) { + size_t const rc = ZSTD_compressStream(&zcs, &out, &in); + if (ZSTD_isError(rc)) { + throw std::runtime_error{"compress stream failed"}; + } + } + size_t const rc = ZSTD_endStream(&zcs, &out); + if (rc != 0) { + throw std::runtime_error{"compress end failed"}; + } + compressed.resize(out.pos); + return compressed; +} + +std::string decompress(ZSTD_DStream &zds, std::string const &compressed, + size_t decompressedSize) { + std::string decompressed; + decompressed.resize(decompressedSize); + ZSTD_inBuffer in = {compressed.data(), compressed.size(), 0}; + ZSTD_outBuffer out = {&decompressed[0], decompressed.size(), 0}; + while (in.pos != in.size) { + size_t const rc = ZSTD_decompressStream(&zds, &out, &in); + if (ZSTD_isError(rc)) { + throw std::runtime_error{"decompress stream failed"}; + } + } + decompressed.resize(out.pos); + return decompressed; +} + +std::string makeData(size_t size) { + std::string result; + result.reserve(size + 20); + while (result.size() < size) { + result += "Hello world"; + } + return result; +} + +std::string const kData = "Hello world"; +std::string const kPlainDict = makeData(10000); +std::string const kZstdDict{ + "\x37\xA4\x30\xEC\x99\x69\x58\x1C\x21\x10\xD8\x4A\x84\x01\xCC\xF3" + "\x3C\xCF\x9B\x25\xBB\xC9\x6E\xB2\x9B\xEC\x26\xAD\xCF\xDF\x4E\xCD" + "\xF3\x2C\x3A\x21\x84\x10\x42\x08\x21\x01\x33\xF1\x78\x3C\x1E\x8F" + "\xC7\xE3\xF1\x78\x3C\xCF\xF3\xBC\xF7\xD4\x42\x41\x41\x41\x41\x41" + "\x41\x41\x41\x41\x41\x41\x41\x41\x41\x41\x41\x41\x41\x41\x41\x41" + "\x41\x41\x41\x41\xA1\x50\x28\x14\x0A\x85\x42\xA1\x50\x28\x14\x0A" + "\x85\xA2\x28\x8A\xA2\x28\x4A\x29\x7D\x74\xE1\xE1\xE1\xE1\xE1\xE1" + "\xE1\xE1\xE1\xE1\xE1\xE1\xE1\xE1\xE1\xE1\xE1\xE1\xE1\xF1\x78\x3C" + "\x1E\x8F\xC7\xE3\xF1\x78\x9E\xE7\x79\xEF\x01\x01\x00\x00\x00\x04" + "\x00\x00\x00\x08\x00\x00\x00" + "0123456789", + 161}; +} + +TEST(Block, CCtx) { + auto cctx = createCCtx(1); + auto const compressed = compress(*cctx, kData, 1); + auto dctx = createDCtx(); + auto const decompressed = decompress(*dctx, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); +} + +TEST(Block, NoContentSize) { + auto cctx = createCCtx(1); + auto const c = compress(*cctx, kData, 1); + auto const size = ZSTD_findDecompressedSize(c.data(), c.size()); + EXPECT_EQ(ZSTD_CONTENTSIZE_UNKNOWN, size); +} + +TEST(Block, ContentSize) { + auto cctx = createCCtx(1); + auto params = ZSTD_getParams(1, 0, 0); + params.fParams.contentSizeFlag = 1; + auto const c = compress(*cctx, kData, params); + auto const size = ZSTD_findDecompressedSize(c.data(), c.size()); + EXPECT_EQ(kData.size(), size); +} + +TEST(Block, CCtxLevelIncrease) { + std::string c; + auto cctx = createCCtx(6); + auto dctx = createDCtx(); + for (int level = 1; level <= 6; ++level) { + auto compressed = compress(*cctx, kData, level); + auto const decompressed = decompress(*dctx, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); + } +} + +TEST(Block, PlainDict) { + auto cctx = createCCtx(1); + auto const compressed = compress(*cctx, kData, 1, kPlainDict); + auto dctx = createDCtx(); + EXPECT_ANY_THROW(decompress(*dctx, compressed, kData.size())); + auto const decompressed = + decompress(*dctx, compressed, kData.size(), kPlainDict); + EXPECT_EQ(kData, decompressed); +} + +TEST(Block, ZstdDict) { + auto cctx = createCCtx(1); + auto const compressed = compress(*cctx, kData, 1, kZstdDict); + auto dctx = createDCtx(); + EXPECT_ANY_THROW(decompress(*dctx, compressed, kData.size())); + auto const decompressed = + decompress(*dctx, compressed, kData.size(), kZstdDict); + EXPECT_EQ(kData, decompressed); +} + +TEST(Block, PreprocessedPlainDict) { + auto cctx = createCCtx(1); + auto const cdict = createCDict(kPlainDict, 1); + auto const compressed = compress(*cctx, kData, *cdict); + auto dctx = createDCtx(); + auto const ddict = createDDict(kPlainDict); + EXPECT_ANY_THROW(decompress(*dctx, compressed, kData.size())); + auto const decompressed = + decompress(*dctx, compressed, kData.size(), *ddict); + EXPECT_EQ(kData, decompressed); +} + +TEST(Block, PreprocessedZstdDict) { + auto cctx = createCCtx(1); + auto const cdict = createCDict(kZstdDict, 1); + auto const compressed = compress(*cctx, kData, *cdict); + auto dctx = createDCtx(); + auto const ddict = createDDict(kZstdDict); + EXPECT_ANY_THROW(decompress(*dctx, compressed, kData.size())); + auto const decompressed = + decompress(*dctx, compressed, kData.size(), *ddict); + EXPECT_EQ(kData, decompressed); +} + +TEST(Block, RecreateCCtx) { + auto cctx = createCCtx(1); + { + auto const compressed = compress(*cctx, kData, 1); + auto dctx = createDCtx(); + auto const decompressed = decompress(*dctx, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); + } + // Create the cctx with the same memory + auto d = cctx.get_deleter(); + auto raw = cctx.release(); + auto params = ZSTD_getParams(1, 0, 0); + cctx.reset( + ZSTD_createCCtx(d.memory, ZSTD_CCtxWorkspaceBound(params.cParams))); + // Repeat + { + auto const compressed = compress(*cctx, kData, 1); + auto dctx = createDCtx(); + auto const decompressed = decompress(*dctx, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); + } +} + +TEST(Block, RecreateDCtx) { + auto dctx = createDCtx(); + { + auto cctx = createCCtx(1); + auto const compressed = compress(*cctx, kData, 1); + auto const decompressed = decompress(*dctx, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); + } + // Create the cctx with the same memory + auto d = dctx.get_deleter(); + auto raw = dctx.release(); + dctx.reset(ZSTD_createDCtx(d.memory, ZSTD_DCtxWorkspaceBound())); + // Repeat + { + auto cctx = createCCtx(1); + auto const compressed = compress(*cctx, kData, 1); + auto dctx = createDCtx(); + auto const decompressed = decompress(*dctx, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); + } +} + +TEST(Stream, Basic) { + auto zcs = createCStream(1); + auto const compressed = compress(*zcs, kData); + auto zds = createDStream(); + auto const decompressed = decompress(*zds, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); +} + +TEST(Stream, PlainDict) { + auto params = ZSTD_getParams(1, kData.size(), kPlainDict.size()); + params.cParams.windowLog = 17; + auto cdict = createCDict(kPlainDict, params); + auto zcs = createCStream(params.cParams, *cdict, kData.size()); + auto const compressed = compress(*zcs, kData); + auto const contentSize = + ZSTD_findDecompressedSize(compressed.data(), compressed.size()); + EXPECT_ANY_THROW(decompress(*createDStream(), compressed, kData.size())); + auto ddict = createDDict(kPlainDict); + auto zds = createDStream(1 << 17, ddict.get()); + auto const decompressed = decompress(*zds, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); +} + +TEST(Stream, ZstdDict) { + auto params = ZSTD_getParams(1, 0, 0); + params.cParams.windowLog = 17; + auto cdict = createCDict(kZstdDict, 1); + auto zcs = createCStream(params.cParams, *cdict); + auto const compressed = compress(*zcs, kData); + EXPECT_ANY_THROW(decompress(*createDStream(), compressed, kData.size())); + auto ddict = createDDict(kZstdDict); + auto zds = createDStream(1 << 17, ddict.get()); + auto const decompressed = decompress(*zds, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); +} + +TEST(Stream, ResetCStream) { + auto zcs = createCStream(1); + auto zds = createDStream(); + { + auto const compressed = compress(*zcs, kData); + auto const decompressed = decompress(*zds, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); + } + { + ZSTD_resetCStream(zcs.get(), 0); + auto const compressed = compress(*zcs, kData); + auto const decompressed = decompress(*zds, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); + } +} + +TEST(Stream, ResetDStream) { + auto zcs = createCStream(1); + auto zds = createDStream(); + auto const compressed = compress(*zcs, kData); + EXPECT_ANY_THROW(decompress(*zds, kData, kData.size())); + EXPECT_ANY_THROW(decompress(*zds, compressed, kData.size())); + ZSTD_resetDStream(zds.get()); + auto const decompressed = decompress(*zds, compressed, kData.size()); + EXPECT_EQ(kData, decompressed); +} + +TEST(Stream, Flush) { + auto zcs = createCStream(1); + auto zds = createDStream(); + std::string compressed; + { + compressed.resize(ZSTD_compressBound(kData.size())); + ZSTD_inBuffer in = {kData.data(), kData.size(), 0}; + ZSTD_outBuffer out = {&compressed[0], compressed.size(), 0}; + while (in.pos != in.size) { + size_t const rc = ZSTD_compressStream(zcs.get(), &out, &in); + if (ZSTD_isError(rc)) { + throw std::runtime_error{"compress stream failed"}; + } + } + EXPECT_EQ(0, out.pos); + size_t const rc = ZSTD_flushStream(zcs.get(), &out); + if (rc != 0) { + throw std::runtime_error{"compress end failed"}; + } + compressed.resize(out.pos); + EXPECT_LT(0, out.pos); + } + std::string decompressed; + { + decompressed.resize(kData.size()); + ZSTD_inBuffer in = {compressed.data(), compressed.size(), 0}; + ZSTD_outBuffer out = {&decompressed[0], decompressed.size(), 0}; + while (in.pos != in.size) { + size_t const rc = ZSTD_decompressStream(zds.get(), &out, &in); + if (ZSTD_isError(rc)) { + throw std::runtime_error{"decompress stream failed"}; + } + } + } + EXPECT_EQ(kData, decompressed); +} diff --git a/contrib/linux-kernel/test/include/asm/unaligned.h b/contrib/linux-kernel/test/include/asm/unaligned.h new file mode 100644 index 000000000..4f4828126 --- /dev/null +++ b/contrib/linux-kernel/test/include/asm/unaligned.h @@ -0,0 +1,177 @@ +#ifndef ASM_UNALIGNED_H +#define ASM_UNALIGNED_H + +#include +#include +#include + +#define _LITTLE_ENDIAN 1 + +static unsigned _isLittleEndian(void) +{ + const union { uint32_t u; uint8_t c[4]; } one = { 1 }; + assert(_LITTLE_ENDIAN == one.c[0]); + return _LITTLE_ENDIAN; +} + +static uint16_t _swap16(uint16_t in) +{ + return ((in & 0xF) << 8) + ((in & 0xF0) >> 8); +} + +static uint32_t _swap32(uint32_t in) +{ + return __builtin_bswap32(in); +} + +static uint64_t _swap64(uint64_t in) +{ + return __builtin_bswap64(in); +} + +/* Little endian */ +static uint16_t get_unaligned_le16(const void* memPtr) +{ + uint16_t val; + memcpy(&val, memPtr, sizeof(val)); + if (!_isLittleEndian()) _swap16(val); + return val; +} + +static uint32_t get_unaligned_le32(const void* memPtr) +{ + uint32_t val; + memcpy(&val, memPtr, sizeof(val)); + if (!_isLittleEndian()) _swap32(val); + return val; +} + +static uint64_t get_unaligned_le64(const void* memPtr) +{ + uint64_t val; + memcpy(&val, memPtr, sizeof(val)); + if (!_isLittleEndian()) _swap64(val); + return val; +} + +static void put_unaligned_le16(uint16_t value, void* memPtr) +{ + if (!_isLittleEndian()) value = _swap16(value); + memcpy(memPtr, &value, sizeof(value)); +} + +static void put_unaligned_le32(uint32_t value, void* memPtr) +{ + if (!_isLittleEndian()) value = _swap32(value); + memcpy(memPtr, &value, sizeof(value)); +} + +static void put_unaligned_le64(uint64_t value, void* memPtr) +{ + if (!_isLittleEndian()) value = _swap64(value); + memcpy(memPtr, &value, sizeof(value)); +} + +/* big endian */ +static uint32_t get_unaligned_be32(const void* memPtr) +{ + uint32_t val; + memcpy(&val, memPtr, sizeof(val)); + if (_isLittleEndian()) _swap32(val); + return val; +} + +static uint64_t get_unaligned_be64(const void* memPtr) +{ + uint64_t val; + memcpy(&val, memPtr, sizeof(val)); + if (_isLittleEndian()) _swap64(val); + return val; +} + +static void put_unaligned_be32(uint32_t value, void* memPtr) +{ + if (_isLittleEndian()) value = _swap32(value); + memcpy(memPtr, &value, sizeof(value)); +} + +static void put_unaligned_be64(uint64_t value, void* memPtr) +{ + if (_isLittleEndian()) value = _swap64(value); + memcpy(memPtr, &value, sizeof(value)); +} + +/* generic */ +extern void __bad_unaligned_access_size(void); + +#define __get_unaligned_le(ptr) ((typeof(*(ptr)))({ \ + __builtin_choose_expr(sizeof(*(ptr)) == 1, *(ptr), \ + __builtin_choose_expr(sizeof(*(ptr)) == 2, get_unaligned_le16((ptr)), \ + __builtin_choose_expr(sizeof(*(ptr)) == 4, get_unaligned_le32((ptr)), \ + __builtin_choose_expr(sizeof(*(ptr)) == 8, get_unaligned_le64((ptr)), \ + __bad_unaligned_access_size())))); \ + })) + +#define __get_unaligned_be(ptr) ((typeof(*(ptr)))({ \ + __builtin_choose_expr(sizeof(*(ptr)) == 1, *(ptr), \ + __builtin_choose_expr(sizeof(*(ptr)) == 2, get_unaligned_be16((ptr)), \ + __builtin_choose_expr(sizeof(*(ptr)) == 4, get_unaligned_be32((ptr)), \ + __builtin_choose_expr(sizeof(*(ptr)) == 8, get_unaligned_be64((ptr)), \ + __bad_unaligned_access_size())))); \ + })) + +#define __put_unaligned_le(val, ptr) \ + ({ \ + void *__gu_p = (ptr); \ + switch (sizeof(*(ptr))) { \ + case 1: \ + *(uint8_t *)__gu_p = (uint8_t)(val); \ + break; \ + case 2: \ + put_unaligned_le16((uint16_t)(val), __gu_p); \ + break; \ + case 4: \ + put_unaligned_le32((uint32_t)(val), __gu_p); \ + break; \ + case 8: \ + put_unaligned_le64((uint64_t)(val), __gu_p); \ + break; \ + default: \ + __bad_unaligned_access_size(); \ + break; \ + } \ + (void)0; \ + }) + +#define __put_unaligned_be(val, ptr) \ + ({ \ + void *__gu_p = (ptr); \ + switch (sizeof(*(ptr))) { \ + case 1: \ + *(uint8_t *)__gu_p = (uint8_t)(val); \ + break; \ + case 2: \ + put_unaligned_be16((uint16_t)(val), __gu_p); \ + break; \ + case 4: \ + put_unaligned_be32((uint32_t)(val), __gu_p); \ + break; \ + case 8: \ + put_unaligned_be64((uint64_t)(val), __gu_p); \ + break; \ + default: \ + __bad_unaligned_access_size(); \ + break; \ + } \ + (void)0; \ + }) + +#if _LITTLE_ENDIAN +# define get_unaligned __get_unaligned_le +# define put_unaligned __put_unaligned_le +#else +# define get_unaligned __get_unaligned_be +# define put_unaligned __put_unaligned_be +#endif + +#endif // ASM_UNALIGNED_H diff --git a/contrib/linux-kernel/test/include/linux/compiler.h b/contrib/linux-kernel/test/include/linux/compiler.h new file mode 100644 index 000000000..7991b8b29 --- /dev/null +++ b/contrib/linux-kernel/test/include/linux/compiler.h @@ -0,0 +1,12 @@ +#ifndef LINUX_COMIPLER_H_ +#define LINUX_COMIPLER_H_ + +#ifndef __always_inline +# define __always_inline inline +#endif + +#ifndef noinline +# define noinline __attribute__((__noinline__)) +#endif + +#endif // LINUX_COMIPLER_H_ diff --git a/contrib/linux-kernel/test/include/linux/kernel.h b/contrib/linux-kernel/test/include/linux/kernel.h new file mode 100644 index 000000000..b208e23ba --- /dev/null +++ b/contrib/linux-kernel/test/include/linux/kernel.h @@ -0,0 +1,14 @@ +#ifndef LINUX_KERNEL_H_ +#define LINUX_KERNEL_H_ + +#define ALIGN(x, a) ({ \ + typeof(x) const __xe = (x); \ + typeof(a) const __ae = (a); \ + typeof(a) const __m = __ae - 1; \ + typeof(x) const __r = __xe & __m; \ + __xe + (__r ? (__ae - __r) : 0); \ + }) + +#define PTR_ALIGN(p, a) (typeof(p))ALIGN((unsigned long long)(p), (a)) + +#endif // LINUX_KERNEL_H_ diff --git a/contrib/linux-kernel/test/include/linux/string.h b/contrib/linux-kernel/test/include/linux/string.h new file mode 100644 index 000000000..3b2f59002 --- /dev/null +++ b/contrib/linux-kernel/test/include/linux/string.h @@ -0,0 +1 @@ +#include diff --git a/contrib/linux-kernel/test/include/linux/types.h b/contrib/linux-kernel/test/include/linux/types.h new file mode 100644 index 000000000..c2d4f4b72 --- /dev/null +++ b/contrib/linux-kernel/test/include/linux/types.h @@ -0,0 +1,2 @@ +#include +#include From b5e3e3c9a82577fb2b3f9ad0e0a95188cd7f565d Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 12:10:47 -0700 Subject: [PATCH 120/305] Add zstd kernel module Makefile --- contrib/linux-kernel/lib/Makefile | 9 +++++++++ 1 file changed, 9 insertions(+) create mode 100644 contrib/linux-kernel/lib/Makefile diff --git a/contrib/linux-kernel/lib/Makefile b/contrib/linux-kernel/lib/Makefile new file mode 100644 index 000000000..53b4deec9 --- /dev/null +++ b/contrib/linux-kernel/lib/Makefile @@ -0,0 +1,9 @@ +obj-$(CONFIG_ZSTD_COMPRESS) += zstd_compress.o +obj-$(CONFIG_ZSTD_DECOMPRESS) += zstd_decompress.o + +ccflags-y += -O3 + +zstd_compress-y := entropy_common.o fse_decompress.o xxhash.o zstd_common.o \ + fse_compress.o huf_compress.o zstd_compress.o +zstd_decompress-y := entropy_common.o fse_decompress.o xxhash.o zstd_common.o \ + huf_decompress.o zstd_decompress.o From 7cf78f1be76473b84a1e505718eb1e2b4e0a3ae0 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 4 Apr 2017 12:38:14 -0700 Subject: [PATCH 121/305] Protects ZSTD_compressBegin_usingCDict() vs NULL cdict dereference Will issue an error (GENERIC) is cdict==NULL --- lib/compress/zstd_compress.c | 5 +++-- lib/zstd.h | 2 +- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 8318843be..b3e0b4b3a 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -2898,6 +2898,7 @@ static ZSTD_parameters ZSTD_getParamsFromCDict(const ZSTD_CDict* cdict) { size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize) { + if (cdict==NULL) return ERROR(GENERIC); /* does not support NULL cdict */ if (cdict->dictContentSize) CHECK_F(ZSTD_copyCCtx(cctx, cdict->refContext, pledgedSrcSize)) else { ZSTD_parameters params = cdict->refContext->params; @@ -2916,9 +2917,9 @@ size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, const void* src, size_t srcSize, const ZSTD_CDict* cdict) { - CHECK_F(ZSTD_compressBegin_usingCDict(cctx, cdict, srcSize)); + CHECK_F(ZSTD_compressBegin_usingCDict(cctx, cdict, srcSize)); /* will check if cdict != NULL */ - if (cdict->refContext->params.fParams.contentSizeFlag==1) { + if (cdict->refContext->params.fParams.contentSizeFlag == 1) { cctx->params.fParams.contentSizeFlag = 1; cctx->frameContentSize = srcSize; } else { diff --git a/lib/zstd.h b/lib/zstd.h index 4531a84bf..153e244be 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -651,7 +651,7 @@ ZSTDLIB_API size_t ZSTD_compressBegin(ZSTD_CCtx* cctx, int compressionLevel); ZSTDLIB_API size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel); ZSTDLIB_API size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */ ZSTDLIB_API size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**< note: if pledgedSrcSize can be 0, indicating unknown size. if it is non-zero, it must be accurate. for 0 size frames, use compressBegin_advanced */ -ZSTDLIB_API size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: if pledgedSrcSize can be 0, indicating unknown size. if it is non-zero, it must be accurate. for 0 size frames, use compressBegin_advanced */ +ZSTDLIB_API size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: fail if cdict==NULL. pledgedSrcSize can be 0, indicating unknown size. For 0 size frames, use compressBegin_advanced */ ZSTDLIB_API size_t ZSTD_compressContinue(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); ZSTDLIB_API size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); From b1b582b9fa9177e3e18fef81d63a1e6ecb74c7ae Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 12:56:35 -0700 Subject: [PATCH 122/305] Add module macros --- contrib/linux-kernel/include/zstd.h | 10 +++--- contrib/linux-kernel/lib/zstd_common.c | 6 ---- contrib/linux-kernel/lib/zstd_compress.c | 42 ++++++++++++++++++++++ contrib/linux-kernel/lib/zstd_decompress.c | 42 ++++++++++++++++++++++ 4 files changed, 90 insertions(+), 10 deletions(-) diff --git a/contrib/linux-kernel/include/zstd.h b/contrib/linux-kernel/include/zstd.h index 9d8fc8e0b..101e3cb21 100644 --- a/contrib/linux-kernel/include/zstd.h +++ b/contrib/linux-kernel/include/zstd.h @@ -49,7 +49,6 @@ #define ZSTD_VERSION_STRING ZSTD_EXPAND_AND_QUOTE(ZSTD_LIB_VERSION) #define ZSTD_VERSION_NUMBER (ZSTD_VERSION_MAJOR *100*100 + ZSTD_VERSION_MINOR *100 + ZSTD_VERSION_RELEASE) -ZSTDLIB_API unsigned ZSTD_versionNumber(void); /**< library version number; to be used when checking dll version */ /*====== Helper functions ======*/ @@ -123,7 +122,6 @@ typedef struct { } ZSTD_parameters; size_t ZSTD_CCtxWorkspaceBound(ZSTD_compressionParameters params); -size_t ZSTD_DCtxWorkspaceBound(void); /*= Compression context * When compressing many times, @@ -139,6 +137,8 @@ ZSTDLIB_API size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapaci ZSTDLIB_API size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void *dict, size_t dictSize, ZSTD_parameters params); +size_t ZSTD_DCtxWorkspaceBound(void); + /*= Decompression context * When decompressing many times, * it is recommended to allocate a context just once, and re-use it for each successive compression operation. @@ -157,7 +157,6 @@ ZSTDLIB_API size_t ZSTD_decompress_usingDict(ZSTD_DCtx* ctx, void* dst, size_t d * Fast dictionary API ****************************/ size_t ZSTD_CDictWorkspaceBound(ZSTD_compressionParameters params); -size_t ZSTD_DDictWorkspaceBound(void); typedef struct ZSTD_CDict_s ZSTD_CDict; @@ -178,6 +177,8 @@ ZSTDLIB_API size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict); +size_t ZSTD_DDictWorkspaceBound(void); + typedef struct ZSTD_DDict_s ZSTD_DDict; /*! ZSTD_createDDict() : @@ -252,7 +253,6 @@ typedef struct ZSTD_outBuffer_s { * *******************************************************************/ size_t ZSTD_CStreamWorkspaceBound(ZSTD_compressionParameters params); -size_t ZSTD_DStreamWorkspaceBound(size_t maxWindowSize); typedef struct ZSTD_CStream_s ZSTD_CStream; /*===== ZSTD_CStream management functions =====*/ @@ -292,6 +292,8 @@ ZSTDLIB_API size_t ZSTD_CStreamOutSize(void); /**< recommended size for output * The return value is a suggested next input size (a hint to improve latency) that will never load more than the current frame. * *******************************************************************************/ +size_t ZSTD_DStreamWorkspaceBound(size_t maxWindowSize); + typedef struct ZSTD_DStream_s ZSTD_DStream; /*===== ZSTD_DStream management functions =====*/ ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream(size_t maxWindowSize, void* workspace, size_t workspaceSize); diff --git a/contrib/linux-kernel/lib/zstd_common.c b/contrib/linux-kernel/lib/zstd_common.c index ca9382c67..106f54055 100644 --- a/contrib/linux-kernel/lib/zstd_common.c +++ b/contrib/linux-kernel/lib/zstd_common.c @@ -17,12 +17,6 @@ #include -/*-**************************************** -* Version -******************************************/ -unsigned ZSTD_versionNumber (void) { return ZSTD_VERSION_NUMBER; } - - /*=************************************************************** * Custom allocator ****************************************************************/ diff --git a/contrib/linux-kernel/lib/zstd_compress.c b/contrib/linux-kernel/lib/zstd_compress.c index e57cd6baa..d4b87f280 100644 --- a/contrib/linux-kernel/lib/zstd_compress.c +++ b/contrib/linux-kernel/lib/zstd_compress.c @@ -12,6 +12,7 @@ * Dependencies ***************************************/ #include +#include #include /* memset */ #include "mem.h" #include "fse.h" @@ -3337,3 +3338,44 @@ ZSTD_parameters ZSTD_getParams(int compressionLevel, unsigned long long srcSize, params.cParams = cParams; return params; } + +EXPORT_SYMBOL(ZSTD_maxCLevel); +EXPORT_SYMBOL(ZSTD_compressBound); + +EXPORT_SYMBOL(ZSTD_CCtxWorkspaceBound); +EXPORT_SYMBOL(ZSTD_createCCtx); +EXPORT_SYMBOL(ZSTD_compressCCtx); +EXPORT_SYMBOL(ZSTD_compress_usingDict); + +EXPORT_SYMBOL(ZSTD_CDictWorkspaceBound); +EXPORT_SYMBOL(ZSTD_createCDict); +EXPORT_SYMBOL(ZSTD_compress_usingCDict); + +EXPORT_SYMBOL(ZSTD_CStreamWorkspaceBound); +EXPORT_SYMBOL(ZSTD_createCStream); +EXPORT_SYMBOL(ZSTD_createCStream_usingCDict); +EXPORT_SYMBOL(ZSTD_resetCStream); +EXPORT_SYMBOL(ZSTD_compressStream); +EXPORT_SYMBOL(ZSTD_flushStream); +EXPORT_SYMBOL(ZSTD_endStream); +EXPORT_SYMBOL(ZSTD_CStreamInSize); +EXPORT_SYMBOL(ZSTD_CStreamOutSize); + +EXPORT_SYMBOL(ZSTD_getCParams); +EXPORT_SYMBOL(ZSTD_getParams); +EXPORT_SYMBOL(ZSTD_checkCParams); +EXPORT_SYMBOL(ZSTD_adjustCParams); + +EXPORT_SYMBOL(ZSTD_compressBegin); +EXPORT_SYMBOL(ZSTD_compressBegin_usingDict); +EXPORT_SYMBOL(ZSTD_compressBegin_advanced); +EXPORT_SYMBOL(ZSTD_copyCCtx); +EXPORT_SYMBOL(ZSTD_compressBegin_usingCDict); +EXPORT_SYMBOL(ZSTD_compressContinue); +EXPORT_SYMBOL(ZSTD_compressEnd); + +EXPORT_SYMBOL(ZSTD_getBlockSizeMax); +EXPORT_SYMBOL(ZSTD_compressBlock); + +MODULE_LICENSE("BSD"); +MODULE_DESCRIPTION("Zstd Compressor"); diff --git a/contrib/linux-kernel/lib/zstd_decompress.c b/contrib/linux-kernel/lib/zstd_decompress.c index 379806e63..9eb210af0 100644 --- a/contrib/linux-kernel/lib/zstd_decompress.c +++ b/contrib/linux-kernel/lib/zstd_decompress.c @@ -24,6 +24,8 @@ /*-******************************************************* * Dependencies *********************************************************/ +#include +#include #include /* memcpy, memmove, memset */ #include "mem.h" /* low level memory routines */ #include "fse.h" @@ -2333,3 +2335,43 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB return nextSrcSizeHint; } } + +EXPORT_SYMBOL(ZSTD_DCtxWorkspaceBound); +EXPORT_SYMBOL(ZSTD_createDCtx); +EXPORT_SYMBOL(ZSTD_decompressDCtx); +EXPORT_SYMBOL(ZSTD_decompress_usingDict); + +EXPORT_SYMBOL(ZSTD_DDictWorkspaceBound); +EXPORT_SYMBOL(ZSTD_createDDict); +EXPORT_SYMBOL(ZSTD_decompress_usingDDict); + +EXPORT_SYMBOL(ZSTD_DStreamWorkspaceBound); +EXPORT_SYMBOL(ZSTD_createDStream); +EXPORT_SYMBOL(ZSTD_createDStream_usingDDict); +EXPORT_SYMBOL(ZSTD_resetDStream); +EXPORT_SYMBOL(ZSTD_decompressStream); +EXPORT_SYMBOL(ZSTD_DStreamInSize); +EXPORT_SYMBOL(ZSTD_DStreamOutSize); + +EXPORT_SYMBOL(ZSTD_findFrameCompressedSize); +EXPORT_SYMBOL(ZSTD_getFrameContentSize); +EXPORT_SYMBOL(ZSTD_findDecompressedSize); + +EXPORT_SYMBOL(ZSTD_isFrame); +EXPORT_SYMBOL(ZSTD_getDictID_fromDict); +EXPORT_SYMBOL(ZSTD_getDictID_fromDDict); +EXPORT_SYMBOL(ZSTD_getDictID_fromFrame); + +EXPORT_SYMBOL(ZSTD_getFrameParams); +EXPORT_SYMBOL(ZSTD_decompressBegin); +EXPORT_SYMBOL(ZSTD_decompressBegin_usingDict); +EXPORT_SYMBOL(ZSTD_copyDCtx); +EXPORT_SYMBOL(ZSTD_nextSrcSizeToDecompress); +EXPORT_SYMBOL(ZSTD_decompressContinue); +EXPORT_SYMBOL(ZSTD_nextInputType); + +EXPORT_SYMBOL(ZSTD_decompressBlock); +EXPORT_SYMBOL(ZSTD_insertBlock); + +MODULE_LICENSE("BSD"); +MODULE_DESCRIPTION("Zstd Decompressor"); From b06507221e263047bb82031a6f5fe245e14123ab Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 12:56:44 -0700 Subject: [PATCH 123/305] Test module macros --- contrib/linux-kernel/test/UserlandTest.cpp | 75 +++++++++++++++++++ .../linux-kernel/test/include/linux/module.h | 10 +++ 2 files changed, 85 insertions(+) create mode 100644 contrib/linux-kernel/test/include/linux/module.h diff --git a/contrib/linux-kernel/test/UserlandTest.cpp b/contrib/linux-kernel/test/UserlandTest.cpp index 7bdbd1bf7..5cff66180 100644 --- a/contrib/linux-kernel/test/UserlandTest.cpp +++ b/contrib/linux-kernel/test/UserlandTest.cpp @@ -477,3 +477,78 @@ TEST(Stream, Flush) { } EXPECT_EQ(kData, decompressed); } + +#define TEST_SYMBOL(symbol) \ + do { \ + extern void *__##symbol; \ + EXPECT_NE((void *)0, __##symbol); \ + } while (0) + +TEST(API, Symbols) { + TEST_SYMBOL(ZSTD_CCtxWorkspaceBound); + TEST_SYMBOL(ZSTD_createCCtx); + TEST_SYMBOL(ZSTD_compressCCtx); + TEST_SYMBOL(ZSTD_compress_usingDict); + TEST_SYMBOL(ZSTD_DCtxWorkspaceBound); + TEST_SYMBOL(ZSTD_createDCtx); + TEST_SYMBOL(ZSTD_decompressDCtx); + TEST_SYMBOL(ZSTD_decompress_usingDict); + + TEST_SYMBOL(ZSTD_CDictWorkspaceBound); + TEST_SYMBOL(ZSTD_createCDict); + TEST_SYMBOL(ZSTD_compress_usingCDict); + TEST_SYMBOL(ZSTD_DDictWorkspaceBound); + TEST_SYMBOL(ZSTD_createDDict); + TEST_SYMBOL(ZSTD_decompress_usingDDict); + + TEST_SYMBOL(ZSTD_CStreamWorkspaceBound); + TEST_SYMBOL(ZSTD_createCStream); + TEST_SYMBOL(ZSTD_createCStream_usingCDict); + TEST_SYMBOL(ZSTD_resetCStream); + TEST_SYMBOL(ZSTD_compressStream); + TEST_SYMBOL(ZSTD_flushStream); + TEST_SYMBOL(ZSTD_endStream); + TEST_SYMBOL(ZSTD_CStreamInSize); + TEST_SYMBOL(ZSTD_CStreamOutSize); + TEST_SYMBOL(ZSTD_DStreamWorkspaceBound); + TEST_SYMBOL(ZSTD_createDStream); + TEST_SYMBOL(ZSTD_createDStream_usingDDict); + TEST_SYMBOL(ZSTD_resetDStream); + TEST_SYMBOL(ZSTD_decompressStream); + TEST_SYMBOL(ZSTD_DStreamInSize); + TEST_SYMBOL(ZSTD_DStreamOutSize); + + TEST_SYMBOL(ZSTD_findFrameCompressedSize); + TEST_SYMBOL(ZSTD_getFrameContentSize); + TEST_SYMBOL(ZSTD_findDecompressedSize); + + TEST_SYMBOL(ZSTD_getCParams); + TEST_SYMBOL(ZSTD_getParams); + TEST_SYMBOL(ZSTD_checkCParams); + TEST_SYMBOL(ZSTD_adjustCParams); + + TEST_SYMBOL(ZSTD_isFrame); + TEST_SYMBOL(ZSTD_getDictID_fromDict); + TEST_SYMBOL(ZSTD_getDictID_fromDDict); + TEST_SYMBOL(ZSTD_getDictID_fromFrame); + + TEST_SYMBOL(ZSTD_compressBegin); + TEST_SYMBOL(ZSTD_compressBegin_usingDict); + TEST_SYMBOL(ZSTD_compressBegin_advanced); + TEST_SYMBOL(ZSTD_copyCCtx); + TEST_SYMBOL(ZSTD_compressBegin_usingCDict); + TEST_SYMBOL(ZSTD_compressContinue); + TEST_SYMBOL(ZSTD_compressEnd); + TEST_SYMBOL(ZSTD_getFrameParams); + TEST_SYMBOL(ZSTD_decompressBegin); + TEST_SYMBOL(ZSTD_decompressBegin_usingDict); + TEST_SYMBOL(ZSTD_copyDCtx); + TEST_SYMBOL(ZSTD_nextSrcSizeToDecompress); + TEST_SYMBOL(ZSTD_decompressContinue); + TEST_SYMBOL(ZSTD_nextInputType); + + TEST_SYMBOL(ZSTD_getBlockSizeMax); + TEST_SYMBOL(ZSTD_compressBlock); + TEST_SYMBOL(ZSTD_decompressBlock); + TEST_SYMBOL(ZSTD_insertBlock); +} diff --git a/contrib/linux-kernel/test/include/linux/module.h b/contrib/linux-kernel/test/include/linux/module.h new file mode 100644 index 000000000..ef514c349 --- /dev/null +++ b/contrib/linux-kernel/test/include/linux/module.h @@ -0,0 +1,10 @@ +#ifndef LINUX_MODULE_H_ +#define LINUX_MODULE_H_ + +#define EXPORT_SYMBOL(symbol) \ + void* __##symbol = symbol +#define MODULE_LICENSE(license) static char const *const LICENSE = license +#define MODULE_DESCRIPTION(description) \ + static char const *const DESCRIPTION = description + +#endif // LINUX_MODULE_H_ From 81d638013958835f85f237f1e134a44708eb9e9c Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 4 Apr 2017 15:21:09 -0700 Subject: [PATCH 124/305] minor bench.c adjustments shorter debug messages no need to check decompressedLength==0 twice --- programs/bench.c | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/programs/bench.c b/programs/bench.c index d0dac497a..712f73fd8 100644 --- a/programs/bench.c +++ b/programs/bench.c @@ -89,7 +89,7 @@ static clock_t g_time = 0; #define DEBUGOUTPUT(...) if (DEBUG) DISPLAY(__VA_ARGS__); #define EXM_THROW(error, ...) \ { \ - DEBUGOUTPUT("Error defined at %s, line %i : \n", __FILE__, __LINE__); \ + DEBUGOUTPUT("%s: %i: \n", __FILE__, __LINE__); \ DISPLAYLEVEL(1, "Error %i : ", error); \ DISPLAYLEVEL(1, __VA_ARGS__); \ DISPLAYLEVEL(1, " \n"); \ @@ -190,8 +190,7 @@ static int BMK_benchMem(const void* srcBuffer, size_t srcSize, srcPtr += fileSizes[fileNb]; } { size_t const decodedSize = (size_t)dSize64; - if (dSize64 > decodedSize) EXM_THROW(32, "original size is too large"); - if (decodedSize==0) EXM_THROW(32, "Impossible to determine original size "); + if (dSize64 > decodedSize) EXM_THROW(32, "original size is too large"); /* size_t overflow */ free(resultBuffer); resultBuffer = malloc(decodedSize); if (!resultBuffer) EXM_THROW(33, "not enough memory"); From c2007388a556613f2e3ce1f810f5ff2cee4dfb9c Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 4 Apr 2017 15:35:06 -0700 Subject: [PATCH 125/305] fixed bench.c : optional advanced parameters applied before creating cdict --- programs/bench.c | 23 ++++++++++++----------- tests/paramgrill.c | 6 +++++- 2 files changed, 17 insertions(+), 12 deletions(-) diff --git a/programs/bench.c b/programs/bench.c index 712f73fd8..d25ff8f04 100644 --- a/programs/bench.c +++ b/programs/bench.c @@ -153,7 +153,7 @@ static int BMK_benchMem(const void* srcBuffer, size_t srcSize, const char* displayName, int cLevel, const size_t* fileSizes, U32 nbFiles, const void* dictBuffer, size_t dictBufferSize, - ZSTD_compressionParameters *comprParams) + const ZSTD_compressionParameters* comprParams) { size_t const blockSize = ((g_blockSize>=32 && !g_decodeOnly) ? g_blockSize : srcSize) + (!srcSize) /* avoid div by 0 */ ; size_t const avgSize = MIN(blockSize, (srcSize / nbFiles)); @@ -176,21 +176,21 @@ static int BMK_benchMem(const void* srcBuffer, size_t srcSize, EXM_THROW(31, "allocation error : not enough memory"); /* init */ - if (strlen(displayName)>17) displayName += strlen(displayName)-17; /* can only display 17 characters */ + if (strlen(displayName)>17) displayName += strlen(displayName)-17; /* display last 17 characters */ UTIL_initTimer(&ticksPerSecond); if (g_decodeOnly) { /* benchmark only decompression : source must be already compressed */ - const char* srcPtr = (const char*) srcBuffer; - U64 dSize64 = 0; + const char* srcPtr = (const char*)srcBuffer; + U64 totalDSize64 = 0; U32 fileNb; for (fileNb=0; fileNb decodedSize) EXM_THROW(32, "original size is too large"); /* size_t overflow */ + { size_t const decodedSize = (size_t)totalDSize64; + if (totalDSize64 > decodedSize) EXM_THROW(32, "original size is too large"); /* size_t overflow */ free(resultBuffer); resultBuffer = malloc(decodedSize); if (!resultBuffer) EXM_THROW(33, "not enough memory"); @@ -259,12 +259,11 @@ static int BMK_benchMem(const void* srcBuffer, size_t srcSize, UTIL_getTime(&clockStart); if (!cCompleted) { /* still some time to do compression tests */ - ZSTD_parameters zparams = ZSTD_getParams(cLevel, avgSize, dictBufferSize); ZSTD_customMem const cmem = { NULL, NULL, NULL }; - U64 clockLoop = g_nbSeconds ? TIMELOOP_MICROSEC : 1; + U64 const clockLoop = g_nbSeconds ? TIMELOOP_MICROSEC : 1; U32 nbLoops = 0; - ZSTD_CDict* const cdict = ZSTD_createCDict_advanced(dictBuffer, dictBufferSize, 1, zparams, cmem); - if (cdict==NULL) EXM_THROW(1, "ZSTD_createCDict_advanced() allocation failure"); + ZSTD_parameters zparams = ZSTD_getParams(cLevel, avgSize, dictBufferSize); + ZSTD_CDict* cdict; if (comprParams->windowLog) zparams.cParams.windowLog = comprParams->windowLog; if (comprParams->chainLog) zparams.cParams.chainLog = comprParams->chainLog; if (comprParams->hashLog) zparams.cParams.hashLog = comprParams->hashLog; @@ -272,6 +271,8 @@ static int BMK_benchMem(const void* srcBuffer, size_t srcSize, if (comprParams->searchLength) zparams.cParams.searchLength = comprParams->searchLength; if (comprParams->targetLength) zparams.cParams.targetLength = comprParams->targetLength; if (comprParams->strategy) zparams.cParams.strategy = (ZSTD_strategy)(comprParams->strategy - 1); + cdict = ZSTD_createCDict_advanced(dictBuffer, dictBufferSize, 1, zparams, cmem); + if (cdict==NULL) EXM_THROW(1, "ZSTD_createCDict_advanced() allocation failure"); do { U32 blockNb; size_t rSize; diff --git a/tests/paramgrill.c b/tests/paramgrill.c index 5eabcba2b..364c0e4b2 100644 --- a/tests/paramgrill.c +++ b/tests/paramgrill.c @@ -106,7 +106,11 @@ static size_t BMK_findMaxMem(U64 requiredMem) } -# define FUZ_rotl32(x,r) ((x << r) | (x >> (32 - r))) +static U32 FUZ_rotl32(U32 x, U32 r) +{ + return ((x << r) | (x >> (32 - r))); +} + U32 FUZ_rand(U32* src) { const U32 prime1 = 2654435761U; From 675839254d93a0f92852d6ba28e2c4abda3c6d03 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 15:41:24 -0700 Subject: [PATCH 126/305] Move zstd.h to linux/zstd.h --- contrib/linux-kernel/include/{ => linux}/zstd.h | 0 contrib/linux-kernel/lib/error_private.h | 2 +- contrib/linux-kernel/lib/zstd_internal.h | 2 +- contrib/linux-kernel/test/UserlandTest.cpp | 2 +- 4 files changed, 3 insertions(+), 3 deletions(-) rename contrib/linux-kernel/include/{ => linux}/zstd.h (100%) diff --git a/contrib/linux-kernel/include/zstd.h b/contrib/linux-kernel/include/linux/zstd.h similarity index 100% rename from contrib/linux-kernel/include/zstd.h rename to contrib/linux-kernel/include/linux/zstd.h diff --git a/contrib/linux-kernel/lib/error_private.h b/contrib/linux-kernel/lib/error_private.h index 9bfbe4b5c..8cf148bc4 100644 --- a/contrib/linux-kernel/lib/error_private.h +++ b/contrib/linux-kernel/lib/error_private.h @@ -16,7 +16,7 @@ * Dependencies ******************************************/ #include /* size_t */ -#include "zstd.h" /* enum list */ +#include /* enum list */ /* **************************************** diff --git a/contrib/linux-kernel/lib/zstd_internal.h b/contrib/linux-kernel/lib/zstd_internal.h index 66f318af6..479d6827d 100644 --- a/contrib/linux-kernel/lib/zstd_internal.h +++ b/contrib/linux-kernel/lib/zstd_internal.h @@ -22,9 +22,9 @@ ***************************************/ #include #include +#include #include "mem.h" #include "error_private.h" -#include "zstd.h" #include "xxhash.h" /* XXH_reset, update, digest */ diff --git a/contrib/linux-kernel/test/UserlandTest.cpp b/contrib/linux-kernel/test/UserlandTest.cpp index 5cff66180..556abcdaf 100644 --- a/contrib/linux-kernel/test/UserlandTest.cpp +++ b/contrib/linux-kernel/test/UserlandTest.cpp @@ -1,5 +1,5 @@ extern "C" { -#include +#include } #include #include From 0888251fb1bd95c116486402f2f543ec16fe0f6a Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 15:49:38 -0700 Subject: [PATCH 127/305] Switch intptr_t to ptrdiff_t --- contrib/linux-kernel/lib/mem.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/contrib/linux-kernel/lib/mem.h b/contrib/linux-kernel/lib/mem.h index 6040aa5c7..76cae04fa 100644 --- a/contrib/linux-kernel/lib/mem.h +++ b/contrib/linux-kernel/lib/mem.h @@ -38,7 +38,7 @@ typedef uint32_t U32; typedef int32_t S32; typedef uint64_t U64; typedef int64_t S64; -typedef intptr_t iPtrDiff; +typedef ptrdiff_t iPtrDiff; typedef uintptr_t uPtrDiff; From 9c257dc268df4b6eb40e4292035c555bf9295fd6 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 15:54:39 -0700 Subject: [PATCH 128/305] Fix up xxhash --- contrib/linux-kernel/lib/xxhash.c | 11 ----------- contrib/linux-kernel/lib/xxhash.h | 4 ++-- 2 files changed, 2 insertions(+), 13 deletions(-) diff --git a/contrib/linux-kernel/lib/xxhash.c b/contrib/linux-kernel/lib/xxhash.c index 05ebedf7f..43c2c5296 100644 --- a/contrib/linux-kernel/lib/xxhash.c +++ b/contrib/linux-kernel/lib/xxhash.c @@ -81,17 +81,6 @@ static void* XXH_memcpy(void* dest, const void* src, size_t size) { return memcp #define FORCE_INLINE static __always_inline -static U32 XXH_read32(const void* memPtr) -{ - return MEM_read32(memPtr); -} - -static U64 XXH_read64(const void* memPtr) -{ - return MEM_read64(memPtr); -} - - /* **************************************** * Compiler-specific Functions and Macros ******************************************/ diff --git a/contrib/linux-kernel/lib/xxhash.h b/contrib/linux-kernel/lib/xxhash.h index 36d140eca..974a81c48 100644 --- a/contrib/linux-kernel/lib/xxhash.h +++ b/contrib/linux-kernel/lib/xxhash.h @@ -177,8 +177,8 @@ When done, free XXH state space if it was allocated dynamically. /* ************************** * Utils ****************************/ -XXH_PUBLIC_API void XXH32_copyState(XXH32_state_t* restrict dst_state, const XXH32_state_t* restrict src_state); -XXH_PUBLIC_API void XXH64_copyState(XXH64_state_t* restrict dst_state, const XXH64_state_t* restrict src_state); +XXH_PUBLIC_API void XXH32_copyState(XXH32_state_t* dst_state, const XXH32_state_t* src_state); +XXH_PUBLIC_API void XXH64_copyState(XXH64_state_t* dst_state, const XXH64_state_t* src_state); /* ************************** From dd62829ccf6c84e6a93c16c663a59b8ae200325f Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 15:55:48 -0700 Subject: [PATCH 129/305] Remove more restrict --- contrib/linux-kernel/lib/xxhash.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/contrib/linux-kernel/lib/xxhash.c b/contrib/linux-kernel/lib/xxhash.c index 43c2c5296..0d301ad86 100644 --- a/contrib/linux-kernel/lib/xxhash.c +++ b/contrib/linux-kernel/lib/xxhash.c @@ -165,12 +165,12 @@ XXH_PUBLIC_API unsigned XXH_versionNumber (void) { return XXH_VERSION_NUMBER; } /* ************************** * Utils ****************************/ -XXH_PUBLIC_API void XXH32_copyState(XXH32_state_t* restrict dstState, const XXH32_state_t* restrict srcState) +XXH_PUBLIC_API void XXH32_copyState(XXH32_state_t* dstState, const XXH32_state_t* srcState) { memcpy(dstState, srcState, sizeof(*dstState)); } -XXH_PUBLIC_API void XXH64_copyState(XXH64_state_t* restrict dstState, const XXH64_state_t* restrict srcState) +XXH_PUBLIC_API void XXH64_copyState(XXH64_state_t* dstState, const XXH64_state_t* srcState) { memcpy(dstState, srcState, sizeof(*dstState)); } From 16a739cab0f69ecc2848c2536eb2d61e7a012da3 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 16:17:21 -0700 Subject: [PATCH 130/305] Switch call of FSE_count() to FSE_count_wksp() --- lib/common/fse.h | 2 +- lib/compress/fse_compress.c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/lib/common/fse.h b/lib/common/fse.h index baac39032..5e43215e3 100644 --- a/lib/common/fse.h +++ b/lib/common/fse.h @@ -353,7 +353,7 @@ unsigned FSE_optimalTableLog_internal(unsigned maxTableLog, size_t srcSize, unsi * Same as FSE_compress2(), but using an externally allocated scratch buffer (`workSpace`). * FSE_WKSP_SIZE_U32() provides the minimum size required for `workSpace` as a table of FSE_CTable. */ -#define FSE_WKSP_SIZE_U32(maxTableLog, maxSymbolValue) ( FSE_CTABLE_SIZE_U32(maxTableLog, maxSymbolValue) + (1<<((maxTableLog>2)?(maxTableLog-2):0)) ) +#define FSE_WKSP_SIZE_U32(maxTableLog, maxSymbolValue) ( FSE_CTABLE_SIZE_U32(maxTableLog, maxSymbolValue) + ((maxTableLog > 12) ? (1 << (maxTableLog - 2)) : 1024) ) size_t FSE_compress_wksp (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize); size_t FSE_buildCTable_raw (FSE_CTable* ct, unsigned nbBits); diff --git a/lib/compress/fse_compress.c b/lib/compress/fse_compress.c index 13654d6e6..0b3b24286 100644 --- a/lib/compress/fse_compress.c +++ b/lib/compress/fse_compress.c @@ -808,7 +808,7 @@ size_t FSE_compress_wksp (void* dst, size_t dstSize, const void* src, size_t src if (!tableLog) tableLog = FSE_DEFAULT_TABLELOG; /* Scan input and build symbol stats */ - { CHECK_V_F(maxCount, FSE_count(count, &maxSymbolValue, src, srcSize) ); + { CHECK_V_F(maxCount, FSE_count_wksp(count, &maxSymbolValue, src, srcSize, scratchBuffer) ); if (maxCount == srcSize) return 1; /* only a single symbol in src : rle */ if (maxCount == 1) return 0; /* each symbol present maximum once => not compressible */ if (maxCount < (srcSize >> 7)) return 0; /* Heuristic : not compressible enough */ From 1af700ea43bc78bfd5b5a4bfbfbd0a55f6778cf6 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 16:22:58 -0700 Subject: [PATCH 131/305] Remove unused FSE/HUF functions --- contrib/linux-kernel/lib/fse.h | 2 +- contrib/linux-kernel/lib/fse_compress.c | 10 +--------- contrib/linux-kernel/lib/huf.h | 1 - contrib/linux-kernel/lib/huf_compress.c | 9 --------- 4 files changed, 2 insertions(+), 20 deletions(-) diff --git a/contrib/linux-kernel/lib/fse.h b/contrib/linux-kernel/lib/fse.h index fbd97fae3..14fa439ee 100644 --- a/contrib/linux-kernel/lib/fse.h +++ b/contrib/linux-kernel/lib/fse.h @@ -272,7 +272,7 @@ unsigned FSE_optimalTableLog_internal(unsigned maxTableLog, size_t srcSize, unsi * Same as FSE_compress2(), but using an externally allocated scratch buffer (`workSpace`). * FSE_WKSP_SIZE_U32() provides the minimum size required for `workSpace` as a table of FSE_CTable. */ -#define FSE_WKSP_SIZE_U32(maxTableLog, maxSymbolValue) ( FSE_CTABLE_SIZE_U32(maxTableLog, maxSymbolValue) + (1<<((maxTableLog>2)?(maxTableLog-2):0)) ) +#define FSE_WKSP_SIZE_U32(maxTableLog, maxSymbolValue) ( FSE_CTABLE_SIZE_U32(maxTableLog, maxSymbolValue) + ((maxTableLog > 12) ? (1 << (maxTableLog - 2)) : 1024) ) size_t FSE_compress_wksp (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize); size_t FSE_buildCTable_raw (FSE_CTable* ct, unsigned nbBits); diff --git a/contrib/linux-kernel/lib/fse_compress.c b/contrib/linux-kernel/lib/fse_compress.c index 7c13a4099..e969fd2ca 100644 --- a/contrib/linux-kernel/lib/fse_compress.c +++ b/contrib/linux-kernel/lib/fse_compress.c @@ -404,14 +404,6 @@ size_t FSE_count_wksp(unsigned* count, unsigned* maxSymbolValuePtr, return FSE_countFast_wksp(count, maxSymbolValuePtr, source, sourceSize, workSpace); } -size_t FSE_count(unsigned* count, unsigned* maxSymbolValuePtr, - const void* src, size_t srcSize) -{ - unsigned tmpCounters[1024]; - return FSE_count_wksp(count, maxSymbolValuePtr, src, srcSize, tmpCounters); -} - - /*-************************************************************** * FSE Compression Code @@ -765,7 +757,7 @@ size_t FSE_compress_wksp (void* dst, size_t dstSize, const void* src, size_t src if (!tableLog) tableLog = FSE_DEFAULT_TABLELOG; /* Scan input and build symbol stats */ - { CHECK_V_F(maxCount, FSE_count(count, &maxSymbolValue, src, srcSize) ); + { CHECK_V_F(maxCount, FSE_count_wksp(count, &maxSymbolValue, src, srcSize, scratchBuffer) ); if (maxCount == srcSize) return 1; /* only a single symbol in src : rle */ if (maxCount == 1) return 0; /* each symbol present maximum once => not compressible */ if (maxCount < (srcSize >> 7)) return 0; /* Heuristic : not compressible enough */ diff --git a/contrib/linux-kernel/lib/huf.h b/contrib/linux-kernel/lib/huf.h index f135226dc..f36aded00 100644 --- a/contrib/linux-kernel/lib/huf.h +++ b/contrib/linux-kernel/lib/huf.h @@ -124,7 +124,6 @@ or to save and regenerate 'CTable' using external methods. /* FSE_count() : find it within "fse.h" */ unsigned HUF_optimalTableLog(unsigned maxTableLog, size_t srcSize, unsigned maxSymbolValue); typedef struct HUF_CElt_s HUF_CElt; /* incomplete type */ -size_t HUF_buildCTable (HUF_CElt* CTable, const unsigned* count, unsigned maxSymbolValue, unsigned maxNbBits); size_t HUF_writeCTable (void* dst, size_t maxDstSize, const HUF_CElt* CTable, unsigned maxSymbolValue, unsigned huffLog); size_t HUF_compress4X_usingCTable(void* dst, size_t dstSize, const void* src, size_t srcSize, const HUF_CElt* CTable); diff --git a/contrib/linux-kernel/lib/huf_compress.c b/contrib/linux-kernel/lib/huf_compress.c index b2e71405c..a1a1d454a 100644 --- a/contrib/linux-kernel/lib/huf_compress.c +++ b/contrib/linux-kernel/lib/huf_compress.c @@ -390,15 +390,6 @@ size_t HUF_buildCTable_wksp (HUF_CElt* tree, const U32* count, U32 maxSymbolValu return maxNbBits; } -/** HUF_buildCTable() : - * Note : count is used before tree is written, so they can safely overlap - */ -size_t HUF_buildCTable (HUF_CElt* tree, const U32* count, U32 maxSymbolValue, U32 maxNbBits) -{ - huffNodeTable nodeTable; - return HUF_buildCTable_wksp(tree, count, maxSymbolValue, maxNbBits, nodeTable, sizeof(nodeTable)); -} - static size_t HUF_estimateCompressedSize(HUF_CElt* CTable, const unsigned* count, unsigned maxSymbolValue) { size_t nbBits = 0; From fd95be0adeb043fa344d0513cf7ab483adfedadf Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 16:30:11 -0700 Subject: [PATCH 132/305] Move sources to lib/zstd/ --- contrib/linux-kernel/lib/{ => zstd}/bitstream.h | 0 contrib/linux-kernel/lib/{ => zstd}/entropy_common.c | 0 contrib/linux-kernel/lib/{ => zstd}/error_private.h | 0 contrib/linux-kernel/lib/{ => zstd}/fse.h | 0 contrib/linux-kernel/lib/{ => zstd}/fse_compress.c | 0 contrib/linux-kernel/lib/{ => zstd}/fse_decompress.c | 0 contrib/linux-kernel/lib/{ => zstd}/huf.h | 0 contrib/linux-kernel/lib/{ => zstd}/huf_compress.c | 0 contrib/linux-kernel/lib/{ => zstd}/huf_decompress.c | 0 contrib/linux-kernel/lib/{ => zstd}/mem.h | 0 contrib/linux-kernel/lib/{ => zstd}/xxhash.c | 0 contrib/linux-kernel/lib/{ => zstd}/xxhash.h | 0 contrib/linux-kernel/lib/{ => zstd}/zstd_common.c | 0 contrib/linux-kernel/lib/{ => zstd}/zstd_compress.c | 0 contrib/linux-kernel/lib/{ => zstd}/zstd_decompress.c | 0 contrib/linux-kernel/lib/{ => zstd}/zstd_internal.h | 0 contrib/linux-kernel/lib/{ => zstd}/zstd_opt.h | 0 contrib/linux-kernel/test/Makefile | 10 +++++----- 18 files changed, 5 insertions(+), 5 deletions(-) rename contrib/linux-kernel/lib/{ => zstd}/bitstream.h (100%) rename contrib/linux-kernel/lib/{ => zstd}/entropy_common.c (100%) rename contrib/linux-kernel/lib/{ => zstd}/error_private.h (100%) rename contrib/linux-kernel/lib/{ => zstd}/fse.h (100%) rename contrib/linux-kernel/lib/{ => zstd}/fse_compress.c (100%) rename contrib/linux-kernel/lib/{ => zstd}/fse_decompress.c (100%) rename contrib/linux-kernel/lib/{ => zstd}/huf.h (100%) rename contrib/linux-kernel/lib/{ => zstd}/huf_compress.c (100%) rename contrib/linux-kernel/lib/{ => zstd}/huf_decompress.c (100%) rename contrib/linux-kernel/lib/{ => zstd}/mem.h (100%) rename contrib/linux-kernel/lib/{ => zstd}/xxhash.c (100%) rename contrib/linux-kernel/lib/{ => zstd}/xxhash.h (100%) rename contrib/linux-kernel/lib/{ => zstd}/zstd_common.c (100%) rename contrib/linux-kernel/lib/{ => zstd}/zstd_compress.c (100%) rename contrib/linux-kernel/lib/{ => zstd}/zstd_decompress.c (100%) rename contrib/linux-kernel/lib/{ => zstd}/zstd_internal.h (100%) rename contrib/linux-kernel/lib/{ => zstd}/zstd_opt.h (100%) diff --git a/contrib/linux-kernel/lib/bitstream.h b/contrib/linux-kernel/lib/zstd/bitstream.h similarity index 100% rename from contrib/linux-kernel/lib/bitstream.h rename to contrib/linux-kernel/lib/zstd/bitstream.h diff --git a/contrib/linux-kernel/lib/entropy_common.c b/contrib/linux-kernel/lib/zstd/entropy_common.c similarity index 100% rename from contrib/linux-kernel/lib/entropy_common.c rename to contrib/linux-kernel/lib/zstd/entropy_common.c diff --git a/contrib/linux-kernel/lib/error_private.h b/contrib/linux-kernel/lib/zstd/error_private.h similarity index 100% rename from contrib/linux-kernel/lib/error_private.h rename to contrib/linux-kernel/lib/zstd/error_private.h diff --git a/contrib/linux-kernel/lib/fse.h b/contrib/linux-kernel/lib/zstd/fse.h similarity index 100% rename from contrib/linux-kernel/lib/fse.h rename to contrib/linux-kernel/lib/zstd/fse.h diff --git a/contrib/linux-kernel/lib/fse_compress.c b/contrib/linux-kernel/lib/zstd/fse_compress.c similarity index 100% rename from contrib/linux-kernel/lib/fse_compress.c rename to contrib/linux-kernel/lib/zstd/fse_compress.c diff --git a/contrib/linux-kernel/lib/fse_decompress.c b/contrib/linux-kernel/lib/zstd/fse_decompress.c similarity index 100% rename from contrib/linux-kernel/lib/fse_decompress.c rename to contrib/linux-kernel/lib/zstd/fse_decompress.c diff --git a/contrib/linux-kernel/lib/huf.h b/contrib/linux-kernel/lib/zstd/huf.h similarity index 100% rename from contrib/linux-kernel/lib/huf.h rename to contrib/linux-kernel/lib/zstd/huf.h diff --git a/contrib/linux-kernel/lib/huf_compress.c b/contrib/linux-kernel/lib/zstd/huf_compress.c similarity index 100% rename from contrib/linux-kernel/lib/huf_compress.c rename to contrib/linux-kernel/lib/zstd/huf_compress.c diff --git a/contrib/linux-kernel/lib/huf_decompress.c b/contrib/linux-kernel/lib/zstd/huf_decompress.c similarity index 100% rename from contrib/linux-kernel/lib/huf_decompress.c rename to contrib/linux-kernel/lib/zstd/huf_decompress.c diff --git a/contrib/linux-kernel/lib/mem.h b/contrib/linux-kernel/lib/zstd/mem.h similarity index 100% rename from contrib/linux-kernel/lib/mem.h rename to contrib/linux-kernel/lib/zstd/mem.h diff --git a/contrib/linux-kernel/lib/xxhash.c b/contrib/linux-kernel/lib/zstd/xxhash.c similarity index 100% rename from contrib/linux-kernel/lib/xxhash.c rename to contrib/linux-kernel/lib/zstd/xxhash.c diff --git a/contrib/linux-kernel/lib/xxhash.h b/contrib/linux-kernel/lib/zstd/xxhash.h similarity index 100% rename from contrib/linux-kernel/lib/xxhash.h rename to contrib/linux-kernel/lib/zstd/xxhash.h diff --git a/contrib/linux-kernel/lib/zstd_common.c b/contrib/linux-kernel/lib/zstd/zstd_common.c similarity index 100% rename from contrib/linux-kernel/lib/zstd_common.c rename to contrib/linux-kernel/lib/zstd/zstd_common.c diff --git a/contrib/linux-kernel/lib/zstd_compress.c b/contrib/linux-kernel/lib/zstd/zstd_compress.c similarity index 100% rename from contrib/linux-kernel/lib/zstd_compress.c rename to contrib/linux-kernel/lib/zstd/zstd_compress.c diff --git a/contrib/linux-kernel/lib/zstd_decompress.c b/contrib/linux-kernel/lib/zstd/zstd_decompress.c similarity index 100% rename from contrib/linux-kernel/lib/zstd_decompress.c rename to contrib/linux-kernel/lib/zstd/zstd_decompress.c diff --git a/contrib/linux-kernel/lib/zstd_internal.h b/contrib/linux-kernel/lib/zstd/zstd_internal.h similarity index 100% rename from contrib/linux-kernel/lib/zstd_internal.h rename to contrib/linux-kernel/lib/zstd/zstd_internal.h diff --git a/contrib/linux-kernel/lib/zstd_opt.h b/contrib/linux-kernel/lib/zstd/zstd_opt.h similarity index 100% rename from contrib/linux-kernel/lib/zstd_opt.h rename to contrib/linux-kernel/lib/zstd/zstd_opt.h diff --git a/contrib/linux-kernel/test/Makefile b/contrib/linux-kernel/test/Makefile index b16964f84..01e877b7e 100644 --- a/contrib/linux-kernel/test/Makefile +++ b/contrib/linux-kernel/test/Makefile @@ -1,7 +1,7 @@ -IFLAGS := -isystem include/ -I ../include/ -I ../lib/ -isystem googletest/googletest/include +IFLAGS := -isystem include/ -I ../include/ -I ../lib/zstd/ -isystem googletest/googletest/include -SOURCES := $(wildcard ../lib/*.c) +SOURCES := $(wildcard ../lib/zstd/*.c) OBJECTS := $(patsubst %.c,%.o,$(SOURCES)) ARFLAGS := rcs @@ -9,10 +9,10 @@ CXXFLAGS += -std=c++11 CFLAGS += -g -O0 CPPFLAGS += $(IFLAGS) -../lib/libzstd.a: $(OBJECTS) +../lib/zstd/libzstd.a: $(OBJECTS) $(AR) $(ARFLAGS) $@ $^ -UserlandTest: UserlandTest.cpp ../lib/libzstd.a +UserlandTest: UserlandTest.cpp ../lib/zstd/libzstd.a $(CXX) $(CXXFLAGS) $(CFLAGS) $(CPPFLAGS) $^ googletest/build/googlemock/gtest/libgtest.a googletest/build/googlemock/gtest/libgtest_main.a -o $@ # Install googletest @@ -24,4 +24,4 @@ googletest: @cd googletest/build && cmake .. && $(MAKE) clean: - $(RM) -f *.{o,a} ../lib/*.{o,a} + $(RM) -f *.{o,a} ../lib/zstd/*.{o,a} From 2724f25567583173dce6b2bdb36c80b620df51a8 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 4 Apr 2017 16:31:17 -0700 Subject: [PATCH 133/305] fixed paramgrill -O# find optimal settings for a minimum speed --- tests/paramgrill.c | 53 ++++++++++++++++++++++++---------------------- 1 file changed, 28 insertions(+), 25 deletions(-) diff --git a/tests/paramgrill.c b/tests/paramgrill.c index 364c0e4b2..69f2aeb10 100644 --- a/tests/paramgrill.c +++ b/tests/paramgrill.c @@ -129,7 +129,7 @@ U32 FUZ_rand(U32* src) *********************************************************/ typedef struct { size_t cSize; - double cSpeed; + double cSpeed; /* bytes / sec */ double dSpeed; } BMK_result_t; @@ -210,7 +210,6 @@ static size_t BMK_benchParam(BMK_result_t* resultPtr, size_t cSize = 0; double fastestC = 100000000., fastestD = 100000000.; double ratio = 0.; - U64 crcCheck = 0; clock_t const benchStart = clock(); DISPLAY("\r%79s\r", ""); @@ -246,8 +245,8 @@ static size_t BMK_benchParam(BMK_result_t* resultPtr, cSize = 0; for (blockNb=0; blockNb", loopNb, name, (U32)srcSize); DISPLAY(" %9u (%4.3f),%7.1f MB/s", (U32)cSize, ratio, (double)srcSize / fastestC / 1000000.); @@ -277,18 +276,18 @@ static size_t BMK_benchParam(BMK_result_t* resultPtr, resultPtr->dSpeed = (double)srcSize / fastestD; /* CRC Checking */ - crcCheck = XXH64(resultBuffer, srcSize, 0); - if (crcOrig!=crcCheck) { - unsigned u; - unsigned eBlockSize = (unsigned)(MIN(65536*2, blockSize)); - DISPLAY("\n!!! WARNING !!! Invalid Checksum : %x != %x\n", (unsigned)crcOrig, (unsigned)crcCheck); - for (u=0; u inFileSize) benchedSize = (size_t)inFileSize; - if (benchedSize < inFileSize) - DISPLAY("Not enough memory for '%s' full size; testing %i MB only...\n", inFileName, (int)(benchedSize>>20)); + if (benchedSize < inFileSize) { + DISPLAY("Not enough memory for '%s' \n", inFileName); + fclose(inFile); + return 11; + } /* Alloc */ origBuff = malloc(benchedSize); @@ -751,10 +755,9 @@ int optimizeForSize(const char* inFileName, U32 targetSpeed) /* bench */ DISPLAY("\r%79s\r", ""); DISPLAY("optimizing for %s - limit speed %u MB/s \n", inFileName, targetSpeed); - targetSpeed *= 1000; + targetSpeed *= 1000000; { ZSTD_CCtx* const ctx = ZSTD_createCCtx(); - ZSTD_compressionParameters params; winnerInfo_t winner; BMK_result_t candidate; const size_t blockSize = g_blockSize ? g_blockSize : benchedSize; @@ -768,14 +771,14 @@ int optimizeForSize(const char* inFileName, U32 targetSpeed) { const int maxSeeds = g_noSeed ? 1 : ZSTD_maxCLevel(); int i; for (i=1; i<=maxSeeds; i++) { - params = ZSTD_getCParams(i, blockSize, 0); - BMK_benchParam(&candidate, origBuff, benchedSize, ctx, params); + ZSTD_compressionParameters const CParams = ZSTD_getCParams(i, blockSize, 0); + BMK_benchParam(&candidate, origBuff, benchedSize, ctx, CParams); if (candidate.cSpeed < targetSpeed) break; if ( (candidate.cSize < winner.result.cSize) | ((candidate.cSize == winner.result.cSize) & (candidate.cSpeed > winner.result.cSpeed)) ) { - winner.params = params; + winner.params = CParams; winner.result = candidate; BMK_printWinner(stdout, i, winner.result, winner.params, benchedSize); } } @@ -785,9 +788,9 @@ int optimizeForSize(const char* inFileName, U32 targetSpeed) /* start tests */ { time_t const grillStart = time(NULL); do { - params = winner.params; + ZSTD_compressionParameters params = winner.params; paramVariation(¶ms); - if ((FUZ_rand(&g_rand) & 15) == 3) params = randomParams(); + if ((FUZ_rand(&g_rand) & 31) == 3) params = randomParams(); /* totally random config to improve search space */ /* exclude faster if already played set of params */ if (FUZ_rand(&g_rand) & ((1 << NB_TESTS_PLAYED(params))-1)) continue; @@ -837,7 +840,7 @@ static int usage_advanced(void) DISPLAY( " -T# : set level 1 speed objective \n"); DISPLAY( " -B# : cut input into blocks of size # (default : single block) \n"); DISPLAY( " -i# : iteration loops [1-9](default : %i) \n", NBLOOPS); - DISPLAY( " -O# : find Optimized parameters for # target speed (default : 0) \n"); + DISPLAY( " -O# : find Optimized parameters for # MB/s compression speed (default : 0) \n"); DISPLAY( " -S : Single run \n"); DISPLAY( " -P# : generated sample compressibility (default : %.1f%%) \n", COMPRESSIBILITY_DEFAULT * 100); return 0; From 405d2a102758a0b76fbe5cca5c4fb2707272a093 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 16:35:31 -0700 Subject: [PATCH 134/305] Explicitly convert scratchBuffer to unsigned* --- lib/compress/fse_compress.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/compress/fse_compress.c b/lib/compress/fse_compress.c index 0b3b24286..26e8052dd 100644 --- a/lib/compress/fse_compress.c +++ b/lib/compress/fse_compress.c @@ -808,7 +808,7 @@ size_t FSE_compress_wksp (void* dst, size_t dstSize, const void* src, size_t src if (!tableLog) tableLog = FSE_DEFAULT_TABLELOG; /* Scan input and build symbol stats */ - { CHECK_V_F(maxCount, FSE_count_wksp(count, &maxSymbolValue, src, srcSize, scratchBuffer) ); + { CHECK_V_F(maxCount, FSE_count_wksp(count, &maxSymbolValue, src, srcSize, (unsigned*)scratchBuffer) ); if (maxCount == srcSize) return 1; /* only a single symbol in src : rle */ if (maxCount == 1) return 0; /* each symbol present maximum once => not compressible */ if (maxCount < (srcSize >> 7)) return 0; /* Heuristic : not compressible enough */ From 2eb623a6ebbffe19388a01b4a1f7eead49d78146 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 16:38:13 -0700 Subject: [PATCH 135/305] Explicitly convert scratchBuffer to unsigned* --- contrib/linux-kernel/lib/zstd/fse_compress.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/contrib/linux-kernel/lib/zstd/fse_compress.c b/contrib/linux-kernel/lib/zstd/fse_compress.c index e969fd2ca..b6a6d4693 100644 --- a/contrib/linux-kernel/lib/zstd/fse_compress.c +++ b/contrib/linux-kernel/lib/zstd/fse_compress.c @@ -757,7 +757,7 @@ size_t FSE_compress_wksp (void* dst, size_t dstSize, const void* src, size_t src if (!tableLog) tableLog = FSE_DEFAULT_TABLELOG; /* Scan input and build symbol stats */ - { CHECK_V_F(maxCount, FSE_count_wksp(count, &maxSymbolValue, src, srcSize, scratchBuffer) ); + { CHECK_V_F(maxCount, FSE_count_wksp(count, &maxSymbolValue, src, srcSize, (unsigned*)scratchBuffer) ); if (maxCount == srcSize) return 1; /* only a single symbol in src : rle */ if (maxCount == 1) return 0; /* each symbol present maximum once => not compressible */ if (maxCount < (srcSize >> 7)) return 0; /* Heuristic : not compressible enough */ From 89b32f3db029456855fc591b55adbdc19d6fabec Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 4 Apr 2017 16:41:11 -0700 Subject: [PATCH 136/305] fix paramgrill -O# ensure proposed config does not require more memory than necessary --- tests/paramgrill.c | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/tests/paramgrill.c b/tests/paramgrill.c index 69f2aeb10..126e54d17 100644 --- a/tests/paramgrill.c +++ b/tests/paramgrill.c @@ -169,6 +169,11 @@ static size_t BMK_benchParam(BMK_result_t* resultPtr, char name[30] = { 0 }; U64 crcOrig; + /* init result for early exit */ + resultPtr->cSize = srcSize; + resultPtr->cSpeed = 0.; + resultPtr->dSpeed = 0.; + /* Memory allocation & restrictions */ snprintf(name, 30, "Sw%02uc%02uh%02us%02ul%1ut%03uS%1u", Wlog, Clog, Hlog, Slog, Slength, Tlength, strat); if (!compressedBuffer || !resultBuffer || !blockTable) { @@ -791,6 +796,7 @@ int optimizeForSize(const char* inFileName, U32 targetSpeed) ZSTD_compressionParameters params = winner.params; paramVariation(¶ms); if ((FUZ_rand(&g_rand) & 31) == 3) params = randomParams(); /* totally random config to improve search space */ + params = ZSTD_adjustCParams(params, blockSize, 0); /* exclude faster if already played set of params */ if (FUZ_rand(&g_rand) & ((1 << NB_TESTS_PLAYED(params))-1)) continue; From 9631e603aeb7f9bf8da0f8f23d7b8854a61e78cb Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 4 Apr 2017 16:54:33 -0700 Subject: [PATCH 137/305] paramgrill : results from optimizer use cli format Best found configuration is displayed using documented cli advanced parameter syntax. --- tests/paramgrill.c | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/tests/paramgrill.c b/tests/paramgrill.c index 126e54d17..3c1f0150b 100644 --- a/tests/paramgrill.c +++ b/tests/paramgrill.c @@ -719,6 +719,12 @@ int benchFiles(const char** fileNamesTable, int nbFiles) } +static void BMK_translateAdvancedParams(ZSTD_compressionParameters params) +{ + DISPLAY("--zstd=windowLog=%u,chainLog=%u,hashLog=%u,searchLog=%u,searchLength=%u,targetLength=%u,strategy=%u \n", + params.windowLog, params.chainLog, params.hashLog, params.searchLog, params.searchLength, params.targetLength, (U32)(params.strategy)); +} + /* optimizeForSize(): * targetSpeed : expressed in MB/s */ int optimizeForSize(const char* inFileName, U32 targetSpeed) @@ -789,6 +795,7 @@ int optimizeForSize(const char* inFileName, U32 targetSpeed) } } } BMK_printWinner(stdout, 99, winner.result, winner.params, benchedSize); + BMK_translateAdvancedParams(winner.params); /* start tests */ { time_t const grillStart = time(NULL); @@ -813,6 +820,7 @@ int optimizeForSize(const char* inFileName, U32 targetSpeed) winner.params = params; winner.result = candidate; BMK_printWinner(stdout, 99, winner.result, winner.params, benchedSize); + BMK_translateAdvancedParams(winner.params); } } while (BMK_timeSpan(grillStart) < g_grillDuration_s); } From d0bbceac40a48376cc2018417bd6661edc5dc286 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 16:56:52 -0700 Subject: [PATCH 138/305] Add fs/btrfs/zstd.c --- contrib/linux-kernel/fs/btrfs/zstd.c | 414 +++++++++++++++++++++++++++ 1 file changed, 414 insertions(+) create mode 100644 contrib/linux-kernel/fs/btrfs/zstd.c diff --git a/contrib/linux-kernel/fs/btrfs/zstd.c b/contrib/linux-kernel/fs/btrfs/zstd.c new file mode 100644 index 000000000..4f856700e --- /dev/null +++ b/contrib/linux-kernel/fs/btrfs/zstd.c @@ -0,0 +1,414 @@ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include "compression.h" + +#define ZSTD_BTRFS_MAX_WINDOWLOG 17 +#define ZSTD_BTRFS_MAX_INPUT (1 << ZSTD_BTRFS_MAX_WINDOWLOG) + +static ZSTD_parameters zstd_get_btrfs_parameters(size_t src_len) +{ + ZSTD_parameters params = ZSTD_getParams(3, src_len, 0); + BUG_ON(src_len > ZSTD_BTRFS_MAX_INPUT); + BUG_ON(params.cParams.windowLog > ZSTD_BTRFS_MAX_WINDOWLOG); + params.fParams.checksumFlag = 1; + return params; +} + +struct workspace { + void *mem; + size_t size; + char *buf; + struct list_head list; +}; + +static void zstd_free_workspace(struct list_head *ws) +{ + struct workspace *workspace = list_entry(ws, struct workspace, list); + + vfree(workspace->mem); + kfree(workspace->buf); + kfree(workspace); +} + +static struct list_head *zstd_alloc_workspace(void) +{ + ZSTD_parameters params = zstd_get_btrfs_parameters(ZSTD_BTRFS_MAX_INPUT); + struct workspace *workspace; + + workspace = kzalloc(sizeof(*workspace), GFP_NOFS); + if (!workspace) return ERR_PTR(-ENOMEM); + + workspace->size = max_t(size_t, ZSTD_CStreamWorkspaceBound(params.cParams), + ZSTD_DStreamWorkspaceBound(ZSTD_BTRFS_MAX_INPUT)); + workspace->mem = vmalloc(workspace->size); + workspace->buf = kmalloc(PAGE_SIZE, GFP_NOFS); + if (!workspace->mem || !workspace->buf) goto fail; + + INIT_LIST_HEAD(&workspace->list); + + return &workspace->list; +fail: + zstd_free_workspace(&workspace->list); + return ERR_PTR(-ENOMEM); +} + +static int zstd_compress_pages(struct list_head *ws, + struct address_space *mapping, + u64 start, unsigned long len, + struct page **pages, + unsigned long nr_dest_pages, + unsigned long *out_pages, + unsigned long *total_in, + unsigned long *total_out, + unsigned long max_out) +{ + struct workspace *workspace = list_entry(ws, struct workspace, list); + ZSTD_parameters params = zstd_get_btrfs_parameters(len); + ZSTD_CStream *stream; + int ret = 0; + int nr_pages = 0; + struct page *in_page = NULL; /* The current page to read */ + struct page *out_page = NULL; /* The current page to write to */ + ZSTD_inBuffer in_buf = { NULL, 0, 0 }; + ZSTD_outBuffer out_buf = { NULL, 0, 0 }; + unsigned long tot_in = 0; + unsigned long tot_out = 0; + + *out_pages = 0; + *total_out = 0; + *total_in = 0; + + /* Initialize the stream */ + stream = ZSTD_createCStream(params, len, workspace->mem, workspace->size); + if (!stream) { + pr_warn("BTRFS: ZSTD_createStream failed\n"); + ret = -EIO; + goto out; + } + + /* map in the first page of input data */ + in_page = find_get_page(mapping, start >> PAGE_SHIFT); + in_buf.src = kmap(in_page); + in_buf.pos = 0; + in_buf.size = min_t(size_t, len, PAGE_SIZE); + + + /* Allocate and map in the output buffer */ + out_page = alloc_page(GFP_NOFS | __GFP_HIGHMEM); + if (out_page == NULL) { + ret = -ENOMEM; + goto out; + } + pages[nr_pages++] = out_page; + out_buf.dst = kmap(out_page); + out_buf.pos = 0; + out_buf.size = min_t(size_t, max_out, PAGE_SIZE); + + while (1) { + const size_t rc = ZSTD_compressStream(stream, &out_buf, &in_buf); + if (ZSTD_isError(rc)) { + pr_debug("BTRFS: ZSTD_compressStream returned %d\n", + ZSTD_getErrorCode(rc)); + ret = -EIO; + goto out; + } + + /* Check to see if we are making it bigger */ + if (tot_in + in_buf.pos > 8192 && + tot_in + in_buf.pos < + tot_out + out_buf.pos) { + ret = -E2BIG; + goto out; + } + + /* We've reached the end of our output range */ + if (out_buf.pos >= max_out) { + tot_out += out_buf.pos; + ret = -E2BIG; + goto out; + } + + /* Check if we need more output space */ + if (out_buf.pos == out_buf.size) { + tot_out += PAGE_SIZE; + max_out -= PAGE_SIZE; + kunmap(out_page); + if (nr_pages == nr_dest_pages) { + out_page = NULL; + ret = -E2BIG; + goto out; + } + out_page = alloc_page(GFP_NOFS | __GFP_HIGHMEM); + if (out_page == NULL) { + ret = -ENOMEM; + goto out; + } + pages[nr_pages++] = out_page; + out_buf.dst = kmap(out_page); + out_buf.pos = 0; + out_buf.size = min_t(size_t, max_out, PAGE_SIZE); + } + + /* We've reached the end of the input */ + if (in_buf.pos >= len) { + tot_in += in_buf.pos; + break; + } + + /* Check if we need more input */ + if (in_buf.pos == in_buf.size) { + tot_in += PAGE_SIZE; + kunmap(in_page); + put_page(in_page); + + start += PAGE_SIZE; + len -= PAGE_SIZE; + in_page = find_get_page(mapping, start >> PAGE_SHIFT); + in_buf.src = kmap(in_page); + in_buf.pos = 0; + in_buf.size = min_t(size_t, len, PAGE_SIZE); + } + } + while (1) { + const size_t rc = ZSTD_endStream(stream, &out_buf); + if (ZSTD_isError(rc)) { + pr_debug("BTRFS: ZSTD_endStream returned %d\n", + ZSTD_getErrorCode(rc)); + ret = -EIO; + goto out; + } + if (rc == 0) { + tot_out += out_buf.pos; + break; + } + if (out_buf.pos >= max_out) { + tot_out += out_buf.pos; + ret = -E2BIG; + goto out; + } + + tot_out += PAGE_SIZE; + max_out -= PAGE_SIZE; + kunmap(out_page); + if (nr_pages == nr_dest_pages) { + out_page = NULL; + ret = -E2BIG; + goto out; + } + out_page = alloc_page(GFP_NOFS | __GFP_HIGHMEM); + if (out_page == NULL) { + ret = -ENOMEM; + goto out; + } + pages[nr_pages++] = out_page; + out_buf.dst = kmap(out_page); + out_buf.pos = 0; + out_buf.size = min_t(size_t, max_out, PAGE_SIZE); + } + + if (tot_out >= tot_in) { + ret = -E2BIG; + goto out; + } + + ret = 0; + *total_in = tot_in; + *total_out = tot_out; +out: + *out_pages = nr_pages; + /* Cleanup */ + if (in_page) { + kunmap(in_page); + put_page(in_page); + } + if (out_page) { kunmap(out_page); } + return ret; +} + +static int zstd_decompress_biovec(struct list_head *ws, struct page **pages_in, + u64 disk_start, + struct bio_vec *bvec, + int vcnt, + size_t srclen) +{ + struct workspace *workspace = list_entry(ws, struct workspace, list); + ZSTD_DStream *stream; + int ret = 0; + unsigned long page_in_index = 0; + unsigned long page_out_index = 0; + unsigned long total_pages_in = DIV_ROUND_UP(srclen, PAGE_SIZE); + unsigned long buf_start; + unsigned long pg_offset; + unsigned long total_out = 0; + ZSTD_inBuffer in_buf = { NULL, 0, 0 }; + ZSTD_outBuffer out_buf = { NULL, 0, 0 }; + + stream = ZSTD_createDStream( + ZSTD_BTRFS_MAX_INPUT, workspace->mem, workspace->size); + if (!stream) { + pr_debug("BTRFS: ZSTD_createDStream failed\n"); + ret = -EIO; + goto done; + } + + in_buf.src = kmap(pages_in[page_in_index]); + in_buf.pos = 0; + in_buf.size = min_t(size_t, srclen, PAGE_SIZE); + + out_buf.dst = workspace->buf; + out_buf.pos = 0; + out_buf.size = PAGE_SIZE; + + pg_offset = 0; + + while (1) { + const size_t rc = ZSTD_decompressStream(stream, &out_buf, &in_buf); + if (zstd_is_error(rc)) { + pr_debug("BTRFS: ZSTD_decompressStream returned %d\n", + zstd_get_error_code(rc)); + ret = -EIO; + goto done; + } + buf_start = total_out; + total_out += out_buf.pos; + out_buf.pos = 0; + + { + int ret2 = btrfs_decompress_buf2page(out_buf.dst, buf_start, + total_out, disk_start, bvec, vcnt, + &page_out_index, &pg_offset); + if (ret2 == 0) { + break; + } + } + + if (in_buf.pos >= srclen) { + break; + } + + /* Check if we've hit the end of a frame */ + if (rc == 0) { + break; + } + + if (in_buf.pos == in_buf.size) { + kunmap(pages_in[page_in_index++]); + if (page_in_index >= total_pages_in) { + in_buf.src = NULL; + ret = -EIO; + goto done; + } + srclen -= PAGE_SIZE; + in_buf.src = kmap(pages_in[page_in_index]); + in_buf.pos = 0; + in_buf.size = min_t(size_t, srclen, PAGE_SIZE); + } + } + btrfs_clear_biovec_end(bvec, vcnt, page_out_index, pg_offset); + ret = 0; +done: + if (in_buf.src) { kunmap(pages_in[page_in_index]); } + return ret; +} + +static int zstd_decompress(struct list_head *ws, unsigned char *data_in, + struct page *dest_page, + unsigned long start_byte, + size_t srclen, size_t destlen) +{ + struct workspace *workspace = list_entry(ws, struct workspace, list); + ZSTD_DStream *stream; + int ret = 0; + ZSTD_inBuffer in_buf = { NULL, 0, 0 }; + ZSTD_outBuffer out_buf = { NULL, 0, 0 }; + unsigned long total_out = 0; + unsigned long pg_offset = 0; + char *kaddr; + + stream = ZSTD_createDStream( + ZSTD_BTRFS_MAX_INPUT, workspace->mem, workspace->size); + if (!stream) { + pr_warn("BTRFS: ZSTD_createDStream failed\n"); + ret = -EIO; + goto finish; + } + + destlen = min_t(size_t, destlen, PAGE_SIZE); + + in_buf.src = data_in; + in_buf.pos = 0; + in_buf.size = srclen; + + out_buf.dst = workspace->buf; + out_buf.pos = 0; + out_buf.size = PAGE_SIZE; + + ret = 1; + while (pg_offset < destlen && in_buf.pos < in_buf.size) { + unsigned long buf_start; + unsigned long buf_offset; + unsigned long bytes; + + /* Check if the frame is over and we still need more input */ + if (ret == 0) { + pr_debug("BTRFS: ZSTD_decompressStream frame ended to early\n"); + ret = -EIO; + goto finish; + } + { + const size_t rc = ZSTD_decompressStream(stream, &out_buf, &in_buf); + if (ZSTD_isError(rc)) { + pr_debug("BTRFS: ZSTD_decompressStream returned %d\n", + ZSTD_getErrorCode(rc)); + ret = -EIO; + goto finish; + } + ret = rc > 0; + } + buf_start = total_out; + total_out += out_buf.pos; + out_buf.pos = 0; + + if (total_out <= start_byte) { + continue; + } + + if (total_out > start_byte && buf_start < start_byte) { + buf_offset = start_byte - buf_start; + } else { + buf_offset = 0; + } + + bytes = min_t(unsigned long, destlen - pg_offset, + out_buf.size - buf_offset); + + kaddr = kmap_atomic(dest_page); + memcpy(kaddr + pg_offset, out_buf.dst + buf_offset, bytes); + kunmap_atomic(kaddr); + + pg_offset += bytes; + } + ret = 0; +finish: + if (pg_offset < destlen) { + kaddr = kmap_atomic(dest_page); + memset(kaddr + pg_offset, 0, destlen - pg_offset); + kunmap_atomic(kaddr); + } + return ret; +} + +const struct btrfs_compress_op btrfs_zstd_compress = { + .alloc_workspace = zstd_alloc_workspace, + .free_workspace = zstd_free_workspace, + .compress_pages = zstd_compress_pages, + .decompress_biovec = zstd_decompress_biovec, + .decompress = zstd_decompress, +}; From 585910139623485db5a96b7ae6ef6059202ae07b Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 17:00:12 -0700 Subject: [PATCH 139/305] Move Makefile to lib/zstd --- contrib/linux-kernel/lib/Makefile | 9 --------- 1 file changed, 9 deletions(-) delete mode 100644 contrib/linux-kernel/lib/Makefile diff --git a/contrib/linux-kernel/lib/Makefile b/contrib/linux-kernel/lib/Makefile deleted file mode 100644 index 53b4deec9..000000000 --- a/contrib/linux-kernel/lib/Makefile +++ /dev/null @@ -1,9 +0,0 @@ -obj-$(CONFIG_ZSTD_COMPRESS) += zstd_compress.o -obj-$(CONFIG_ZSTD_DECOMPRESS) += zstd_decompress.o - -ccflags-y += -O3 - -zstd_compress-y := entropy_common.o fse_decompress.o xxhash.o zstd_common.o \ - fse_compress.o huf_compress.o zstd_compress.o -zstd_decompress-y := entropy_common.o fse_decompress.o xxhash.o zstd_common.o \ - huf_decompress.o zstd_decompress.o From 2e4dbd2105faf5f43f65c18d920dc0e365c0c2b5 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 17:02:28 -0700 Subject: [PATCH 140/305] Really move the makefile --- contrib/linux-kernel/lib/zstd/Makefile | 9 +++++++++ 1 file changed, 9 insertions(+) create mode 100644 contrib/linux-kernel/lib/zstd/Makefile diff --git a/contrib/linux-kernel/lib/zstd/Makefile b/contrib/linux-kernel/lib/zstd/Makefile new file mode 100644 index 000000000..53b4deec9 --- /dev/null +++ b/contrib/linux-kernel/lib/zstd/Makefile @@ -0,0 +1,9 @@ +obj-$(CONFIG_ZSTD_COMPRESS) += zstd_compress.o +obj-$(CONFIG_ZSTD_DECOMPRESS) += zstd_decompress.o + +ccflags-y += -O3 + +zstd_compress-y := entropy_common.o fse_decompress.o xxhash.o zstd_common.o \ + fse_compress.o huf_compress.o zstd_compress.o +zstd_decompress-y := entropy_common.o fse_decompress.o xxhash.o zstd_common.o \ + huf_decompress.o zstd_decompress.o From c53bacc8810d93e125bf56b34acbd87650608c8a Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 17:03:48 -0700 Subject: [PATCH 141/305] Fix up some old names --- contrib/linux-kernel/fs/btrfs/zstd.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/contrib/linux-kernel/fs/btrfs/zstd.c b/contrib/linux-kernel/fs/btrfs/zstd.c index 4f856700e..23a3692ad 100644 --- a/contrib/linux-kernel/fs/btrfs/zstd.c +++ b/contrib/linux-kernel/fs/btrfs/zstd.c @@ -270,9 +270,9 @@ static int zstd_decompress_biovec(struct list_head *ws, struct page **pages_in, while (1) { const size_t rc = ZSTD_decompressStream(stream, &out_buf, &in_buf); - if (zstd_is_error(rc)) { + if (ZSTD_isError(rc)) { pr_debug("BTRFS: ZSTD_decompressStream returned %d\n", - zstd_get_error_code(rc)); + ZSTD_getErrorCode(rc)); ret = -EIO; goto done; } From 1b3856463e24652e0f753b1ee8e1ff19d7b006d6 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 17:21:49 -0700 Subject: [PATCH 142/305] Undef current if defined --- contrib/linux-kernel/lib/zstd/zstd_compress.c | 3 +++ 1 file changed, 3 insertions(+) diff --git a/contrib/linux-kernel/lib/zstd/zstd_compress.c b/contrib/linux-kernel/lib/zstd/zstd_compress.c index d4b87f280..00d18069a 100644 --- a/contrib/linux-kernel/lib/zstd/zstd_compress.c +++ b/contrib/linux-kernel/lib/zstd/zstd_compress.c @@ -19,6 +19,9 @@ #include "huf.h" #include "zstd_internal.h" /* includes zstd.h */ +#ifdef current +# undef current +#endif /*-************************************* * Constants From bb213798d9cf3e688c0c9e48efceb988714cf406 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 18:09:02 -0700 Subject: [PATCH 143/305] Rename zstd_(de)compress.c to (de)compress.c --- contrib/linux-kernel/lib/zstd/Makefile | 4 ++-- contrib/linux-kernel/lib/zstd/{zstd_compress.c => compress.c} | 0 .../linux-kernel/lib/zstd/{zstd_decompress.c => decompress.c} | 0 3 files changed, 2 insertions(+), 2 deletions(-) rename contrib/linux-kernel/lib/zstd/{zstd_compress.c => compress.c} (100%) rename contrib/linux-kernel/lib/zstd/{zstd_decompress.c => decompress.c} (100%) diff --git a/contrib/linux-kernel/lib/zstd/Makefile b/contrib/linux-kernel/lib/zstd/Makefile index 53b4deec9..067f68d19 100644 --- a/contrib/linux-kernel/lib/zstd/Makefile +++ b/contrib/linux-kernel/lib/zstd/Makefile @@ -4,6 +4,6 @@ obj-$(CONFIG_ZSTD_DECOMPRESS) += zstd_decompress.o ccflags-y += -O3 zstd_compress-y := entropy_common.o fse_decompress.o xxhash.o zstd_common.o \ - fse_compress.o huf_compress.o zstd_compress.o + fse_compress.o huf_compress.o compress.o zstd_decompress-y := entropy_common.o fse_decompress.o xxhash.o zstd_common.o \ - huf_decompress.o zstd_decompress.o + huf_decompress.o decompress.o diff --git a/contrib/linux-kernel/lib/zstd/zstd_compress.c b/contrib/linux-kernel/lib/zstd/compress.c similarity index 100% rename from contrib/linux-kernel/lib/zstd/zstd_compress.c rename to contrib/linux-kernel/lib/zstd/compress.c diff --git a/contrib/linux-kernel/lib/zstd/zstd_decompress.c b/contrib/linux-kernel/lib/zstd/decompress.c similarity index 100% rename from contrib/linux-kernel/lib/zstd/zstd_decompress.c rename to contrib/linux-kernel/lib/zstd/decompress.c From 60b34b0d1cf0ed532d26f2b77e2c2768308282dc Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 18:12:40 -0700 Subject: [PATCH 144/305] fix gitignore --- contrib/linux-kernel/.gitignore | 4 ++++ 1 file changed, 4 insertions(+) create mode 100644 contrib/linux-kernel/.gitignore diff --git a/contrib/linux-kernel/.gitignore b/contrib/linux-kernel/.gitignore new file mode 100644 index 000000000..d8dfeef21 --- /dev/null +++ b/contrib/linux-kernel/.gitignore @@ -0,0 +1,4 @@ +!lib/zstd +!lib/zstd/* +*.o +*.a From 7f510a7797c0b96fd16ae62844ceca3954ace0c1 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 18:37:55 -0700 Subject: [PATCH 145/305] Add Kconfig and Makefile diffs --- contrib/linux-kernel/lib/Kconfig.diff | 17 +++++++++++++++++ contrib/linux-kernel/lib/Makefile.diff | 13 +++++++++++++ 2 files changed, 30 insertions(+) create mode 100644 contrib/linux-kernel/lib/Kconfig.diff create mode 100644 contrib/linux-kernel/lib/Makefile.diff diff --git a/contrib/linux-kernel/lib/Kconfig.diff b/contrib/linux-kernel/lib/Kconfig.diff new file mode 100644 index 000000000..07ae5398f --- /dev/null +++ b/contrib/linux-kernel/lib/Kconfig.diff @@ -0,0 +1,17 @@ +diff --git a/lib/Kconfig b/lib/Kconfig +index 260a80e..39d9347 100644 +--- a/lib/Kconfig ++++ b/lib/Kconfig +@@ -239,6 +239,12 @@ config LZ4HC_COMPRESS + config LZ4_DECOMPRESS + tristate + ++config ZSTD_COMPRESS ++ tristate ++ ++config ZSTD_DECOMPRESS ++ tristate ++ + source "lib/xz/Kconfig" + + # diff --git a/contrib/linux-kernel/lib/Makefile.diff b/contrib/linux-kernel/lib/Makefile.diff new file mode 100644 index 000000000..be6182b39 --- /dev/null +++ b/contrib/linux-kernel/lib/Makefile.diff @@ -0,0 +1,13 @@ +diff --git a/lib/Makefile b/lib/Makefile +index 50144a3..b30a998 100644 +--- a/lib/Makefile ++++ b/lib/Makefile +@@ -106,6 +106,8 @@ obj-$(CONFIG_LZO_DECOMPRESS) += lzo/ + obj-$(CONFIG_LZ4_COMPRESS) += lz4/ + obj-$(CONFIG_LZ4HC_COMPRESS) += lz4/ + obj-$(CONFIG_LZ4_DECOMPRESS) += lz4/ ++obj-$(CONFIG_ZSTD_COMPRESS) += zstd/ ++obj-$(CONFIG_ZSTD_DECOMPRESS) += zstd/ + obj-$(CONFIG_XZ_DEC) += xz/ + obj-$(CONFIG_RAID6_PQ) += raid6/ + From 97693aff9870db58fbc22bec9ec46ef1f74b3556 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 18:39:22 -0700 Subject: [PATCH 146/305] Add a README --- contrib/linux-kernel/README.md | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) create mode 100644 contrib/linux-kernel/README.md diff --git a/contrib/linux-kernel/README.md b/contrib/linux-kernel/README.md new file mode 100644 index 000000000..11938ad62 --- /dev/null +++ b/contrib/linux-kernel/README.md @@ -0,0 +1,26 @@ +# Linux Kernel Patch + +There are two pieces, the `zstd_compress` and `zstd_decompress` kernel modules, and the BtrFS patch. +The patches are based off of the linux kernel version 4.9. +The BtrFS patch is not present in its entirety yet. + +## Zstd Kernel modules + +* The header is in `include/linux/zstd.h`. +* It is split up into `zstd_compress` and `zstd_decompress`, which can be loaded independently. +* Source files are in `lib/zstd/`. +* `lib/Kconfig` and `lib/Makefile` need to be modified by applying `lib/Kconfig.diff` and `lib/Makefile.diff` respectively. +* `test/UserlandTest.cpp` contains tests for the patch in userland by mocking the kernel headers. + It can be run with the following commands: + ``` + cd test + make googletest + make UserlandTest + ./UserlandTest + ``` + +## BtrFS + +* `fs/btrfs/zstd.c` is provided. +* Some more glue is required to integrate it with BtrFS, but I haven't included the patches yet. + In the meantime see https://github.com/terrelln/linux/commit/1914f7d4ca6c539369c84853eafa4ac104883047 if you're interested. From 9c7371b3eed89787f0ca469635f2e304e368375b Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 4 Apr 2017 18:49:07 -0700 Subject: [PATCH 147/305] Improve comments and ordering --- contrib/linux-kernel/include/linux/zstd.h | 98 +++++++++++++++-------- 1 file changed, 65 insertions(+), 33 deletions(-) diff --git a/contrib/linux-kernel/include/linux/zstd.h b/contrib/linux-kernel/include/linux/zstd.h index 101e3cb21..dcfcebff0 100644 --- a/contrib/linux-kernel/include/linux/zstd.h +++ b/contrib/linux-kernel/include/linux/zstd.h @@ -32,10 +32,6 @@ The compression ratio achievable on small data can be highly improved using compression with a dictionary in: - a single step (described as Simple dictionary API) - a single step, reusing a dictionary (described as Fast dictionary API) - - Advanced experimental functions can be accessed using #define ZSTD_STATIC_LINKING_ONLY before including zstd.h. - These APIs shall never be used with a dynamic library. - They are not "stable", their definition may change in the future. Only static linking is allowed. *********************************************************************************************************/ /*------ Version ------*/ @@ -121,28 +117,51 @@ typedef struct { ZSTD_frameParameters fParams; } ZSTD_parameters; -size_t ZSTD_CCtxWorkspaceBound(ZSTD_compressionParameters params); +/*! ZSTD_getCParams() : +* @return ZSTD_compressionParameters structure for a selected compression level and estimated srcSize. +* `estimatedSrcSize` value is optional, select 0 if not known */ +ZSTDLIB_API ZSTD_compressionParameters ZSTD_getCParams(int compressionLevel, unsigned long long estimatedSrcSize, size_t dictSize); + +/*! ZSTD_getParams() : +* same as ZSTD_getCParams(), but @return a full `ZSTD_parameters` object instead of sub-component `ZSTD_compressionParameters`. +* All fields of `ZSTD_frameParameters` are set to default (0) */ +ZSTDLIB_API ZSTD_parameters ZSTD_getParams(int compressionLevel, unsigned long long estimatedSrcSize, size_t dictSize); + + +/*! ZSTD_CCtxWorkspaceBound() : +* Returns the minimum amount of memory that needs to be passed to ZSTD_createCCtx() in order to compress with `params.cParams` +* or a `cdict` created with `params.cParams`. */ +size_t ZSTD_CCtxWorkspaceBound(ZSTD_compressionParameters cParams); /*= Compression context * When compressing many times, * it is recommended to allocate a context just once, and re-use it for each successive compression operation. -* This will make workload friendlier for system's memory. +* The context pointer is placed in `workspace`, which must outlive the returned context. * Use one context per thread for parallel execution in multi-threaded environments. */ typedef struct ZSTD_CCtx_s ZSTD_CCtx; ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx(void* workspace, size_t workspaceSize); /*! ZSTD_compressCCtx() : - Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). */ +* Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). +* Note : The workspace passed to ZSTD_createCCtx() must have been at least ZSTD_CCtxWorkspaceBound(params.cParams) bytes. */ ZSTDLIB_API size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, ZSTD_parameters params); +/*! ZSTD_compress_usingDict() : +* Compression using a predefined Dictionary (see dictBuilder/zdict.h). +* Note : The workspace passed to ZSTD_createCCtx() must have been at least ZSTD_CCtxWorkspaceBound(params.cParams) bytes. +* Note : This function loads the dictionary, resulting in significant startup delay. +* Note : When `dict == NULL || dictSize < 8` no dictionary is used. */ ZSTDLIB_API size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void *dict, size_t dictSize, ZSTD_parameters params); +/*! ZSTD_DCtxWorkspaceBound() : + * Returns the minimum amount of memory that needs to be passed to ZSTD_createDCtx(). */ size_t ZSTD_DCtxWorkspaceBound(void); /*= Decompression context * When decompressing many times, * it is recommended to allocate a context just once, and re-use it for each successive compression operation. -* This will make workload friendlier for system's memory. +* The context pointer is placed in `workspace`, which must outlive the returned context. +* `workspace` must be at least ZSTD_DCtxWorkspaceBound() bytes. * Use one context per thread for parallel execution in multi-threaded environments. */ typedef struct ZSTD_DCtx_s ZSTD_DCtx; ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx(void* workspace, size_t workspaceSize); @@ -151,12 +170,19 @@ ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx(void* workspace, size_t workspaceSize); * Same as ZSTD_decompress(), requires an allocated ZSTD_DCtx (see ZSTD_createDCtx()). */ ZSTDLIB_API size_t ZSTD_decompressDCtx(ZSTD_DCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); +/*! ZSTD_decompress_usingDict() : +* Decompression using a predefined Dictionary (see dictBuilder/zdict.h). +* Dictionary must be identical to the one used during compression. +* Note : This function loads the dictionary, resulting in significant startup delay. +* Note : When `dict == NULL || dictSize < 8` no dictionary is used. */ ZSTDLIB_API size_t ZSTD_decompress_usingDict(ZSTD_DCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void *dict, size_t dictSize); /**************************** * Fast dictionary API ****************************/ -size_t ZSTD_CDictWorkspaceBound(ZSTD_compressionParameters params); +/*! ZSTD_CDictWorkspaceBound() : + * Returns the minimum amount of memory that needs to be passed to ZSTD_createCDict() when called with the given `params.cParams`. */ +size_t ZSTD_CDictWorkspaceBound(ZSTD_compressionParameters cParams); typedef struct ZSTD_CDict_s ZSTD_CDict; @@ -164,7 +190,9 @@ typedef struct ZSTD_CDict_s ZSTD_CDict; * When compressing multiple messages / blocks with the same dictionary, it's recommended to load it just once. * ZSTD_createCDict() will create a digested dictionary, ready to start future compression operations without startup delay. * ZSTD_CDict can be created once and used by multiple threads concurrently, as its usage is read-only. -* `dictBuffer` content is referenced, and it must remain accessible throughout the lifetime of the CDict */ +* `dictBuffer` content is referenced, and it must remain accessible throughout the lifetime of the CDict. +* The cdict pointer is placed in `workspace`, which must outlive the returned cdict. +* `workspace` must be at least ZSTD_CDictWorkspaceBound(params.cParams) bytes. */ ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict(const void* dictBuffer, size_t dictSize, ZSTD_parameters params, void* workspace, size_t workspaceSize); /*! ZSTD_compress_usingCDict() : @@ -177,13 +205,17 @@ ZSTDLIB_API size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict); +/*! ZSTD_DDictWorkspaceBound() : + * Returns the minimum amount of memory that needs to be passed to ZSTD_createDDict(). */ size_t ZSTD_DDictWorkspaceBound(void); typedef struct ZSTD_DDict_s ZSTD_DDict; /*! ZSTD_createDDict() : * Create a digested dictionary, ready to start decompression operation without startup delay. -* `dictBuffer` content is referenced, and it must remain accessible throughout the lifetime of the DDict */ +* `dictBuffer` content is referenced, and it must remain accessible throughout the lifetime of the DDict. +* The ddict pointer is placed in `workspace`, which must outlive the returned ddict. +* `workspace` must be at least ZSTD_DDictWorkspaceBound() bytes. */ ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict(const void* dictBuffer, size_t dictSize, void* workspace, size_t workspaceSize); /*! ZSTD_decompress_usingDDict() : @@ -219,13 +251,12 @@ typedef struct ZSTD_outBuffer_s { * A ZSTD_CStream object is required to track streaming operation. * Use ZSTD_createCStream() and ZSTD_freeCStream() to create/release resources. * ZSTD_CStream objects can be reused multiple times on consecutive compression operations. -* It is recommended to re-use ZSTD_CStream in situations where many streaming operations will be achieved consecutively, -* since it will play nicer with system's memory, by re-using already allocated memory. +* It is recommended to re-use ZSTD_CStream in situations where many streaming operations will be achieved consecutively. * Use one separate ZSTD_CStream per thread for parallel execution. * * Start a new compression by initializing ZSTD_CStream. * Use ZSTD_initCStream() to start a new compression operation. -* Use ZSTD_initCStream_usingDict() or ZSTD_initCStream_usingCDict() for a compression which requires a dictionary (experimental section) +* Use ZSTD_initCStream_usingCDict() for a compression which requires a dictionary. * * Use ZSTD_compressStream() repetitively to consume input stream. * The function will automatically update both `pos` fields. @@ -252,11 +283,22 @@ typedef struct ZSTD_outBuffer_s { * * *******************************************************************/ -size_t ZSTD_CStreamWorkspaceBound(ZSTD_compressionParameters params); +/*! ZSTD_CStreamWorkspaceBound() : + * Returns the minimum amount of memory that needs to be passed to ZSTD_createCStream() or ZSTD_createCStream_usingCDict() + * when called with the given `params.cParams` or `cdict` created with `params.cParams`. */ +size_t ZSTD_CStreamWorkspaceBound(ZSTD_compressionParameters cParams); typedef struct ZSTD_CStream_s ZSTD_CStream; /*===== ZSTD_CStream management functions =====*/ +/*! ZSTD_createCStream() : +* Creates a cstream using params. +* Callers may optionally provide the size of the source they intend to compress, or pass 0 if unknown. +* The stream is placed in `workspace`, which must outlive the returned stream. +* `workspace` must be at least ZSTD_CStreamWorkspaceBound(params.cParams) bytes. */ ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream(ZSTD_parameters params, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize); +/*! ZSTD_createCStream_usingCDict() : +* Similar to ZSTD_createCStream(), but use the given preprocessed dictionary. +*/ ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_usingCDict(const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize); /*===== Streaming compression functions =====*/ @@ -292,11 +334,19 @@ ZSTDLIB_API size_t ZSTD_CStreamOutSize(void); /**< recommended size for output * The return value is a suggested next input size (a hint to improve latency) that will never load more than the current frame. * *******************************************************************************/ +/*! ZSTD_DStreamWorkspaceBound() : + * Returns the minimum amount of memory that needs to be passed to ZSTD_createDStream() to decompress frames with windowSize <= maxWindowSize. */ size_t ZSTD_DStreamWorkspaceBound(size_t maxWindowSize); typedef struct ZSTD_DStream_s ZSTD_DStream; /*===== ZSTD_DStream management functions =====*/ +/*! ZSTD_createDStream() : +* Creates a dstream that can decompress frames with windowSize up to maxWindowSize. +* The stream is placed in `workspace`, which must outlive the returned stream. +* `workspace` must be at least ZSTD_DStreamWorkspaceBound(maxWindowSize) bytes. */ ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream(size_t maxWindowSize, void* workspace, size_t workspaceSize); +/*! ZSTD_createDStream_usingDDict() : +* Similar to ZSTD_createCStream(), but use the given preprocessed dictionary. */ ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream_usingDDict(size_t maxWindowSize, const ZSTD_DDict* ddict, void* workspace, size_t workspaceSize); /*===== Streaming decompression functions =====*/ @@ -307,14 +357,6 @@ ZSTDLIB_API size_t ZSTD_DStreamInSize(void); /*!< recommended size for input ZSTDLIB_API size_t ZSTD_DStreamOutSize(void); /*!< recommended size for output buffer. Guarantee to successfully flush at least one complete block in all circumstances. */ -/**************************************************************************************** - * START OF ADVANCED AND EXPERIMENTAL FUNCTIONS - * The definitions in this section are considered experimental. - * They should never be used with a dynamic library, as they may change in the future. - * They are provided for advanced usages. - * Use them only in association with static linking. - * ***************************************************************************************/ - /* --- Constants ---*/ #define ZSTD_MAGICNUMBER 0xFD2FB528 /* >= v0.8.0 */ #define ZSTD_MAGIC_SKIPPABLE_START 0x184D2A50U @@ -397,16 +439,6 @@ ZSTDLIB_API unsigned long long ZSTD_findDecompressedSize(const void* src, size_t /*************************************** * Advanced compression functions ***************************************/ -/*! ZSTD_getCParams() : -* @return ZSTD_compressionParameters structure for a selected compression level and estimated srcSize. -* `estimatedSrcSize` value is optional, select 0 if not known */ -ZSTDLIB_API ZSTD_compressionParameters ZSTD_getCParams(int compressionLevel, unsigned long long estimatedSrcSize, size_t dictSize); - -/*! ZSTD_getParams() : -* same as ZSTD_getCParams(), but @return a full `ZSTD_parameters` object instead of sub-component `ZSTD_compressionParameters`. -* All fields of `ZSTD_frameParameters` are set to default (0) */ -ZSTDLIB_API ZSTD_parameters ZSTD_getParams(int compressionLevel, unsigned long long estimatedSrcSize, size_t dictSize); - /*! ZSTD_checkCParams() : * Ensure param values remain within authorized range */ ZSTDLIB_API size_t ZSTD_checkCParams(ZSTD_compressionParameters params); From 02d37aa1c10d71e8b96dbe8903448f15ee7c6061 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 5 Apr 2017 14:53:51 -0700 Subject: [PATCH 148/305] ensure correct size of internal buffers in case of error --- lib/compress/zstd_compress.c | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index b3e0b4b3a..6032e16bc 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -3037,18 +3037,21 @@ size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, /* allocate buffers */ { size_t const neededInBuffSize = (size_t)1 << params.cParams.windowLog; if (zcs->inBuffSize < neededInBuffSize) { - zcs->inBuffSize = neededInBuffSize; + zcs->inBuffSize = 0; ZSTD_free(zcs->inBuff, zcs->customMem); zcs->inBuff = (char*) ZSTD_malloc(neededInBuffSize, zcs->customMem); if (zcs->inBuff == NULL) return ERROR(memory_allocation); + zcs->inBuffSize = neededInBuffSize; } zcs->blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, neededInBuffSize); } if (zcs->outBuffSize < ZSTD_compressBound(zcs->blockSize)+1) { - zcs->outBuffSize = ZSTD_compressBound(zcs->blockSize)+1; + size_t const outBuffSize = ZSTD_compressBound(zcs->blockSize)+1; + zcs->outBuffSize = 0; ZSTD_free(zcs->outBuff, zcs->customMem); - zcs->outBuff = (char*) ZSTD_malloc(zcs->outBuffSize, zcs->customMem); + zcs->outBuff = (char*) ZSTD_malloc(outBuffSize, zcs->customMem); if (zcs->outBuff == NULL) return ERROR(memory_allocation); + zcs->outBuffSize = outBuffSize; } if (dict && dictSize >= 8) { From 003a24432432e137aac68e8d73ea5f56d4da9fb6 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 5 Apr 2017 15:28:56 -0700 Subject: [PATCH 149/305] DStream : ensure correct size of internal buffers in case of error --- lib/decompress/zstd_decompress.c | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c index 06337dbd5..7499b8ae1 100644 --- a/lib/decompress/zstd_decompress.c +++ b/lib/decompress/zstd_decompress.c @@ -2380,15 +2380,17 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB zds->blockSize = blockSize; if (zds->inBuffSize < blockSize) { ZSTD_free(zds->inBuff, zds->customMem); - zds->inBuffSize = blockSize; + zds->inBuffSize = 0; zds->inBuff = (char*)ZSTD_malloc(blockSize, zds->customMem); if (zds->inBuff == NULL) return ERROR(memory_allocation); + zds->inBuffSize = blockSize; } if (zds->outBuffSize < neededOutSize) { ZSTD_free(zds->outBuff, zds->customMem); - zds->outBuffSize = neededOutSize; + zds->outBuffSize = 0; zds->outBuff = (char*)ZSTD_malloc(neededOutSize, zds->customMem); if (zds->outBuff == NULL) return ERROR(memory_allocation); + zds->outBuffSize = neededOutSize; } } zds->stage = zdss_read; /* pass-through */ From ce80098f14ee5ea502f3c7ea9a7fcbbeb6111662 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 5 Apr 2017 16:34:09 -0700 Subject: [PATCH 150/305] improved zstreamtest --mt to trap bug #644 --- tests/zstreamtest.c | 16 ++++++++++------ 1 file changed, 10 insertions(+), 6 deletions(-) diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index e1e68b01c..270c0221b 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -667,7 +667,8 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, double compres if ((FUZ_rand(&lseed)&1) /* at beginning, to keep same nb of rand */ && oldTestLog /* at least one test happened */ && resetAllowed) { maxTestSize = FUZ_randomLength(&lseed, oldTestLog+2); - if (maxTestSize >= srcBufferSize) maxTestSize = srcBufferSize-1; + if (maxTestSize >= srcBufferSize) + maxTestSize = srcBufferSize-1; { U64 const pledgedSrcSize = (FUZ_rand(&lseed) & 3) ? 0 : maxTestSize; size_t const resetError = ZSTD_resetCStream(zc, pledgedSrcSize); CHECK(ZSTD_isError(resetError), "ZSTD_resetCStream error : %s", ZSTD_getErrorName(resetError)); @@ -675,14 +676,14 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, double compres } else { U32 const testLog = FUZ_rand(&lseed) % maxSrcLog; U32 const dictLog = FUZ_rand(&lseed) % maxSrcLog; - U32 const cLevel = (FUZ_rand(&lseed) % + U32 const cLevel = ( FUZ_rand(&lseed) % (ZSTD_maxCLevel() - - (MAX(testLog, dictLog) / cLevelLimiter))) + - 1; + (MAX(testLog, dictLog) / cLevelLimiter))) + + 1; maxTestSize = FUZ_rLogLength(&lseed, testLog); oldTestLog = testLog; /* random dictionary selection */ - dictSize = ((FUZ_rand(&lseed)&63)==1) ? FUZ_rLogLength(&lseed, dictLog) : 0; + dictSize = ((FUZ_rand(&lseed)&1)==1) ? FUZ_rLogLength(&lseed, dictLog) : 0; { size_t const dictStart = FUZ_rand(&lseed) % (srcBufferSize - dictSize); dict = srcBuffer + dictStart; } @@ -935,9 +936,11 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp } { U64 const pledgedSrcSize = (FUZ_rand(&lseed) & 3) ? 0 : maxTestSize; ZSTD_parameters params = ZSTD_getParams(cLevel, pledgedSrcSize, dictSize); - DISPLAYLEVEL(5, "Init with windowLog = %u \n", params.cParams.windowLog); + DISPLAYLEVEL(5, "Init with windowLog = %u and pledgedSrcSize = %u \n", + params.cParams.windowLog, (U32)pledgedSrcSize); params.fParams.checksumFlag = FUZ_rand(&lseed) & 1; params.fParams.noDictIDFlag = FUZ_rand(&lseed) & 1; + params.fParams.contentSizeFlag = pledgedSrcSize>0; { size_t const initError = ZSTDMT_initCStream_advanced(zc, dict, dictSize, params, pledgedSrcSize); CHECK (ZSTD_isError(initError),"ZSTDMT_initCStream_advanced error : %s", ZSTD_getErrorName(initError)); } ZSTDMT_setMTCtxParameter(zc, ZSTDMT_p_overlapSectionLog, FUZ_rand(&lseed) % 12); @@ -1009,6 +1012,7 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp size_t const dstBuffSize = MIN(dstBufferSize - totalGenSize, randomDstSize); inBuff.size = inBuff.pos + readCSrcSize; outBuff.size = inBuff.pos + dstBuffSize; + DISPLAYLEVEL(5, "ZSTD_decompressStream input %u bytes \n", (U32)readCSrcSize); decompressionResult = ZSTD_decompressStream(zd, &outBuff, &inBuff); CHECK (ZSTD_isError(decompressionResult), "decompression error : %s", ZSTD_getErrorName(decompressionResult)); } From 36c2a03757feb1dc44a19bf03f2b78e66325313c Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 5 Apr 2017 22:06:21 -0700 Subject: [PATCH 151/305] updated comments for ZSTD_resetCStream() --- lib/zstd.h | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/lib/zstd.h b/lib/zstd.h index 153e244be..6066db45e 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -588,13 +588,21 @@ ZSTDLIB_API unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize); /*===== Advanced Streaming compression functions =====*/ ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem); +ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs); /**< size of CStream is variable, depending primarily on compression level */ ZSTDLIB_API size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize); /**< pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ ZSTDLIB_API size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */ ZSTDLIB_API size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */ ZSTDLIB_API size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict); /**< note : cdict will just be referenced, and must outlive compression session */ -ZSTDLIB_API size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize); /**< re-use compression parameters from previous init; skip dictionary loading stage; zcs must be init at least once before. note: pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ -ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs); + +/*! ZSTD_resetCStream() : + * start a new compression job, using same parameters from previous job. + * This is typically useful to skip dictionary loading stage, since it will re-use it in-place.. + * Note that zcs must be init at least once before using ZSTD_resetCStream(). + * pledgedSrcSize==0 means "srcSize unknown". + * If pledgedSrcSize > 0, its value must be correct, as it will be written in header, and controlled at the end. + * @return : 0, or an error code (which can be tested using ZSTD_isError()) */ +ZSTDLIB_API size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize); /*===== Advanced Streaming decompression functions =====*/ From 0181fef545229e54b77e78dcf1d18078010940b3 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 6 Apr 2017 01:25:26 -0700 Subject: [PATCH 152/305] ensure cctx internal buffer is correctly sized in case of memory error --- lib/compress/zstd_compress.c | 1 + 1 file changed, 1 insertion(+) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 6032e16bc..4ad978a4b 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -278,6 +278,7 @@ static size_t ZSTD_resetCCtx_advanced (ZSTD_CCtx* zc, size_t const neededSpace = tableSpace + (256*sizeof(U32)) /* huffTable */ + tokenSpace + (((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) ? optSpace : 0); if (zc->workSpaceSize < neededSpace) { + zc->workSpaceSize = 0; ZSTD_free(zc->workSpace, zc->customMem); zc->workSpace = ZSTD_malloc(neededSpace, zc->customMem); if (zc->workSpace == NULL) return ERROR(memory_allocation); From e128ac096f0d3a9eb683d9f1158a298adeae8f13 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 6 Apr 2017 10:15:32 -0700 Subject: [PATCH 153/305] Don't build zstd release binary with debug symbols --- appveyor.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/appveyor.yml b/appveyor.yml index 81f76926e..5f0b82e80 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -88,6 +88,8 @@ ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] COPY programs\zstd.exe zstd_%PLATFORM%.exe && appveyor PushArtifact zstd_%PLATFORM%.exe ) && ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] lib\dll\example\build_package.bat && + make -C programs DEBUGFLAGS= clean zstd && + cp programs\zstd.exe bin\zstd.exe && make -C programs DEBUGFLAGS= clean zstdmt && cp programs\zstd.exe bin\zstdmt.exe && cd bin\ && 7z a -tzip zstd-win-release-%PLATFORM%.zip * && From 17c12cbca62100389240006689413a1b011365c1 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 6 Apr 2017 10:34:09 -0700 Subject: [PATCH 154/305] Push both artifacts without debug symbols --- appveyor.yml | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/appveyor.yml b/appveyor.yml index 5f0b82e80..27994853a 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -85,10 +85,11 @@ SET "CPPFLAGS=-I../../zlib" && SET "LDFLAGS=../../zlib/libz.a" && sh -c "%SCRIPT%" && - ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] COPY programs\zstd.exe zstd_%PLATFORM%.exe && appveyor PushArtifact zstd_%PLATFORM%.exe ) && ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] lib\dll\example\build_package.bat && make -C programs DEBUGFLAGS= clean zstd && + cp programs\zstd.exe zstd_%PLATFORM%.exe && + appveyor PushArtifact zstd_%PLATFORM%.exe && cp programs\zstd.exe bin\zstd.exe && make -C programs DEBUGFLAGS= clean zstdmt && cp programs\zstd.exe bin\zstdmt.exe && From 279be2015b4af98200b4d6b3e32c1b0155f51cd2 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 6 Apr 2017 12:56:40 -0700 Subject: [PATCH 155/305] Let zstd overwrite read-only files --- programs/fileio.c | 47 +++++++++++++++++++++++++++++++++++------------ 1 file changed, 35 insertions(+), 12 deletions(-) diff --git a/programs/fileio.c b/programs/fileio.c index 78fb9a268..b384c3c2b 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -31,6 +31,11 @@ #include /* clock */ #include /* errno */ +#if defined (_MSC_VER) +# include +# include +#endif + #include "mem.h" #include "fileio.h" #define ZSTD_STATIC_LINKING_ONLY /* ZSTD_magicNumber, ZSTD_frameHeaderSize_max */ @@ -188,6 +193,18 @@ void FIO_setOverlapLog(unsigned overlapLog){ /*-************************************* * Functions ***************************************/ +/** FIO_remove() : + * @result : Unlink `fileName`, even if it's read-only */ +static int FIO_remove(const char* path) +{ +#if defined(_WIN32) || defined(WIN32) + /* windows doesn't allow remove read-only files, so try to make it + * writable first */ + chmod(path, _S_IWRITE); +#endif + return remove(path); +} + /** FIO_openSrcFile() : * condition : `dstFileName` must be non-NULL. * @result : FILE* to `dstFileName`, or NULL if it fails */ @@ -230,23 +247,29 @@ static FILE* FIO_openDstFile(const char* dstFileName) if (g_sparseFileSupport == 1) { g_sparseFileSupport = ZSTD_SPARSE_DEFAULT; } - if (!g_overwrite && strcmp (dstFileName, nulmark)) { /* Check if destination file already exists */ + if (strcmp (dstFileName, nulmark)) { /* Check if destination file already exists */ f = fopen( dstFileName, "rb" ); if (f != 0) { /* dest file exists, prompt for overwrite authorization */ fclose(f); - if (g_displayLevel <= 1) { - /* No interaction possible */ - DISPLAY("zstd: %s already exists; not overwritten \n", dstFileName); - return NULL; - } - DISPLAY("zstd: %s already exists; do you wish to overwrite (y/N) ? ", dstFileName); - { int ch = getchar(); - if ((ch!='Y') && (ch!='y')) { - DISPLAY(" not overwritten \n"); + if (!g_overwrite) { + if (g_displayLevel <= 1) { + /* No interaction possible */ + DISPLAY("zstd: %s already exists; not overwritten \n", dstFileName); return NULL; } - while ((ch!=EOF) && (ch!='\n')) ch = getchar(); /* flush rest of input line */ - } } } + DISPLAY("zstd: %s already exists; do you wish to overwrite (y/N) ? ", dstFileName); + { int ch = getchar(); + if ((ch!='Y') && (ch!='y')) { + DISPLAY(" not overwritten \n"); + return NULL; + } + while ((ch!=EOF) && (ch!='\n')) ch = getchar(); /* flush rest of input line */ + } + } + + /* need to unlink */ + FIO_remove(dstFileName); + } } f = fopen( dstFileName, "wb" ); if (f==NULL) DISPLAYLEVEL(1, "zstd: %s: %s\n", dstFileName, strerror(errno)); } From 9da11c6aaecff37e2b167f1e5d1e5c1463689f9a Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 6 Apr 2017 12:58:49 -0700 Subject: [PATCH 156/305] Add test for overwriting read-only --- tests/playTests.sh | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/tests/playTests.sh b/tests/playTests.sh index 5abbb14e3..5409d9aa1 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -109,6 +109,13 @@ $ZSTD -q tmp && die "overwrite check failed!" $ECHO "test : force overwrite" $ZSTD -q -f tmp $ZSTD -q --force tmp +$ECHO "test : overwrite readonly file" +$ECHO foo > tmpro.zst +$ECHO foo > tmpro +chmod 400 tmpro.zst +$ZSTD -q --force tmpro || die "failed to overwrite read-only file" +chmod 600 tmpro.zst +rm tmpro tmpro.zst $ECHO "test : file removal" $ZSTD -f --rm tmp ls tmp && die "tmp should no longer be present" From 1652172b2d11e87ef26ad04a0406428cd13be3c1 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 6 Apr 2017 17:06:30 -0700 Subject: [PATCH 157/305] Add refuse to overwrite test and fix format --- tests/playTests.sh | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/tests/playTests.sh b/tests/playTests.sh index 5409d9aa1..266bbd91b 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -110,12 +110,13 @@ $ECHO "test : force overwrite" $ZSTD -q -f tmp $ZSTD -q --force tmp $ECHO "test : overwrite readonly file" +rm -f tmpro tmpro.zst $ECHO foo > tmpro.zst $ECHO foo > tmpro chmod 400 tmpro.zst -$ZSTD -q --force tmpro || die "failed to overwrite read-only file" -chmod 600 tmpro.zst -rm tmpro tmpro.zst +$ZSTD -q tmpro && die "should have refused to overwrite read-only file" +$ZSTD -q -f tmpro +rm -f tmpro tmpro.zst $ECHO "test : file removal" $ZSTD -f --rm tmp ls tmp && die "tmp should no longer be present" From bed6c0a4b022f216933fd63db825a00f013a839a Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Mon, 10 Apr 2017 14:49:34 -0700 Subject: [PATCH 158/305] Remove ZSTDLIB_API from decompress.c --- contrib/linux-kernel/lib/zstd/decompress.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/contrib/linux-kernel/lib/zstd/decompress.c b/contrib/linux-kernel/lib/zstd/decompress.c index 9eb210af0..3fcef0105 100644 --- a/contrib/linux-kernel/lib/zstd/decompress.c +++ b/contrib/linux-kernel/lib/zstd/decompress.c @@ -1399,7 +1399,7 @@ size_t ZSTD_decompressBlock(ZSTD_DCtx* dctx, /** ZSTD_insertBlock() : insert `src` block into `dctx` history. Useful to track uncompressed blocks. */ -ZSTDLIB_API size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize) +size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize) { ZSTD_checkContinuity(dctx, blockStart); dctx->previousDstEnd = (const char*)blockStart + blockSize; From adb1974aec3428e953732c9794922f1746602559 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Mon, 10 Apr 2017 14:50:03 -0700 Subject: [PATCH 159/305] Switch comments to kernel style + limit to 80 cols --- contrib/linux-kernel/include/linux/zstd.h | 1509 ++++++++++++++------- 1 file changed, 1006 insertions(+), 503 deletions(-) diff --git a/contrib/linux-kernel/include/linux/zstd.h b/contrib/linux-kernel/include/linux/zstd.h index dcfcebff0..38b2bca69 100644 --- a/contrib/linux-kernel/include/linux/zstd.h +++ b/contrib/linux-kernel/include/linux/zstd.h @@ -7,354 +7,749 @@ * of patent rights can be found in the PATENTS file in the same directory. */ -#ifndef ZSTD_H_235446 -#define ZSTD_H_235446 +#ifndef ZSTD_H +#define ZSTD_H /* ====== Dependency ======*/ #include /* size_t */ -/* ===== ZSTDLIB_API : control library symbols visibility ===== */ -#define ZSTDLIB_API - - -/******************************************************************************************************* - Introduction - - zstd, short for Zstandard, is a fast lossless compression algorithm, targeting real-time compression scenarios - at zlib-level and better compression ratios. The zstd compression library provides in-memory compression and - decompression functions. The library supports compression levels from 1 up to ZSTD_maxCLevel() which is 22. - Levels >= 20, labeled `--ultra`, should be used with caution, as they require more memory. - Compression can be done in: - - a single step (described as Simple API) - - a single step, reusing a context (described as Explicit memory management) - - unbounded multiple steps (described as Streaming compression) - The compression ratio achievable on small data can be highly improved using compression with a dictionary in: - - a single step (described as Simple dictionary API) - - a single step, reusing a dictionary (described as Fast dictionary API) -*********************************************************************************************************/ - -/*------ Version ------*/ -#define ZSTD_VERSION_MAJOR 1 -#define ZSTD_VERSION_MINOR 1 -#define ZSTD_VERSION_RELEASE 5 - -#define ZSTD_LIB_VERSION ZSTD_VERSION_MAJOR.ZSTD_VERSION_MINOR.ZSTD_VERSION_RELEASE -#define ZSTD_QUOTE(str) #str -#define ZSTD_EXPAND_AND_QUOTE(str) ZSTD_QUOTE(str) -#define ZSTD_VERSION_STRING ZSTD_EXPAND_AND_QUOTE(ZSTD_LIB_VERSION) - -#define ZSTD_VERSION_NUMBER (ZSTD_VERSION_MAJOR *100*100 + ZSTD_VERSION_MINOR *100 + ZSTD_VERSION_RELEASE) - +/*-***************************************************************************** + * Introduction + * + * zstd, short for Zstandard, is a fast lossless compression algorithm, + * targeting real-time compression scenarios at zlib-level and better + * compression ratios. The zstd compression library provides in-memory + * compression and decompression functions. The library supports compression + * levels from 1 up to ZSTD_maxCLevel() which is 22. Levels >= 20, labeled + * ultra, should be used with caution, as they require more memory. + * Compression can be done in: + * - a single step, reusing a context (described as Explicit memory management) + * - unbounded multiple steps (described as Streaming compression) + * The compression ratio achievable on small data can be highly improved using + * compression with a dictionary in: + * - a single step (described as Simple dictionary API) + * - a single step, reusing a dictionary (described as Fast dictionary API) + ******************************************************************************/ /*====== Helper functions ======*/ + +/** + * enum ZSTD_ErrorCode - zstd error codes + * + * Functions that return size_t can be checked for errors using ZSTD_isError() + * and the ZSTD_ErrorCode can be extracted using ZSTD_getErrorCode(). + */ typedef enum { - ZSTD_error_no_error, - ZSTD_error_GENERIC, - ZSTD_error_prefix_unknown, - ZSTD_error_version_unsupported, - ZSTD_error_parameter_unknown, - ZSTD_error_frameParameter_unsupported, - ZSTD_error_frameParameter_unsupportedBy32bits, - ZSTD_error_frameParameter_windowTooLarge, - ZSTD_error_compressionParameter_unsupported, - ZSTD_error_init_missing, - ZSTD_error_memory_allocation, - ZSTD_error_stage_wrong, - ZSTD_error_dstSize_tooSmall, - ZSTD_error_srcSize_wrong, - ZSTD_error_corruption_detected, - ZSTD_error_checksum_wrong, - ZSTD_error_tableLog_tooLarge, - ZSTD_error_maxSymbolValue_tooLarge, - ZSTD_error_maxSymbolValue_tooSmall, - ZSTD_error_dictionary_corrupted, - ZSTD_error_dictionary_wrong, - ZSTD_error_dictionaryCreation_failed, - ZSTD_error_maxCode + ZSTD_error_no_error, + ZSTD_error_GENERIC, + ZSTD_error_prefix_unknown, + ZSTD_error_version_unsupported, + ZSTD_error_parameter_unknown, + ZSTD_error_frameParameter_unsupported, + ZSTD_error_frameParameter_unsupportedBy32bits, + ZSTD_error_frameParameter_windowTooLarge, + ZSTD_error_compressionParameter_unsupported, + ZSTD_error_init_missing, + ZSTD_error_memory_allocation, + ZSTD_error_stage_wrong, + ZSTD_error_dstSize_tooSmall, + ZSTD_error_srcSize_wrong, + ZSTD_error_corruption_detected, + ZSTD_error_checksum_wrong, + ZSTD_error_tableLog_tooLarge, + ZSTD_error_maxSymbolValue_tooLarge, + ZSTD_error_maxSymbolValue_tooSmall, + ZSTD_error_dictionary_corrupted, + ZSTD_error_dictionary_wrong, + ZSTD_error_dictionaryCreation_failed, + ZSTD_error_maxCode } ZSTD_ErrorCode; -ZSTDLIB_API int ZSTD_maxCLevel(void); /*!< maximum compression level available */ -ZSTDLIB_API size_t ZSTD_compressBound(size_t srcSize); /*!< maximum compressed size in worst case scenario */ -/*! ZSTD_isError() : -* tells if a `size_t` function result is an error code */ -ZSTDLIB_API static __attribute__((unused)) unsigned ZSTD_isError(size_t code) { +/** + * ZSTD_maxCLevel() - maximum compression level available + * + * Return: Maximum compression level available. + */ +int ZSTD_maxCLevel(void); +/** + * ZSTD_compressBound() - maximum compressed size in worst case scenario + * @srcSize: The size of the data to compress. + * + * Return: The maximum compressed size in the worst case scenario. + */ +size_t ZSTD_compressBound(size_t srcSize); +/** + * ZSTD_isError() - tells if a size_t function result is an error code + * @code: The function result to check for error. + * + * Return: Non-zero iff the code is an error. + */ +static __attribute__((unused)) unsigned int ZSTD_isError(size_t code) +{ return code > (size_t)-ZSTD_error_maxCode; } -/*! ZSTD_getErrorCode() : -* convert a `size_t` function result into a proper ZSTD_errorCode enum */ -ZSTDLIB_API static __attribute__((unused)) ZSTD_ErrorCode ZSTD_getErrorCode(size_t functionResult) { - if (!ZSTD_isError(functionResult)) { +/** + * ZSTD_getErrorCode() - translates an error function result to a ZSTD_ErrorCode + * @functionResult: The result of a function for which ZSTD_isError() is true. + * + * Return: The ZSTD_ErrorCode corresponding to the functionResult or 0 + * if the functionResult isn't an error. + */ +static __attribute__((unused)) ZSTD_ErrorCode ZSTD_getErrorCode( + size_t functionResult) +{ + if (!ZSTD_isError(functionResult)) return (ZSTD_ErrorCode)0; - } return (ZSTD_ErrorCode)(0 - functionResult); } -/*************************************** -* Explicit memory management -***************************************/ - -typedef enum { ZSTD_fast, ZSTD_dfast, ZSTD_greedy, ZSTD_lazy, ZSTD_lazy2, ZSTD_btlazy2, ZSTD_btopt, ZSTD_btopt2 } ZSTD_strategy; /* from faster to stronger */ +/** + * enum ZSTD_strategy - zstd compression search strategy + * + * From faster to stronger. + */ +typedef enum { + ZSTD_fast, + ZSTD_dfast, + ZSTD_greedy, + ZSTD_lazy, + ZSTD_lazy2, + ZSTD_btlazy2, + ZSTD_btopt, + ZSTD_btopt2 +} ZSTD_strategy; +/** + * struct ZSTD_compressionParameters - zstd compression parameters + * @windowLog: Log of the largest match distance. Larger means more + * compression, and more memory needed during decompression. + * @chainLog: Fully searched segment. Larger means more compression, slower, + * and more memory (useless for fast). + * @hashLog: Dispatch table. Larger means more compression, + * slower, and more memory. + * @searchLog: Number of searches. Larger means more compression and slower. + * @searchLength: Match length searched. Larger means faster decompression, + * sometimes less compression. + * @targetLength: Acceptable match size for optimal parser (only). Larger means + * more compression, and slower. + * @strategy: The zstd compression strategy. + */ typedef struct { - unsigned windowLog; /**< largest match distance : larger == more compression, more memory needed during decompression */ - unsigned chainLog; /**< fully searched segment : larger == more compression, slower, more memory (useless for fast) */ - unsigned hashLog; /**< dispatch table : larger == faster, more memory */ - unsigned searchLog; /**< nb of searches : larger == more compression, slower */ - unsigned searchLength; /**< match length searched : larger == faster decompression, sometimes less compression */ - unsigned targetLength; /**< acceptable match size for optimal parser (only) : larger == more compression, slower */ + unsigned int windowLog; + unsigned int chainLog; + unsigned int hashLog; + unsigned int searchLog; + unsigned int searchLength; + unsigned int targetLength; ZSTD_strategy strategy; } ZSTD_compressionParameters; +/** + * struct ZSTD_frameParameters - zstd frame parameters + * @contentSizeFlag: Controls whether content size will be present in the frame + * header (when known). + * @checksumFlag: Controls whether a 32-bit checksum is generated at the end + * of the frame for error detection. + * @noDictIDFlag: Controls whether dictID will be saved into the frame header + * when using dictionary compression. + * + * The default value is all fields set to 0. + */ typedef struct { - unsigned contentSizeFlag; /**< 1: content size will be in frame header (when known) */ - unsigned checksumFlag; /**< 1: generate a 32-bits checksum at end of frame, for error detection */ - unsigned noDictIDFlag; /**< 1: no dictID will be saved into frame header (if dictionary compression) */ + unsigned int contentSizeFlag; + unsigned int checksumFlag; + unsigned int noDictIDFlag; } ZSTD_frameParameters; +/** + * struct ZSTD_parameters - zstd parameters + * @cParams: The compression parameters. + * @fParams: The frame parameters. + */ typedef struct { ZSTD_compressionParameters cParams; ZSTD_frameParameters fParams; } ZSTD_parameters; -/*! ZSTD_getCParams() : -* @return ZSTD_compressionParameters structure for a selected compression level and estimated srcSize. -* `estimatedSrcSize` value is optional, select 0 if not known */ -ZSTDLIB_API ZSTD_compressionParameters ZSTD_getCParams(int compressionLevel, unsigned long long estimatedSrcSize, size_t dictSize); +/** + * ZSTD_getCParams() - returns ZSTD_compressionParameters for selected level + * @compressionLevel: The compression level from 1 to ZSTD_maxCLevel(). + * @estimatedSrcSize: The estimated source size to compress or 0 if unknown. + * @dictSize: The dictionary size or 0 if a dictionary isn't being used. + * + * Return: The selected ZSTD_compressionParameters. + */ +ZSTD_compressionParameters ZSTD_getCParams(int compressionLevel, + unsigned long long estimatedSrcSize, size_t dictSize); -/*! ZSTD_getParams() : -* same as ZSTD_getCParams(), but @return a full `ZSTD_parameters` object instead of sub-component `ZSTD_compressionParameters`. -* All fields of `ZSTD_frameParameters` are set to default (0) */ -ZSTDLIB_API ZSTD_parameters ZSTD_getParams(int compressionLevel, unsigned long long estimatedSrcSize, size_t dictSize); +/** + * ZSTD_getParams() - returns ZSTD_parameters for selected level + * @compressionLevel: The compression level from 1 to ZSTD_maxCLevel(). + * @estimatedSrcSize: The estimated source size to compress or 0 if unknown. + * @dictSize: The dictionary size or 0 if a dictionary isn't being used. + * + * The same as ZSTD_getCParams() except also selects the default frame + * parameters (all zero). + * + * Return: The selected ZSTD_parameters. + */ +ZSTD_parameters ZSTD_getParams(int compressionLevel, + unsigned long long estimatedSrcSize, size_t dictSize); +/*-************************************* + * Explicit memory management + **************************************/ -/*! ZSTD_CCtxWorkspaceBound() : -* Returns the minimum amount of memory that needs to be passed to ZSTD_createCCtx() in order to compress with `params.cParams` -* or a `cdict` created with `params.cParams`. */ +/** + * ZSTD_CCtxWorkspaceBound() - the amount of memory needed to create a ZSTD_CCtx + * @cParams: The compression parameters to be used for compression. + * + * If multiple compression parameters might be used, the caller must call + * ZSTD_CCtxWorkspaceBound() for each set of parameters and use the maximum + * size. + * + * Return: A lower bound on the size of the workspace that is passed to + * ZSTD_createCCtx(). + */ size_t ZSTD_CCtxWorkspaceBound(ZSTD_compressionParameters cParams); -/*= Compression context -* When compressing many times, -* it is recommended to allocate a context just once, and re-use it for each successive compression operation. -* The context pointer is placed in `workspace`, which must outlive the returned context. -* Use one context per thread for parallel execution in multi-threaded environments. */ +/** + * struct ZSTD_CCtx - the zstd compression context + * + * When compressing many times it is recommended to allocate a context just once + * and reuse it for each successive compression operation. + */ typedef struct ZSTD_CCtx_s ZSTD_CCtx; -ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx(void* workspace, size_t workspaceSize); +/** + * ZSTD_createCCtx() - create a zstd compression context + * @workspace: The workspace to emplace the context into. It must outlive + * the returned context. + * @workspaceSize: The size of workspace. Use ZSTD_CCtxWorkspaceBound() to + * determine how large the workspace must be. + * + * Return: A compression context emplaced into workspace. + */ +ZSTD_CCtx *ZSTD_createCCtx(void *workspace, size_t workspaceSize); -/*! ZSTD_compressCCtx() : -* Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). -* Note : The workspace passed to ZSTD_createCCtx() must have been at least ZSTD_CCtxWorkspaceBound(params.cParams) bytes. */ -ZSTDLIB_API size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, ZSTD_parameters params); +/** + * ZSTD_compressCCtx() - compress src into dst + * @ctx: The context. Must have been created with a workspace at least + * as large as ZSTD_CCtxWorkspaceBound(params.cParams). + * @dst: The buffer to compress src into. + * @dstCapacity: The size of the destination buffer. May be any size, but + * ZSTD_compressBound(srcSize) is guaranteed to be large enough. + * @src: The data to compress. + * @srcSize: The size of the data to compress. + * @params: The parameters to use for compression. See ZSTD_getParams(). + * + * Return: The compressed size or an error, which can be checked using + * ZSTD_isError(). + */ +size_t ZSTD_compressCCtx(ZSTD_CCtx *ctx, void *dst, size_t dstCapacity, + const void *src, size_t srcSize, ZSTD_parameters params); -/*! ZSTD_compress_usingDict() : -* Compression using a predefined Dictionary (see dictBuilder/zdict.h). -* Note : The workspace passed to ZSTD_createCCtx() must have been at least ZSTD_CCtxWorkspaceBound(params.cParams) bytes. -* Note : This function loads the dictionary, resulting in significant startup delay. -* Note : When `dict == NULL || dictSize < 8` no dictionary is used. */ -ZSTDLIB_API size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void *dict, size_t dictSize, ZSTD_parameters params); - -/*! ZSTD_DCtxWorkspaceBound() : - * Returns the minimum amount of memory that needs to be passed to ZSTD_createDCtx(). */ +/** + * ZSTD_DCtxWorkspaceBound() - the amount of memory needed to create a ZSTD_DCtx + * + * Return: A lower bound on the size of the workspace that is passed to + * ZSTD_createDCtx(). + */ size_t ZSTD_DCtxWorkspaceBound(void); -/*= Decompression context -* When decompressing many times, -* it is recommended to allocate a context just once, and re-use it for each successive compression operation. -* The context pointer is placed in `workspace`, which must outlive the returned context. -* `workspace` must be at least ZSTD_DCtxWorkspaceBound() bytes. -* Use one context per thread for parallel execution in multi-threaded environments. */ +/** + * struct ZSTD_DCtx - the zstd decompression context + * + * When decompressing many times it is recommended to allocate a context just + * once and reuse it for each successive decompression operation. + */ typedef struct ZSTD_DCtx_s ZSTD_DCtx; -ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx(void* workspace, size_t workspaceSize); +/** + * ZSTD_createDCtx() - create a zstd decompression context + * @workspace: The workspace to emplace the context into. It must outlive + * the returned context. + * @workspaceSize: The size of workspace. Use ZSTD_DCtxWorkspaceBound() to + * determine how large the workspace must be. + * + * Return: A decompression context emplaced into workspace. + */ +ZSTD_DCtx *ZSTD_createDCtx(void *workspace, size_t workspaceSize); -/*! ZSTD_decompressDCtx() : -* Same as ZSTD_decompress(), requires an allocated ZSTD_DCtx (see ZSTD_createDCtx()). */ -ZSTDLIB_API size_t ZSTD_decompressDCtx(ZSTD_DCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); +/** + * ZSTD_decompressDCtx() - decompress zstd compressed src into dst + * @ctx: The decompression context. + * @dst: The buffer to decompress src into. + * @dstCapacity: The size of the destination buffer. Must be at least as large + * as the decompressed size. If the caller cannot upper bound the + * decompressed size, then it's better to use the streaming API. + * @src: The zstd compressed data to decompress. Multiple concatenated + * frames and skippable frames are allowed. + * @srcSize: The exact size of the data to decompress. + * + * Return: The decompressed size or an error, which can be checked using + * ZSTD_isError(). + */ +size_t ZSTD_decompressDCtx(ZSTD_DCtx *ctx, void *dst, size_t dstCapacity, + const void *src, size_t srcSize); -/*! ZSTD_decompress_usingDict() : -* Decompression using a predefined Dictionary (see dictBuilder/zdict.h). -* Dictionary must be identical to the one used during compression. -* Note : This function loads the dictionary, resulting in significant startup delay. -* Note : When `dict == NULL || dictSize < 8` no dictionary is used. */ -ZSTDLIB_API size_t ZSTD_decompress_usingDict(ZSTD_DCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void *dict, size_t dictSize); +/*-************************ + * Simple dictionary API + **************************/ -/**************************** -* Fast dictionary API -****************************/ -/*! ZSTD_CDictWorkspaceBound() : - * Returns the minimum amount of memory that needs to be passed to ZSTD_createCDict() when called with the given `params.cParams`. */ +/** + * ZSTD_compress_usingDict() - compress src into dst using a dictionary + * @ctx: The context. Must have been created with a workspace at least + * as large as ZSTD_CCtxWorkspaceBound(params.cParams). + * @dst: The buffer to compress src into. + * @dstCapacity: The size of the destination buffer. May be any size, but + * ZSTD_compressBound(srcSize) is guaranteed to be large enough. + * @src: The data to compress. + * @srcSize: The size of the data to compress. + * @dict: The dictionary to use for compression. + * @dictSize: The size of the dictionary. + * @params: The parameters to use for compression. See ZSTD_getParams(). + * + * Compression using a predefined dictionary. The same dictionary must be used + * during decompression. + * + * Return: The compressed size or an error, which can be checked using + * ZSTD_isError(). + */ +size_t ZSTD_compress_usingDict(ZSTD_CCtx *ctx, void *dst, size_t dstCapacity, + const void *src, size_t srcSize, const void *dict, size_t dictSize, + ZSTD_parameters params); + +/** + * ZSTD_decompress_usingDict() - decompress src into dst using a dictionary + * @ctx: The decompression context. + * @dst: The buffer to decompress src into. + * @dstCapacity: The size of the destination buffer. Must be at least as large + * as the decompressed size. If the caller cannot upper bound the + * decompressed size, then it's better to use the streaming API. + * @src: The zstd compressed data to decompress. Multiple concatenated + * frames and skippable frames are allowed. + * @srcSize: The exact size of the data to decompress. + * @dict: The dictionary to use for decompression. The same dictionary + * must've been used to compress the data. + * @dictSize: The size of the dictionary. + * + * Return: The decompressed size or an error, which can be checked using + * ZSTD_isError(). + */ +size_t ZSTD_decompress_usingDict(ZSTD_DCtx *ctx, void *dst, size_t dstCapacity, + const void *src, size_t srcSize, const void *dict, size_t dictSize); + +/*-************************** + * Fast dictionary API + ***************************/ + +/** + * ZSTD_CDictWorkspaceBound() - amount of memory needed to create a ZSTD_CDict + * @cParams: The compression parameters to be used for compression. + * + * Return: A lower bound on the size of the workspace that is passed to + * ZSTD_createCDict(). + */ size_t ZSTD_CDictWorkspaceBound(ZSTD_compressionParameters cParams); +/** + * struct ZSTD_CDict - a digested dictionary to be used for compression + */ typedef struct ZSTD_CDict_s ZSTD_CDict; -/*! ZSTD_createCDict() : -* When compressing multiple messages / blocks with the same dictionary, it's recommended to load it just once. -* ZSTD_createCDict() will create a digested dictionary, ready to start future compression operations without startup delay. -* ZSTD_CDict can be created once and used by multiple threads concurrently, as its usage is read-only. -* `dictBuffer` content is referenced, and it must remain accessible throughout the lifetime of the CDict. -* The cdict pointer is placed in `workspace`, which must outlive the returned cdict. -* `workspace` must be at least ZSTD_CDictWorkspaceBound(params.cParams) bytes. */ -ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict(const void* dictBuffer, size_t dictSize, ZSTD_parameters params, void* workspace, size_t workspaceSize); +/** + * ZSTD_createCDict() - create a digested dictionary for compression + * @dictBuffer: The dictionary to digest. The buffer is referenced by the + * ZSTD_CDict so it must outlive the returned ZSTD_CDict. + * @dictSize: The size of the dictionary. + * @params: The parameters to use for compression. See ZSTD_getParams(). + * @workspace: The workspace. It must outlive the returned ZSTD_CDict. + * @workspaceSize: The workspace size. Must be at least + * ZSTD_CDictWorkspaceBound(params.cParams). + * + * When compressing multiple messages / blocks with the same dictionary it is + * recommended to load it just once. The ZSTD_CDict merely references the + * dictBuffer, so it must outlive the returned ZSTD_CDict. + * + * Return: The digested dictionary emplaced into workspace. + */ +ZSTD_CDict *ZSTD_createCDict(const void *dictBuffer, size_t dictSize, + ZSTD_parameters params, void *workspace, size_t workspaceSize); -/*! ZSTD_compress_usingCDict() : -* Compression using a digested Dictionary. -* Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. -* Note that compression level is decided during dictionary creation. */ -ZSTDLIB_API size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const ZSTD_CDict* cdict); +/** + * ZSTD_compress_usingCDict() - compress src into dst using a ZSTD_CDict + * @ctx: The context. Must have been created with a workspace at least + * as large as ZSTD_CCtxWorkspaceBound(cParams) where cParams are + * the compression parameters used to create cdict. + * @dst: The buffer to compress src into. + * @dstCapacity: The size of the destination buffer. May be any size, but + * ZSTD_compressBound(srcSize) is guaranteed to be large enough. + * @src: The data to compress. + * @srcSize: The size of the data to compress. + * @cdict: The digested dictionary to use for compression. + * @params: The parameters to use for compression. See ZSTD_getParams(). + * + * Compression using a digested dictionary. The same dictionary must be used + * during decompression. + * + * Return: The compressed size or an error, which can be checked using + * ZSTD_isError(). + */ +size_t ZSTD_compress_usingCDict(ZSTD_CCtx *cctx, void *dst, size_t dstCapacity, + const void *src, size_t srcSize, const ZSTD_CDict *cdict); -/*! ZSTD_DDictWorkspaceBound() : - * Returns the minimum amount of memory that needs to be passed to ZSTD_createDDict(). */ +/** + * ZSTD_DDictWorkspaceBound() - amount of memory needed to create a ZSTD_DDict + * + * Return: A lower bound on the size of the workspace that is passed to + * ZSTD_createDDict(). + */ size_t ZSTD_DDictWorkspaceBound(void); +/** + * struct ZSTD_DDict - a digested dictionary to be used for decompression + */ typedef struct ZSTD_DDict_s ZSTD_DDict; -/*! ZSTD_createDDict() : -* Create a digested dictionary, ready to start decompression operation without startup delay. -* `dictBuffer` content is referenced, and it must remain accessible throughout the lifetime of the DDict. -* The ddict pointer is placed in `workspace`, which must outlive the returned ddict. -* `workspace` must be at least ZSTD_DDictWorkspaceBound() bytes. */ -ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict(const void* dictBuffer, size_t dictSize, void* workspace, size_t workspaceSize); +/** + * ZSTD_createDDict() - create a digested dictionary for decompression + * @dictBuffer: The dictionary to digest. The buffer is referenced by the + * ZSTD_DDict so it must outlive the returned ZSTD_DDict. + * @dictSize: The size of the dictionary. + * @workspace: The workspace. It must outlive the returned ZSTD_DDict. + * @workspaceSize: The workspace size. Must be at least + * ZSTD_DDictWorkspaceBound(). + * + * When decompressing multiple messages / blocks with the same dictionary it is + * recommended to load it just once. The ZSTD_DDict merely references the + * dictBuffer, so it must outlive the returned ZSTD_DDict. + * + * Return: The digested dictionary emplaced into workspace. + */ +ZSTD_DDict *ZSTD_createDDict(const void *dictBuffer, size_t dictSize, + void *workspace, size_t workspaceSize); -/*! ZSTD_decompress_usingDDict() : -* Decompression using a digested Dictionary. -* Faster startup than ZSTD_decompress_usingDict(), recommended when same dictionary is used multiple times. */ -ZSTDLIB_API size_t ZSTD_decompress_usingDDict(ZSTD_DCtx* dctx, - void* dst, size_t dstCapacity, - const void* src, size_t srcSize, - const ZSTD_DDict* ddict); +/** + * ZSTD_decompress_usingDDict() - decompress src into dst using a ZSTD_DDict + * @ctx: The decompression context. + * @dst: The buffer to decompress src into. + * @dstCapacity: The size of the destination buffer. Must be at least as large + * as the decompressed size. If the caller cannot upper bound the + * decompressed size, then it's better to use the streaming API. + * @src: The zstd compressed data to decompress. Multiple concatenated + * frames and skippable frames are allowed. + * @srcSize: The exact size of the data to decompress. + * @ddict: The digested dictionary to use for decompression. The same + * dictionary must've been used to compress the data. + * + * Return: The decompressed size or an error, which can be checked using + * ZSTD_isError(). + */ +size_t ZSTD_decompress_usingDDict(ZSTD_DCtx *dctx, void *dst, + size_t dstCapacity, const void *src, size_t srcSize, + const ZSTD_DDict *ddict); -/**************************** -* Streaming -****************************/ +/*-************************** + * Streaming + ***************************/ +/** + * struct ZSTD_inBuffer - input buffer for streaming + * @src: Start of the input buffer. + * @size: Size of the input buffer. + * @pos: Position where reading stopped. Will be updated. + * Necessarily 0 <= pos <= size. + */ typedef struct ZSTD_inBuffer_s { - const void* src; /**< start of input buffer */ - size_t size; /**< size of input buffer */ - size_t pos; /**< position where reading stopped. Will be updated. Necessarily 0 <= pos <= size */ + const void *src; + size_t size; + size_t pos; } ZSTD_inBuffer; +/** + * struct ZSTD_outBuffer - output buffer for streaming + * @dst: Start of the output buffer. + * @size: Size of the output buffer. + * @pos: Position where writing stopped. Will be updated. + * Necessarily 0 <= pos <= size. + */ typedef struct ZSTD_outBuffer_s { - void* dst; /**< start of output buffer */ - size_t size; /**< size of output buffer */ - size_t pos; /**< position where writing stopped. Will be updated. Necessarily 0 <= pos <= size */ + void *dst; + size_t size; + size_t pos; } ZSTD_outBuffer; -/*-*********************************************************************** -* Streaming compression - HowTo -* -* A ZSTD_CStream object is required to track streaming operation. -* Use ZSTD_createCStream() and ZSTD_freeCStream() to create/release resources. -* ZSTD_CStream objects can be reused multiple times on consecutive compression operations. -* It is recommended to re-use ZSTD_CStream in situations where many streaming operations will be achieved consecutively. -* Use one separate ZSTD_CStream per thread for parallel execution. -* -* Start a new compression by initializing ZSTD_CStream. -* Use ZSTD_initCStream() to start a new compression operation. -* Use ZSTD_initCStream_usingCDict() for a compression which requires a dictionary. -* -* Use ZSTD_compressStream() repetitively to consume input stream. -* The function will automatically update both `pos` fields. -* Note that it may not consume the entire input, in which case `pos < size`, -* and it's up to the caller to present again remaining data. -* @return : a size hint, preferred nb of bytes to use as input for next function call -* or an error code, which can be tested using ZSTD_isError(). -* Note 1 : it's just a hint, to help latency a little, any other value will work fine. -* Note 2 : size hint is guaranteed to be <= ZSTD_CStreamInSize() -* -* At any moment, it's possible to flush whatever data remains within internal buffer, using ZSTD_flushStream(). -* `output->pos` will be updated. -* Note that some content might still be left within internal buffer if `output->size` is too small. -* @return : nb of bytes still present within internal buffer (0 if it's empty) -* or an error code, which can be tested using ZSTD_isError(). -* -* ZSTD_endStream() instructs to finish a frame. -* It will perform a flush and write frame epilogue. -* The epilogue is required for decoders to consider a frame completed. -* Similar to ZSTD_flushStream(), it may not be able to flush the full content if `output->size` is too small. -* In which case, call again ZSTD_endStream() to complete the flush. -* @return : nb of bytes still present within internal buffer (0 if it's empty, hence compression completed) -* or an error code, which can be tested using ZSTD_isError(). -* -* *******************************************************************/ +/*-***************************************************************************** + * Streaming compression - HowTo + * + * A ZSTD_CStream object is required to track streaming operation. + * Use ZSTD_createCStream() to create and initialize a ZSTD_CStream object. + * ZSTD_CStream objects can be reused multiple times on consecutive compression + * operations. It is recommended to re-use ZSTD_CStream in situations where many + * streaming operations will be achieved consecutively. Use one separate + * ZSTD_CStream per thread for parallel execution. + * + * Use ZSTD_compressStream() repetitively to consume input stream. + * The function will automatically update both `pos` fields. + * Note that it may not consume the entire input, in which case `pos < size`, + * and it's up to the caller to present again remaining data. + * It returns a hint for the preferred number of bytes to use as an input for + * the next function call. + * + * At any moment, it's possible to flush whatever data remains within internal + * buffer, using ZSTD_flushStream(). `output->pos` will be updated. There might + * still be some content left within the internal buffer if `output->size` is + * too small. It returns the number of bytes left in the internal buffer and + * must be called until it returns 0. + * + * ZSTD_endStream() instructs to finish a frame. It will perform a flush and + * write frame epilogue. The epilogue is required for decoders to consider a + * frame completed. Similar to ZSTD_flushStream(), it may not be able to flush + * the full content if `output->size` is too small. In which case, call again + * ZSTD_endStream() to complete the flush. It returns the number of bytes left + * in the internal buffer and must be called until it returns 0. + ******************************************************************************/ -/*! ZSTD_CStreamWorkspaceBound() : - * Returns the minimum amount of memory that needs to be passed to ZSTD_createCStream() or ZSTD_createCStream_usingCDict() - * when called with the given `params.cParams` or `cdict` created with `params.cParams`. */ +/** + * ZSTD_CStreamWorkspaceBound() - memory needed to create a ZSTD_CStream + * @cParams: The compression parameters to be used for compression. + * + * Return: A lower bound on the size of the workspace that is passed to + * ZSTD_createCStream() and ZSTD_createCStream_usingCDict(). + */ size_t ZSTD_CStreamWorkspaceBound(ZSTD_compressionParameters cParams); +/** + * struct ZSTD_CStream - the zstd streaming compression context + */ typedef struct ZSTD_CStream_s ZSTD_CStream; + /*===== ZSTD_CStream management functions =====*/ -/*! ZSTD_createCStream() : -* Creates a cstream using params. -* Callers may optionally provide the size of the source they intend to compress, or pass 0 if unknown. -* The stream is placed in `workspace`, which must outlive the returned stream. -* `workspace` must be at least ZSTD_CStreamWorkspaceBound(params.cParams) bytes. */ -ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream(ZSTD_parameters params, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize); -/*! ZSTD_createCStream_usingCDict() : -* Similar to ZSTD_createCStream(), but use the given preprocessed dictionary. -*/ -ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_usingCDict(const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize); +/** + * ZSTD_createCStream() - create a zstd streaming compression context + * @params: The zstd compression parameters. + * @pledgedSrcSize: If params.fParams.contentSizeFlag == 1 then the caller must + * pass the source size (zero means empty source). Otherwise, + * the caller may optionally pass the source size, or zero if + * unknown. + * @workspace: The workspace to emplace the context into. It must outlive + * the returned context. + * @workspaceSize: The size of workspace. + * Use ZSTD_CStreamWorkspaceBound(params.cParams) to determine + * how large the workspace must be. + * + * Return: The zstd streaming compression context. + */ +ZSTD_CStream *ZSTD_createCStream(ZSTD_parameters params, + unsigned long long pledgedSrcSize, void *workspace, + size_t workspaceSize); + +/** + * ZSTD_createCStream_usingCDict() - create a zstd streaming compression context + * @cdict: The digested dictionary to use for compression. + * @pledgedSrcSize: Optionally the source size, or zero if unknown. + * @workspace: The workspace to emplace the context into. It must outlive + * the returned context. + * @workspaceSize: The size of workspace. Call ZSTD_CStreamWorkspaceBound() + * with the cParams used to create the cdict to determine how + * large the workspace must be. + * + * Return: The zstd streaming compression context. + */ +ZSTD_CStream *ZSTD_createCStream_usingCDict(const ZSTD_CDict *cdict, + unsigned long long pledgedSrcSize, void *workspace, + size_t workspaceSize); /*===== Streaming compression functions =====*/ -ZSTDLIB_API size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize); /**< re-use compression parameters from previous init; skip dictionary loading stage; zcs must be init at least once before. note: pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ -ZSTDLIB_API size_t ZSTD_compressStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input); -ZSTDLIB_API size_t ZSTD_flushStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output); -ZSTDLIB_API size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output); +/** + * ZSTD_resetCStream() - reset the context using parameters from creation + * @zcs: The zstd streaming compression context to reset. + * @pledgedSrcSize: Optionally the source size, or zero if unknown. + * + * Resets the context using the parameters from creation. Skips dictionary + * loading, since it can be reused. If `pledgedSrcSize` is non-zero the frame + * content size is always written into the frame header. + * + * Return: Zero or an error, which can be checked using ZSTD_isError(). + */ +size_t ZSTD_resetCStream(ZSTD_CStream *zcs, unsigned long long pledgedSrcSize); +/** + * ZSTD_compressStream() - streaming compress some of input into output + * @zcs: The zstd streaming compression context. + * @output: Destination buffer. `output->pos` is updated to indicate how much + * compressed data was written. + * @input: Source buffer. `input->pos` is updated to indicate how much data was + * read. Note that it may not consume the entire input, in which case + * `input->pos < input->size`, and it's up to the caller to present + * remaining data again. + * + * The `input` and `output` buffers may be any size. Guaranteed to make some + * forward progress if `input` and `output` are not empty. + * + * Return: A hint for the number of bytes to use as the input for the next + * function call or an error, which can be checked using + * ZSTD_isError(). + */ +size_t ZSTD_compressStream(ZSTD_CStream *zcs, ZSTD_outBuffer *output, + ZSTD_inBuffer *input); +/** + * ZSTD_flushStream() - flush internal buffers into output + * @zcs: The zstd streaming compression context. + * @output: Destination buffer. `output->pos` is updated to indicate how much + * compressed data was written. + * + * ZSTD_flushStream() must be called until it returns 0, meaning all the data + * has been flushed. Since ZSTD_flushStream() causes a block to be ended, + * calling it too often will degrade the compression ratio. + * + * Return: The number of bytes still present within internal buffers or an + * error, which can be checked using ZSTD_isError(). + */ +size_t ZSTD_flushStream(ZSTD_CStream *zcs, ZSTD_outBuffer *output); +/** + * ZSTD_endStream() - flush internal buffers into output and end the frame + * @zcs: The zstd streaming compression context. + * @output: Destination buffer. `output->pos` is updated to indicate how much + * compressed data was written. + * + * ZSTD_endStream() must be called until it returns 0, meaning all the data has + * been flushed and the frame epilogue has been written. + * + * Return: The number of bytes still present within internal buffers or an + * error, which can be checked using ZSTD_isError(). + */ +size_t ZSTD_endStream(ZSTD_CStream *zcs, ZSTD_outBuffer *output); -ZSTDLIB_API size_t ZSTD_CStreamInSize(void); /**< recommended size for input buffer */ -ZSTDLIB_API size_t ZSTD_CStreamOutSize(void); /**< recommended size for output buffer. Guarantee to successfully flush at least one complete compressed block in all circumstances. */ +/** + * ZSTD_CStreamInSize() - recommended size for the input buffer + * + * Return: The recommended size for the input buffer. + */ +size_t ZSTD_CStreamInSize(void); +/** + * ZSTD_CStreamOutSize() - recommended size for the output buffer + * + * When the output buffer is at least this large, it is guaranteed to be large + * enough to flush at least one complete compressed block. + * + * Return: The recommended size for the output buffer. + */ +size_t ZSTD_CStreamOutSize(void); -/*-*************************************************************************** -* Streaming decompression - HowTo -* -* A ZSTD_DStream object is required to track streaming operations. -* Use ZSTD_createDStream() and ZSTD_freeDStream() to create/release resources. -* ZSTD_DStream objects can be re-used multiple times. -* -* Use ZSTD_initDStream() to start a new decompression operation, -* or ZSTD_initDStream_usingDict() if decompression requires a dictionary. -* @return : recommended first input size -* -* Use ZSTD_decompressStream() repetitively to consume your input. -* The function will update both `pos` fields. -* If `input.pos < input.size`, some input has not been consumed. -* It's up to the caller to present again remaining data. -* If `output.pos < output.size`, decoder has flushed everything it could. -* @return : 0 when a frame is completely decoded and fully flushed, -* an error code, which can be tested using ZSTD_isError(), -* any other value > 0, which means there is still some decoding to do to complete current frame. -* The return value is a suggested next input size (a hint to improve latency) that will never load more than the current frame. -* *******************************************************************************/ +/*-***************************************************************************** + * Streaming decompression - HowTo + * + * A ZSTD_DStream object is required to track streaming operations. + * Use ZSTD_createDStream() to initialize a ZSTD_DStream object. + * ZSTD_DStream objects can be re-used multiple times. + * + * Use ZSTD_decompressStream() repetitively to consume your input. + * The function will update both `pos` fields. + * If `input->pos < input->size`, some input has not been consumed. + * It's up to the caller to present again remaining data. + * If `output->pos < output->size`, decoder has flushed everything it could. + * Returns 0 iff a frame is completely decoded and fully flushed. + * Otherwise it returns a suggested next input size that will never load more + * than the current frame. + ******************************************************************************/ -/*! ZSTD_DStreamWorkspaceBound() : - * Returns the minimum amount of memory that needs to be passed to ZSTD_createDStream() to decompress frames with windowSize <= maxWindowSize. */ +/** + * ZSTD_DStreamWorkspaceBound() - memory needed to create a ZSTD_DStream + * @maxWindowSize: The maximum window size allowed for compressed frames. + * + * Return: A lower bound on the size of the workspace that is passed to + * ZSTD_createDStream() and ZSTD_createDStream_usingDDict(). + */ size_t ZSTD_DStreamWorkspaceBound(size_t maxWindowSize); +/** + * struct ZSTD_DStream - the zstd streaming decompression context + */ typedef struct ZSTD_DStream_s ZSTD_DStream; /*===== ZSTD_DStream management functions =====*/ -/*! ZSTD_createDStream() : -* Creates a dstream that can decompress frames with windowSize up to maxWindowSize. -* The stream is placed in `workspace`, which must outlive the returned stream. -* `workspace` must be at least ZSTD_DStreamWorkspaceBound(maxWindowSize) bytes. */ -ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream(size_t maxWindowSize, void* workspace, size_t workspaceSize); -/*! ZSTD_createDStream_usingDDict() : -* Similar to ZSTD_createCStream(), but use the given preprocessed dictionary. */ -ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream_usingDDict(size_t maxWindowSize, const ZSTD_DDict* ddict, void* workspace, size_t workspaceSize); +/** + * ZSTD_createDStream() - create a zstd streaming decompression context + * @maxWindowSize: The maximum window size allowed for compressed frames. + * @workspace: The workspace to emplace the context into. It must outlive + * the returned context. + * @workspaceSize: The size of workspace. + * Use ZSTD_DStreamWorkspaceBound(maxWindowSize) to determine + * how large the workspace must be. + * + * Return: The zstd streaming decompression context. + */ +ZSTD_DStream *ZSTD_createDStream(size_t maxWindowSize, void *workspace, + size_t workspaceSize); +/** + * ZSTD_createDStream_usingDDict() - create zstd streaming decompression context + * @maxWindowSize: The maximum window size allowed for compressed frames. + * @ddict: The digested dictionary to use for decompression. + * @workspace: The workspace to emplace the context into. It must outlive + * the returned context. + * @workspaceSize: The size of workspace. + * Use ZSTD_DStreamWorkspaceBound(maxWindowSize) to determine + * how large the workspace must be. + * + * Return: The zstd streaming decompression context. + */ +ZSTD_DStream *ZSTD_createDStream_usingDDict(size_t maxWindowSize, + const ZSTD_DDict *ddict, void *workspace, size_t workspaceSize); /*===== Streaming decompression functions =====*/ -ZSTDLIB_API size_t ZSTD_resetDStream(ZSTD_DStream* zds); /**< re-use decompression parameters from previous init; saves dictionary loading */ -ZSTDLIB_API size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input); +/** + * ZSTD_resetDStream() - reset the context using parameters from creation + * @zds: The zstd streaming decompression context to reset. + * + * Resets the context using the parameters from creation. Skips dictionary + * loading, since it can be reused. + * + * Return: Zero or an error, which can be checked using ZSTD_isError(). + */ +size_t ZSTD_resetDStream(ZSTD_DStream *zds); +/** + * ZSTD_decompressStream() - streaming decompress some of input into output + * @zds: The zstd streaming decompression context. + * @output: Destination buffer. `output.pos` is updated to indicate how much + * decompressed data was written. + * @input: Source buffer. `input.pos` is updated to indicate how much data was + * read. Note that it may not consume the entire input, in which case + * `input.pos < input.size`, and it's up to the caller to present + * remaining data again. + * + * The `input` and `output` buffers may be any size. Guaranteed to make some + * forward progress if `input` and `output` are not empty. + * ZSTD_decompressStream() will not consume the last byte of the frame until + * the entire frame is flushed. + * + * Return: Returns 0 iff a frame is completely decoded and fully flushed. + * Otherwise returns a hint for the number of bytes to use as the input + * for the next function call or an error, which can be checked using + * ZSTD_isError(). The size hint will never load more than the frame. + */ +size_t ZSTD_decompressStream(ZSTD_DStream *zds, ZSTD_outBuffer *output, + ZSTD_inBuffer *input); -ZSTDLIB_API size_t ZSTD_DStreamInSize(void); /*!< recommended size for input buffer */ -ZSTDLIB_API size_t ZSTD_DStreamOutSize(void); /*!< recommended size for output buffer. Guarantee to successfully flush at least one complete block in all circumstances. */ +/** + * ZSTD_DStreamInSize() - recommended size for the input buffer + * + * Return: The recommended size for the input buffer. + */ +size_t ZSTD_DStreamInSize(void); +/** + * ZSTD_DStreamOutSize() - recommended size for the output buffer + * + * When the output buffer is at least this large, it is guaranteed to be large + * enough to flush at least one complete decompressed block. + * + * Return: The recommended size for the output buffer. + */ +size_t ZSTD_DStreamOutSize(void); /* --- Constants ---*/ @@ -366,281 +761,389 @@ ZSTDLIB_API size_t ZSTD_DStreamOutSize(void); /*!< recommended size for output #define ZSTD_WINDOWLOG_MAX_32 27 #define ZSTD_WINDOWLOG_MAX_64 27 -#define ZSTD_WINDOWLOG_MAX ((unsigned)(sizeof(size_t) == 4 ? ZSTD_WINDOWLOG_MAX_32 : ZSTD_WINDOWLOG_MAX_64)) -#define ZSTD_WINDOWLOG_MIN 10 -#define ZSTD_HASHLOG_MAX ZSTD_WINDOWLOG_MAX +#define ZSTD_WINDOWLOG_MAX \ + ((unsigned int)(sizeof(size_t) == 4 \ + ? ZSTD_WINDOWLOG_MAX_32 \ + : ZSTD_WINDOWLOG_MAX_64)) +#define ZSTD_WINDOWLOG_MIN 10 +#define ZSTD_HASHLOG_MAX ZSTD_WINDOWLOG_MAX #define ZSTD_HASHLOG_MIN 6 #define ZSTD_CHAINLOG_MAX (ZSTD_WINDOWLOG_MAX+1) #define ZSTD_CHAINLOG_MIN ZSTD_HASHLOG_MIN #define ZSTD_HASHLOG3_MAX 17 #define ZSTD_SEARCHLOG_MAX (ZSTD_WINDOWLOG_MAX-1) #define ZSTD_SEARCHLOG_MIN 1 -#define ZSTD_SEARCHLENGTH_MAX 7 /* only for ZSTD_fast, other strategies are limited to 6 */ -#define ZSTD_SEARCHLENGTH_MIN 3 /* only for ZSTD_btopt, other strategies are limited to 4 */ +/* only for ZSTD_fast, other strategies are limited to 6 */ +#define ZSTD_SEARCHLENGTH_MAX 7 +/* only for ZSTD_btopt, other strategies are limited to 4 */ +#define ZSTD_SEARCHLENGTH_MIN 3 #define ZSTD_TARGETLENGTH_MIN 4 #define ZSTD_TARGETLENGTH_MAX 999 -#define ZSTD_FRAMEHEADERSIZE_MAX 18 /* for static allocation */ +/* for static allocation */ +#define ZSTD_FRAMEHEADERSIZE_MAX 18 #define ZSTD_FRAMEHEADERSIZE_MIN 6 static const size_t ZSTD_frameHeaderSize_prefix = 5; static const size_t ZSTD_frameHeaderSize_min = ZSTD_FRAMEHEADERSIZE_MIN; static const size_t ZSTD_frameHeaderSize_max = ZSTD_FRAMEHEADERSIZE_MAX; -static const size_t ZSTD_skippableHeaderSize = 8; /* magic number + skippable frame length */ +/* magic number + skippable frame length */ +static const size_t ZSTD_skippableHeaderSize = 8; -/*************************************** -* Compressed size functions -***************************************/ +/*-************************************* + * Compressed size functions + **************************************/ -/*! ZSTD_findFrameCompressedSize() : - * `src` should point to the start of a ZSTD encoded frame or skippable frame - * `srcSize` must be at least as large as the frame - * @return : the compressed size of the frame pointed to by `src`, suitable to pass to - * `ZSTD_decompress` or similar, or an error code if given invalid input. */ -ZSTDLIB_API size_t ZSTD_findFrameCompressedSize(const void* src, size_t srcSize); +/** + * ZSTD_findFrameCompressedSize() - returns the size of a compressed frame + * @src: Source buffer. It should point to the start of a zstd encoded frame + * or a skippable frame. + * @srcSize: The size of the source buffer. It must be at least as large as the + * size of the frame. + * + * Return: The compressed size of the frame pointed to by `src` or an error, + * which can be check with ZSTD_isError(). + * Suitable to pass to ZSTD_decompress() or similar functions. + */ +size_t ZSTD_findFrameCompressedSize(const void *src, size_t srcSize); -/*************************************** -* Decompressed size functions -***************************************/ -/*! ZSTD_getFrameContentSize() : -* `src` should point to the start of a ZSTD encoded frame -* `srcSize` must be at least as large as the frame header. A value greater than or equal -* to `ZSTD_frameHeaderSize_max` is guaranteed to be large enough in all cases. -* @return : decompressed size of the frame pointed to be `src` if known, otherwise -* - ZSTD_CONTENTSIZE_UNKNOWN if the size cannot be determined -* - ZSTD_CONTENTSIZE_ERROR if an error occurred (e.g. invalid magic number, srcSize too small) */ -ZSTDLIB_API unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize); +/*-************************************* + * Decompressed size functions + **************************************/ +/** + * ZSTD_getFrameContentSize() - returns the content size in a zstd frame header + * @src: It should point to the start of a zstd encoded frame. + * @srcSize: The size of the source buffer. It must be at least as large as the + * frame header. `ZSTD_frameHeaderSize_max` is always large enough. + * + * Return: The frame content size stored in the frame header if known. + * `ZSTD_CONTENTSIZE_UNKNOWN` if the content size isn't stored in the + * frame header. `ZSTD_CONTENTSIZE_ERROR` on invalid input. + */ +unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize); -/*! ZSTD_findDecompressedSize() : -* `src` should point the start of a series of ZSTD encoded and/or skippable frames -* `srcSize` must be the _exact_ size of this series -* (i.e. there should be a frame boundary exactly `srcSize` bytes after `src`) -* @return : the decompressed size of all data in the contained frames, as a 64-bit value _if known_ -* - if the decompressed size cannot be determined: ZSTD_CONTENTSIZE_UNKNOWN -* - if an error occurred: ZSTD_CONTENTSIZE_ERROR -* -* note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode. -* When `return==ZSTD_CONTENTSIZE_UNKNOWN`, data to decompress could be any size. -* In which case, it's necessary to use streaming mode to decompress data. -* Optionally, application can still use ZSTD_decompress() while relying on implied limits. -* (For example, data may be necessarily cut into blocks <= 16 KB). -* note 2 : decompressed size is always present when compression is done with ZSTD_compress() -* note 3 : decompressed size can be very large (64-bits value), -* potentially larger than what local system can handle as a single memory segment. -* In which case, it's necessary to use streaming mode to decompress data. -* note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified. -* Always ensure result fits within application's authorized limits. -* Each application can set its own limits. -* note 5 : ZSTD_findDecompressedSize handles multiple frames, and so it must traverse the input to -* read each contained frame header. This is efficient as most of the data is skipped, -* however it does mean that all frame data must be present and valid. */ -ZSTDLIB_API unsigned long long ZSTD_findDecompressedSize(const void* src, size_t srcSize); +/** + * ZSTD_findDecompressedSize() - returns decompressed size of a series of frames + * @src: It should point to the start of a series of zstd encoded and/or + * skippable frames. + * @srcSize: The exact size of the series of frames. + * + * If any zstd encoded frame in the series doesn't have the frame content size + * set, `ZSTD_CONTENTSIZE_UNKNOWN` is returned. But frame content size is always + * set when using ZSTD_compress(). The decompressed size can be very large. + * If the source is untrusted, the decompressed size could be wrong or + * intentionally modified. Always ensure the result fits within the + * application's authorized limits. ZSTD_findDecompressedSize() handles multiple + * frames, and so it must traverse the input to read each frame header. This is + * efficient as most of the data is skipped, however it does mean that all frame + * data must be present and valid. + * + * Return: Decompressed size of all the data contained in the frames if known. + * `ZSTD_CONTENTSIZE_UNKNOWN` if the decompressed size is unknown. + * `ZSTD_CONTENTSIZE_ERROR` if an error occurred. + */ +unsigned long long ZSTD_findDecompressedSize(const void *src, size_t srcSize); -/*************************************** -* Advanced compression functions -***************************************/ -/*! ZSTD_checkCParams() : -* Ensure param values remain within authorized range */ -ZSTDLIB_API size_t ZSTD_checkCParams(ZSTD_compressionParameters params); +/*-************************************* + * Advanced compression functions + **************************************/ +/** + * ZSTD_checkCParams() - ensure parameter values remain within authorized range + * @cParams: The zstd compression parameters. + * + * Return: Zero or an error, which can be checked using ZSTD_isError(). + */ +size_t ZSTD_checkCParams(ZSTD_compressionParameters cParams); -/*! ZSTD_adjustCParams() : -* optimize params for a given `srcSize` and `dictSize`. -* both values are optional, select `0` if unknown. */ -ZSTDLIB_API ZSTD_compressionParameters ZSTD_adjustCParams(ZSTD_compressionParameters cPar, unsigned long long srcSize, size_t dictSize); +/** + * ZSTD_adjustCParams() - optimize parameters for a given srcSize and dictSize + * @srcSize: Optionally the estimated source size, or zero if unknown. + * @dictSize: Optionally the estimated dictionary size, or zero if unknown. + * + * Return: The optimized parameters. + */ +ZSTD_compressionParameters ZSTD_adjustCParams( + ZSTD_compressionParameters cParams, unsigned long long srcSize, + size_t dictSize); /*--- Advanced decompression functions ---*/ -/*! ZSTD_isFrame() : - * Tells if the content of `buffer` starts with a valid Frame Identifier. - * Note : Frame Identifier is 4 bytes. If `size < 4`, @return will always be 0. - * Note 2 : Legacy Frame Identifiers are considered valid only if Legacy Support is enabled. - * Note 3 : Skippable Frame Identifiers are considered valid. */ -ZSTDLIB_API unsigned ZSTD_isFrame(const void* buffer, size_t size); - -/*! ZSTD_getDictID_fromDict() : - * Provides the dictID stored within dictionary. - * if @return == 0, the dictionary is not conformant with Zstandard specification. - * It can still be loaded, but as a content-only dictionary. */ -ZSTDLIB_API unsigned ZSTD_getDictID_fromDict(const void* dict, size_t dictSize); - -/*! ZSTD_getDictID_fromDDict() : - * Provides the dictID of the dictionary loaded into `ddict`. - * If @return == 0, the dictionary is not conformant to Zstandard specification, or empty. - * Non-conformant dictionaries can still be loaded, but as content-only dictionaries. */ -ZSTDLIB_API unsigned ZSTD_getDictID_fromDDict(const ZSTD_DDict* ddict); - -/*! ZSTD_getDictID_fromFrame() : - * Provides the dictID required to decompressed the frame stored within `src`. - * If @return == 0, the dictID could not be decoded. - * This could for one of the following reasons : - * - The frame does not require a dictionary to be decoded (most common case). - * - The frame was built with dictID intentionally removed. Whatever dictionary is necessary is a hidden information. - * Note : this use case also happens when using a non-conformant dictionary. - * - `srcSize` is too small, and as a result, the frame header could not be decoded (only possible if `srcSize < ZSTD_FRAMEHEADERSIZE_MAX`). - * - This is not a Zstandard frame. - * When identifying the exact failure cause, it's possible to used ZSTD_getFrameParams(), which will provide a more precise error code. */ -ZSTDLIB_API unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize); - - -/********************************************************************* -* Buffer-less and synchronous inner streaming functions -* -* This is an advanced API, giving full control over buffer management, for users which need direct control over memory. -* But it's also a complex one, with many restrictions (documented below). -* Prefer using normal streaming API for an easier experience -********************************************************************* */ +/** + * ZSTD_isFrame() - returns true iff the buffer starts with a valid frame + * @buffer: The source buffer to check. + * @size: The size of the source buffer, must be at least 4 bytes. + * + * Return: True iff the buffer starts with a zstd or skippable frame identifier. + */ +unsigned int ZSTD_isFrame(const void *buffer, size_t size); /** - Buffer-less streaming compression (synchronous mode) + * ZSTD_getDictID_fromDict() - returns the dictionary id stored in a dictionary + * @dict: The dictionary buffer. + * @dictSize: The size of the dictionary buffer. + * + * Return: The dictionary id stored within the dictionary or 0 if the + * dictionary is not a zstd dictionary. If it returns 0 the + * dictionary can still be loaded as a content-only dictionary. + */ +unsigned int ZSTD_getDictID_fromDict(const void *dict, size_t dictSize); - A ZSTD_CCtx object is required to track streaming operations. - Use ZSTD_createCCtx() / ZSTD_freeCCtx() to manage resource. - ZSTD_CCtx object can be re-used multiple times within successive compression operations. +/** + * ZSTD_getDictID_fromDDict() - returns the dictionary id stored in a ZSTD_DDict + * @ddict: The ddict to find the id of. + * + * Return: The dictionary id stored within `ddict` or 0 if the dictionary is not + * a zstd dictionary. If it returns 0 `ddict` will be loaded as a + * content-only dictionary. + */ +unsigned int ZSTD_getDictID_fromDDict(const ZSTD_DDict *ddict); - Start by initializing a context. - Use ZSTD_compressBegin(), or ZSTD_compressBegin_usingDict() for dictionary compression, - or ZSTD_compressBegin_advanced(), for finer parameter control. - It's also possible to duplicate a reference context which has already been initialized, using ZSTD_copyCCtx() - - Then, consume your input using ZSTD_compressContinue(). - There are some important considerations to keep in mind when using this advanced function : - - ZSTD_compressContinue() has no internal buffer. It uses externally provided buffer only. - - Interface is synchronous : input is consumed entirely and produce 1+ (or more) compressed blocks. - - Caller must ensure there is enough space in `dst` to store compressed data under worst case scenario. - Worst case evaluation is provided by ZSTD_compressBound(). - ZSTD_compressContinue() doesn't guarantee recover after a failed compression. - - ZSTD_compressContinue() presumes prior input ***is still accessible and unmodified*** (up to maximum distance size, see WindowLog). - It remembers all previous contiguous blocks, plus one separated memory segment (which can itself consists of multiple contiguous blocks) - - ZSTD_compressContinue() detects that prior input has been overwritten when `src` buffer overlaps. - In which case, it will "discard" the relevant memory section from its history. - - Finish a frame with ZSTD_compressEnd(), which will write the last block(s) and optional checksum. - It's possible to use srcSize==0, in which case, it will write a final empty block to end the frame. - Without last block mark, frames will be considered unfinished (corrupted) by decoders. - - `ZSTD_CCtx` object can be re-used (ZSTD_compressBegin()) to compress some new frame. -*/ - -/*===== Buffer-less streaming compression functions =====*/ -ZSTDLIB_API size_t ZSTD_compressBegin(ZSTD_CCtx* cctx, int compressionLevel); -ZSTDLIB_API size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel); -ZSTDLIB_API size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */ -ZSTDLIB_API size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**< note: if pledgedSrcSize can be 0, indicating unknown size. if it is non-zero, it must be accurate. for 0 size frames, use compressBegin_advanced */ -ZSTDLIB_API size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: if pledgedSrcSize can be 0, indicating unknown size. if it is non-zero, it must be accurate. for 0 size frames, use compressBegin_advanced */ -ZSTDLIB_API size_t ZSTD_compressContinue(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); -ZSTDLIB_API size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); - - - -/*- - Buffer-less streaming decompression (synchronous mode) - - A ZSTD_DCtx object is required to track streaming operations. - Use ZSTD_createDCtx() / ZSTD_freeDCtx() to manage it. - A ZSTD_DCtx object can be re-used multiple times. - - First typical operation is to retrieve frame parameters, using ZSTD_getFrameParams(). - It fills a ZSTD_frameParams structure which provide important information to correctly decode the frame, - such as the minimum rolling buffer size to allocate to decompress data (`windowSize`), - and the dictionary ID used. - (Note : content size is optional, it may not be present. 0 means : content size unknown). - Note that these values could be wrong, either because of data malformation, or because an attacker is spoofing deliberate false information. - As a consequence, check that values remain within valid application range, especially `windowSize`, before allocation. - Each application can set its own limit, depending on local restrictions. For extended interoperability, it is recommended to support at least 8 MB. - Frame parameters are extracted from the beginning of the compressed frame. - Data fragment must be large enough to ensure successful decoding, typically `ZSTD_frameHeaderSize_max` bytes. - @result : 0 : successful decoding, the `ZSTD_frameParams` structure is correctly filled. - >0 : `srcSize` is too small, please provide at least @result bytes on next attempt. - errorCode, which can be tested using ZSTD_isError(). - - Start decompression, with ZSTD_decompressBegin() or ZSTD_decompressBegin_usingDict(). - Alternatively, you can copy a prepared context, using ZSTD_copyDCtx(). - - Then use ZSTD_nextSrcSizeToDecompress() and ZSTD_decompressContinue() alternatively. - ZSTD_nextSrcSizeToDecompress() tells how many bytes to provide as 'srcSize' to ZSTD_decompressContinue(). - ZSTD_decompressContinue() requires this _exact_ amount of bytes, or it will fail. - - @result of ZSTD_decompressContinue() is the number of bytes regenerated within 'dst' (necessarily <= dstCapacity). - It can be zero, which is not an error; it just means ZSTD_decompressContinue() has decoded some metadata item. - It can also be an error code, which can be tested with ZSTD_isError(). - - ZSTD_decompressContinue() needs previous data blocks during decompression, up to `windowSize`. - They should preferably be located contiguously, prior to current block. - Alternatively, a round buffer of sufficient size is also possible. Sufficient size is determined by frame parameters. - ZSTD_decompressContinue() is very sensitive to contiguity, - if 2 blocks don't follow each other, make sure that either the compressor breaks contiguity at the same place, - or that previous contiguous segment is large enough to properly handle maximum back-reference. - - A frame is fully decoded when ZSTD_nextSrcSizeToDecompress() returns zero. - Context can then be reset to start a new decompression. - - Note : it's possible to know if next input to present is a header or a block, using ZSTD_nextInputType(). - This information is not required to properly decode a frame. - - == Special case : skippable frames == - - Skippable frames allow integration of user-defined data into a flow of concatenated frames. - Skippable frames will be ignored (skipped) by a decompressor. The format of skippable frames is as follows : - a) Skippable frame ID - 4 Bytes, Little endian format, any value from 0x184D2A50 to 0x184D2A5F - b) Frame Size - 4 Bytes, Little endian format, unsigned 32-bits - c) Frame Content - any content (User Data) of length equal to Frame Size - For skippable frames ZSTD_decompressContinue() always returns 0. - For skippable frames ZSTD_getFrameParams() returns fparamsPtr->windowLog==0 what means that a frame is skippable. - Note : If fparamsPtr->frameContentSize==0, it is ambiguous: the frame might actually be a Zstd encoded frame with no content. - For purposes of decompression, it is valid in both cases to skip the frame using - ZSTD_findFrameCompressedSize to find its size in bytes. - It also returns Frame Size as fparamsPtr->frameContentSize. -*/ +/** + * ZSTD_getDictID_fromFrame() - returns the dictionary id stored in a zstd frame + * @src: Source buffer. It must be a zstd encoded frame. + * @srcSize: The size of the source buffer. It must be at least as large as the + * frame header. `ZSTD_frameHeaderSize_max` is always large enough. + * + * Return: The dictionary id required to decompress the frame stored within + * `src` or 0 if the dictionary id could not be decoded. It can return + * 0 if the frame does not require a dictionary, the dictionary id + * wasn't stored in the frame, `src` is not a zstd frame, or `srcSize` + * is too small. + */ +unsigned int ZSTD_getDictID_fromFrame(const void *src, size_t srcSize); +/** + * struct ZSTD_frameParams - zstd frame parameters stored in the frame header + * @frameContentSize: The frame content size, or 0 if not present. + * @windowSize: The window size, or 0 if the frame is a skippable frame. + * @dictID: The dictionary id, or 0 if not present. + * @checksumFlag: Whether a checksum was used. + */ typedef struct { unsigned long long frameContentSize; - unsigned windowSize; - unsigned dictID; - unsigned checksumFlag; + unsigned int windowSize; + unsigned int dictID; + unsigned int checksumFlag; } ZSTD_frameParams; -/*===== Buffer-less streaming decompression functions =====*/ -ZSTDLIB_API size_t ZSTD_getFrameParams(ZSTD_frameParams* fparamsPtr, const void* src, size_t srcSize); /**< doesn't consume input, see details below */ -ZSTDLIB_API size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx); -ZSTDLIB_API size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx* dctx, const void* dict, size_t dictSize); -ZSTDLIB_API void ZSTD_copyDCtx(ZSTD_DCtx* dctx, const ZSTD_DCtx* preparedDCtx); -ZSTDLIB_API size_t ZSTD_nextSrcSizeToDecompress(ZSTD_DCtx* dctx); -ZSTDLIB_API size_t ZSTD_decompressContinue(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); -typedef enum { ZSTDnit_frameHeader, ZSTDnit_blockHeader, ZSTDnit_block, ZSTDnit_lastBlock, ZSTDnit_checksum, ZSTDnit_skippableFrame } ZSTD_nextInputType_e; -ZSTDLIB_API ZSTD_nextInputType_e ZSTD_nextInputType(ZSTD_DCtx* dctx); - /** - Block functions + * ZSTD_getFrameParams() - extracts parameters from a zstd or skippable frame + * @fparamsPtr: On success the frame parameters are written here. + * @src: The source buffer. It must point to a zstd or skippable frame. + * @srcSize: The size of the source buffer. `ZSTD_frameHeaderSize_max` is + * always large enough to succeed. + * + * Return: 0 on success. If more data is required it returns how many bytes + * must be provided to make forward progress. Otherwise it returns + * an error, which can be checked using ZSTD_isError(). + */ +size_t ZSTD_getFrameParams(ZSTD_frameParams *fparamsPtr, const void *src, + size_t srcSize); - Block functions produce and decode raw zstd blocks, without frame metadata. - Frame metadata cost is typically ~18 bytes, which can be non-negligible for very small blocks (< 100 bytes). - User will have to take in charge required information to regenerate data, such as compressed and content sizes. +/*-***************************************************************************** + * Buffer-less and synchronous inner streaming functions + * + * This is an advanced API, giving full control over buffer management, for + * users which need direct control over memory. + * But it's also a complex one, with many restrictions (documented below). + * Prefer using normal streaming API for an easier experience + ******************************************************************************/ - A few rules to respect : - - Compressing and decompressing require a context structure - + Use ZSTD_createCCtx() and ZSTD_createDCtx() - - It is necessary to init context before starting - + compression : ZSTD_compressBegin() - + decompression : ZSTD_decompressBegin() - + variants _usingDict() are also allowed - + copyCCtx() and copyDCtx() work too - - Block size is limited, it must be <= ZSTD_getBlockSizeMax() - + If you need to compress more, cut data into multiple blocks - + Consider using the regular ZSTD_compress() instead, as frame metadata costs become negligible when source size is large. - - When a block is considered not compressible enough, ZSTD_compressBlock() result will be zero. - In which case, nothing is produced into `dst`. - + User must test for such outcome and deal directly with uncompressed data - + ZSTD_decompressBlock() doesn't accept uncompressed data as input !!! - + In case of multiple successive blocks, decoder must be informed of uncompressed block existence to follow proper history. - Use ZSTD_insertBlock() in such a case. -*/ +/*-***************************************************************************** + * Buffer-less streaming compression (synchronous mode) + * + * A ZSTD_CCtx object is required to track streaming operations. + * Use ZSTD_createCCtx() to create a context. + * ZSTD_CCtx object can be re-used multiple times within successive compression + * operations. + * + * Start by initializing a context. + * Use ZSTD_compressBegin(), or ZSTD_compressBegin_usingDict() for dictionary + * compression, + * or ZSTD_compressBegin_advanced(), for finer parameter control. + * It's also possible to duplicate a reference context which has already been + * initialized, using ZSTD_copyCCtx() + * + * Then, consume your input using ZSTD_compressContinue(). + * There are some important considerations to keep in mind when using this + * advanced function : + * - ZSTD_compressContinue() has no internal buffer. It uses externally provided + * buffer only. + * - Interface is synchronous : input is consumed entirely and produce 1+ + * (or more) compressed blocks. + * - Caller must ensure there is enough space in `dst` to store compressed data + * under worst case scenario. Worst case evaluation is provided by + * ZSTD_compressBound(). + * ZSTD_compressContinue() doesn't guarantee recover after a failed + * compression. + * - ZSTD_compressContinue() presumes prior input ***is still accessible and + * unmodified*** (up to maximum distance size, see WindowLog). + * It remembers all previous contiguous blocks, plus one separated memory + * segment (which can itself consists of multiple contiguous blocks) + * - ZSTD_compressContinue() detects that prior input has been overwritten when + * `src` buffer overlaps. In which case, it will "discard" the relevant memory + * section from its history. + * + * Finish a frame with ZSTD_compressEnd(), which will write the last block(s) + * and optional checksum. It's possible to use srcSize==0, in which case, it + * will write a final empty block to end the frame. Without last block mark, + * frames will be considered unfinished (corrupted) by decoders. + * + * `ZSTD_CCtx` object can be re-used (ZSTD_compressBegin()) to compress some new + * frame. + ******************************************************************************/ -#define ZSTD_BLOCKSIZE_ABSOLUTEMAX (128 * 1024) /* define, for static allocation */ +/*===== Buffer-less streaming compression functions =====*/ +size_t ZSTD_compressBegin(ZSTD_CCtx *cctx, int compressionLevel); +size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx *cctx, const void *dict, + size_t dictSize, int compressionLevel); +size_t ZSTD_compressBegin_advanced(ZSTD_CCtx *cctx, const void *dict, + size_t dictSize, ZSTD_parameters params, + unsigned long long pledgedSrcSize); +size_t ZSTD_copyCCtx(ZSTD_CCtx *cctx, const ZSTD_CCtx *preparedCCtx, + unsigned long long pledgedSrcSize); +size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx *cctx, const ZSTD_CDict *cdict, + unsigned long long pledgedSrcSize); +size_t ZSTD_compressContinue(ZSTD_CCtx *cctx, void *dst, size_t dstCapacity, + const void *src, size_t srcSize); +size_t ZSTD_compressEnd(ZSTD_CCtx *cctx, void *dst, size_t dstCapacity, + const void *src, size_t srcSize); + + + +/*-***************************************************************************** + * Buffer-less streaming decompression (synchronous mode) + * + * A ZSTD_DCtx object is required to track streaming operations. + * Use ZSTD_createDCtx() to create a context. + * A ZSTD_DCtx object can be re-used multiple times. + * + * First typical operation is to retrieve frame parameters, using + * ZSTD_getFrameParams(). It fills a ZSTD_frameParams structure which provide + * important information to correctly decode the frame, such as the minimum + * rolling buffer size to allocate to decompress data (`windowSize`), and the + * dictionary ID used. + * Note: content size is optional, it may not be present. 0 means unknown. + * Note that these values could be wrong, either because of data malformation, + * or because an attacker is spoofing deliberate false information. As a + * consequence, check that values remain within valid application range, + * especially `windowSize`, before allocation. Each application can set its own + * limit, depending on local restrictions. For extended interoperability, it is + * recommended to support at least 8 MB. + * Frame parameters are extracted from the beginning of the compressed frame. + * Data fragment must be large enough to ensure successful decoding, typically + * `ZSTD_frameHeaderSize_max` bytes. + * Result: 0: successful decoding, the `ZSTD_frameParams` structure is filled. + * >0: `srcSize` is too small, provide at least this many bytes. + * errorCode, which can be tested using ZSTD_isError(). + * + * Start decompression, with ZSTD_decompressBegin() or + * ZSTD_decompressBegin_usingDict(). Alternatively, you can copy a prepared + * context, using ZSTD_copyDCtx(). + * + * Then use ZSTD_nextSrcSizeToDecompress() and ZSTD_decompressContinue() + * alternatively. + * ZSTD_nextSrcSizeToDecompress() tells how many bytes to provide as 'srcSize' + * to ZSTD_decompressContinue(). + * ZSTD_decompressContinue() requires this _exact_ amount of bytes, or it will + * fail. + * + * The result of ZSTD_decompressContinue() is the number of bytes regenerated + * within 'dst' (necessarily <= dstCapacity). It can be zero, which is not an + * error; it just means ZSTD_decompressContinue() has decoded some metadata + * item. It can also be an error code, which can be tested with ZSTD_isError(). + * + * ZSTD_decompressContinue() needs previous data blocks during decompression, up + * to `windowSize`. They should preferably be located contiguously, prior to + * current block. Alternatively, a round buffer of sufficient size is also + * possible. Sufficient size is determined by frame parameters. + * ZSTD_decompressContinue() is very sensitive to contiguity, if 2 blocks don't + * follow each other, make sure that either the compressor breaks contiguity at + * the same place, or that previous contiguous segment is large enough to + * properly handle maximum back-reference. + * + * A frame is fully decoded when ZSTD_nextSrcSizeToDecompress() returns zero. + * Context can then be reset to start a new decompression. + * + * Note: it's possible to know if next input to present is a header or a block, + * using ZSTD_nextInputType(). This information is not required to properly + * decode a frame. + * + * == Special case: skippable frames == + * + * Skippable frames allow integration of user-defined data into a flow of + * concatenated frames. Skippable frames will be ignored (skipped) by a + * decompressor. The format of skippable frames is as follows: + * a) Skippable frame ID - 4 Bytes, Little endian format, any value from + * 0x184D2A50 to 0x184D2A5F + * b) Frame Size - 4 Bytes, Little endian format, unsigned 32-bits + * c) Frame Content - any content (User Data) of length equal to Frame Size + * For skippable frames ZSTD_decompressContinue() always returns 0. + * For skippable frames ZSTD_getFrameParams() returns fparamsPtr->windowLog==0 + * what means that a frame is skippable. + * Note: If fparamsPtr->frameContentSize==0, it is ambiguous: the frame might + * actually be a zstd encoded frame with no content. For purposes of + * decompression, it is valid in both cases to skip the frame using + * ZSTD_findFrameCompressedSize() to find its size in bytes. + * It also returns frame size as fparamsPtr->frameContentSize. + ******************************************************************************/ + +/*===== Buffer-less streaming decompression functions =====*/ +size_t ZSTD_decompressBegin(ZSTD_DCtx *dctx); +size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx *dctx, const void *dict, + size_t dictSize); +void ZSTD_copyDCtx(ZSTD_DCtx *dctx, const ZSTD_DCtx *preparedDCtx); +size_t ZSTD_nextSrcSizeToDecompress(ZSTD_DCtx *dctx); +size_t ZSTD_decompressContinue(ZSTD_DCtx *dctx, void *dst, size_t dstCapacity, + const void *src, size_t srcSize); +typedef enum { + ZSTDnit_frameHeader, + ZSTDnit_blockHeader, + ZSTDnit_block, + ZSTDnit_lastBlock, + ZSTDnit_checksum, + ZSTDnit_skippableFrame +} ZSTD_nextInputType_e; +ZSTD_nextInputType_e ZSTD_nextInputType(ZSTD_DCtx *dctx); + +/*-***************************************************************************** + * Block functions + * + * Block functions produce and decode raw zstd blocks, without frame metadata. + * Frame metadata cost is typically ~18 bytes, which can be non-negligible for + * very small blocks (< 100 bytes). User will have to take in charge required + * information to regenerate data, such as compressed and content sizes. + * + * A few rules to respect: + * - Compressing and decompressing require a context structure + * + Use ZSTD_createCCtx() and ZSTD_createDCtx() + * - It is necessary to init context before starting + * + compression : ZSTD_compressBegin() + * + decompression : ZSTD_decompressBegin() + * + variants _usingDict() are also allowed + * + copyCCtx() and copyDCtx() work too + * - Block size is limited, it must be <= ZSTD_getBlockSizeMax() + * + If you need to compress more, cut data into multiple blocks + * + Consider using the regular ZSTD_compress() instead, as frame metadata + * costs become negligible when source size is large. + * - When a block is considered not compressible enough, ZSTD_compressBlock() + * result will be zero. In which case, nothing is produced into `dst`. + * + User must test for such outcome and deal directly with uncompressed data + * + ZSTD_decompressBlock() doesn't accept uncompressed data as input!!! + * + In case of multiple successive blocks, decoder must be informed of + * uncompressed block existence to follow proper history. Use + * ZSTD_insertBlock() in such a case. + ******************************************************************************/ + +/* Define for static allocation */ +#define ZSTD_BLOCKSIZE_ABSOLUTEMAX (128 * 1024) /*===== Raw zstd block functions =====*/ -ZSTDLIB_API size_t ZSTD_getBlockSizeMax(ZSTD_CCtx* cctx); -ZSTDLIB_API size_t ZSTD_compressBlock (ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); -ZSTDLIB_API size_t ZSTD_decompressBlock(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize); -ZSTDLIB_API size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize); /**< insert block into `dctx` history. Useful for uncompressed blocks */ +size_t ZSTD_getBlockSizeMax(ZSTD_CCtx *cctx); +size_t ZSTD_compressBlock(ZSTD_CCtx *cctx, void *dst, size_t dstCapacity, + const void *src, size_t srcSize); +size_t ZSTD_decompressBlock(ZSTD_DCtx *dctx, void *dst, size_t dstCapacity, + const void *src, size_t srcSize); +size_t ZSTD_insertBlock(ZSTD_DCtx *dctx, const void *blockStart, + size_t blockSize); - -#endif /* ZSTD_H_235446 */ +#endif /* ZSTD_H */ From 4b987ad8ce9fe0deb90853eace96a9fe9ef4622e Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Mon, 10 Apr 2017 17:50:44 -0700 Subject: [PATCH 160/305] Introduce ZSTD_initCStream_internal() This is now the regroup point for ZSTD_initCStream*() functions ZSTD_initCStream_advanced() now properly checks for parameters validity. Also : added usage inside zstd_compress.c Needs ZSTD_DEBUG=1 macro to be triggered. Will be triggered by default from `tests` directory --- doc/zstd_manual.html | 14 +++++++++++--- lib/compress/zstd_compress.c | 30 ++++++++++++++++++++++++------ tests/Makefile | 4 +++- tests/zstreamtest.c | 10 ++++++++++ 4 files changed, 48 insertions(+), 10 deletions(-) diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index a66f0a49c..a5b66f7d3 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -497,14 +497,22 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v

    Advanced streaming functions

    
     
     

    Advanced Streaming compression functions

    ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem);
    +size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);   /**< size of CStream is variable, depending primarily on compression level */
     size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize);   /**< pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
     size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
     size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize,
                                                  ZSTD_parameters params, unsigned long long pledgedSrcSize);  /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */
     size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict);  /**< note : cdict will just be referenced, and must outlive compression session */
    -size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize);  /**< re-use compression parameters from previous init; skip dictionary loading stage; zcs must be init at least once before. note: pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
    -size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);
     

    +
    size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize);
    +

    start a new compression job, using same parameters from previous job. + This is typically useful to skip dictionary loading stage, since it will re-use it in-place.. + Note that zcs must be init at least once before using ZSTD_resetCStream(). + pledgedSrcSize==0 means "srcSize unknown". + If pledgedSrcSize > 0, its value must be correct, as it will be written in header, and controlled at the end. + @return : 0, or an error code (which can be tested using ZSTD_isError()) +


    +

    Advanced Streaming decompression functions

    typedef enum { DStream_p_maxWindowSize } ZSTD_DStreamParameter_e;
     ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem);
     size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
    @@ -553,7 +561,7 @@ size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds);
     size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel);
     size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */
     size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**<  note: if pledgedSrcSize can be 0, indicating unknown size.  if it is non-zero, it must be accurate.  for 0 size frames, use compressBegin_advanced */
    -size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: if pledgedSrcSize can be 0, indicating unknown size.  if it is non-zero, it must be accurate.  for 0 size frames, use compressBegin_advanced */
    +size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: fail if cdict==NULL. pledgedSrcSize can be 0, indicating unknown size.  For 0 size frames, use compressBegin_advanced */
     size_t ZSTD_compressContinue(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
     size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
     

    diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 4ad978a4b..c0ccf83cd 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -31,7 +31,14 @@ typedef enum { ZSTDcs_created=0, ZSTDcs_init, ZSTDcs_ongoing, ZSTDcs_ending } ZS /*-************************************* * Helper functions ***************************************/ +#if defined(ZSTD_DEBUG) && (ZSTD_DEBUG==1) +# include +#else +# define assert(condition) ((void)0) +#endif + #define ZSTD_STATIC_ASSERT(c) { enum { ZSTD_static_assert = 1/(int)(!!(c)) }; } + size_t ZSTD_compressBound(size_t srcSize) { size_t const lowLimit = 256 KB; size_t const margin = (srcSize < lowLimit) ? (lowLimit-srcSize) >> 12 : 0; /* from 64 to 0 */ @@ -3031,10 +3038,13 @@ size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize) return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } -size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, - const void* dict, size_t dictSize, - ZSTD_parameters params, unsigned long long pledgedSrcSize) +/* ZSTD_initCStream_internal() : + * params are supposed validated at this stage */ +static size_t ZSTD_initCStream_internal(ZSTD_CStream* zcs, + const void* dict, size_t dictSize, + ZSTD_parameters params, unsigned long long pledgedSrcSize) { + assert(!ZSTD_isError(ZSTD_checkCParams(params.cParams))); /* allocate buffers */ { size_t const neededInBuffSize = (size_t)1 << params.cParams.windowLog; if (zcs->inBuffSize < neededInBuffSize) { @@ -3068,11 +3078,19 @@ size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } +size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, + const void* dict, size_t dictSize, + ZSTD_parameters params, unsigned long long pledgedSrcSize) +{ + CHECK_F( ZSTD_checkCParams(params.cParams) ); + return ZSTD_initCStream_internal(zcs, dict, dictSize, params, pledgedSrcSize); +} + /* note : cdict must outlive compression session */ size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) { ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict); - size_t const initError = ZSTD_initCStream_advanced(zcs, NULL, 0, params, 0); + size_t const initError = ZSTD_initCStream_internal(zcs, NULL, 0, params, 0); zcs->cdict = cdict; if (ZSTD_isError(initError)) return initError; return ZSTD_resetCStream_internal(zcs, 0); @@ -3081,14 +3099,14 @@ size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel) { ZSTD_parameters const params = ZSTD_getParams(compressionLevel, 0, dictSize); - return ZSTD_initCStream_advanced(zcs, dict, dictSize, params, 0); + return ZSTD_initCStream_internal(zcs, dict, dictSize, params, 0); } size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize) { ZSTD_parameters params = ZSTD_getParams(compressionLevel, pledgedSrcSize, 0); if (pledgedSrcSize) params.fParams.contentSizeFlag = 1; - return ZSTD_initCStream_advanced(zcs, NULL, 0, params, pledgedSrcSize); + return ZSTD_initCStream_internal(zcs, NULL, 0, params, pledgedSrcSize); } size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel) diff --git a/tests/Makefile b/tests/Makefile index 809c90df5..70ef51878 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -26,7 +26,9 @@ PYTHON ?= python3 TESTARTEFACT := versionsTest namespaceTest -CPPFLAGS+= -I$(ZSTDDIR) -I$(ZSTDDIR)/common -I$(ZSTDDIR)/compress -I$(ZSTDDIR)/dictBuilder -I$(ZSTDDIR)/deprecated -I$(PRGDIR) +CPPFLAGS+= -I$(ZSTDDIR) -I$(ZSTDDIR)/common -I$(ZSTDDIR)/compress \ + -I$(ZSTDDIR)/dictBuilder -I$(ZSTDDIR)/deprecated -I$(PRGDIR) \ + -DZSTD_DEBUG=1 CFLAGS ?= -O3 CFLAGS += -g -Wall -Wextra -Wcast-qual -Wcast-align -Wshadow \ -Wstrict-aliasing=1 -Wswitch-enum -Wdeclaration-after-statement \ diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index 270c0221b..6f26ea555 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -207,6 +207,16 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo DISPLAYLEVEL(3, "OK (%u bytes) \n", (U32)s); } + /* Attempt bad compression parameters */ + DISPLAYLEVEL(3, "test%3i : use bad compression parameters : ", testNb++); + { size_t r; + ZSTD_parameters params = ZSTD_getParams(1, 0, 0); + params.cParams.searchLength = 2; + r = ZSTD_initCStream_advanced(zc, NULL, 0, params, 0); + if (!ZSTD_isError(r)) goto _output_error; + DISPLAYLEVEL(3, "init error : %s \n", ZSTD_getErrorName(r)); + } + /* skippable frame test */ DISPLAYLEVEL(3, "test%3i : decompress skippable frame : ", testNb++); ZSTD_initDStream_usingDict(zd, CNBuffer, 128 KB); From e88034fe2639f2d83470df861e1ce95676113236 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Mon, 10 Apr 2017 22:24:02 -0700 Subject: [PATCH 161/305] simplified ZSTD_initCStream*() flow all variants converge towards ZSTD_initCStream_stage2() --- lib/compress/zstd_compress.c | 60 +++++++++++++++++++++--------------- 1 file changed, 36 insertions(+), 24 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index c0ccf83cd..a4a3ee084 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -3034,17 +3034,18 @@ size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize) { zcs->params.fParams.contentSizeFlag = (pledgedSrcSize > 0); - return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } /* ZSTD_initCStream_internal() : - * params are supposed validated at this stage */ -static size_t ZSTD_initCStream_internal(ZSTD_CStream* zcs, - const void* dict, size_t dictSize, - ZSTD_parameters params, unsigned long long pledgedSrcSize) + * params are supposed validated at this stage + * and zcs->cdict is supposed to be correct */ +static size_t ZSTD_initCStream_stage2(ZSTD_CStream* zcs, + const ZSTD_parameters params, + unsigned long long pledgedSrcSize) { assert(!ZSTD_isError(ZSTD_checkCParams(params.cParams))); + /* allocate buffers */ { size_t const neededInBuffSize = (size_t)1 << params.cParams.windowLog; if (zcs->inBuffSize < neededInBuffSize) { @@ -3065,19 +3066,39 @@ static size_t ZSTD_initCStream_internal(ZSTD_CStream* zcs, zcs->outBuffSize = outBuffSize; } - if (dict && dictSize >= 8) { - ZSTD_freeCDict(zcs->cdictLocal); - zcs->cdictLocal = ZSTD_createCDict_advanced(dict, dictSize, 0, params, zcs->customMem); - if (zcs->cdictLocal == NULL) return ERROR(memory_allocation); - zcs->cdict = zcs->cdictLocal; - } else zcs->cdict = NULL; - zcs->checksum = params.fParams.checksumFlag > 0; zcs->params = params; return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } +/* note : cdict must outlive compression session */ +size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) +{ + if (!cdict) return ERROR(GENERIC); /* cannot handle NULL cdict (does not know what to do) */ + { ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict); + zcs->cdict = cdict; + return ZSTD_initCStream_stage2(zcs, params, 0); + } +} + +static size_t ZSTD_initCStream_internal(ZSTD_CStream* zcs, + const void* dict, size_t dictSize, + ZSTD_parameters params, unsigned long long pledgedSrcSize) +{ + assert(!ZSTD_isError(ZSTD_checkCParams(params.cParams))); + zcs->cdict = NULL; + + if (dict && dictSize >= 8) { + ZSTD_freeCDict(zcs->cdictLocal); + zcs->cdictLocal = ZSTD_createCDict_advanced(dict, dictSize, 0 /* copy */, params, zcs->customMem); + if (zcs->cdictLocal == NULL) return ERROR(memory_allocation); + zcs->cdict = zcs->cdictLocal; + } + + return ZSTD_initCStream_stage2(zcs, params, pledgedSrcSize); +} + size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize) @@ -3086,16 +3107,6 @@ size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, return ZSTD_initCStream_internal(zcs, dict, dictSize, params, pledgedSrcSize); } -/* note : cdict must outlive compression session */ -size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) -{ - ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict); - size_t const initError = ZSTD_initCStream_internal(zcs, NULL, 0, params, 0); - zcs->cdict = cdict; - if (ZSTD_isError(initError)) return initError; - return ZSTD_resetCStream_internal(zcs, 0); -} - size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel) { ZSTD_parameters const params = ZSTD_getParams(compressionLevel, 0, dictSize); @@ -3105,13 +3116,14 @@ size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t di size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize) { ZSTD_parameters params = ZSTD_getParams(compressionLevel, pledgedSrcSize, 0); - if (pledgedSrcSize) params.fParams.contentSizeFlag = 1; + params.fParams.contentSizeFlag = (pledgedSrcSize>0); return ZSTD_initCStream_internal(zcs, NULL, 0, params, pledgedSrcSize); } size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel) { - return ZSTD_initCStream_usingDict(zcs, NULL, 0, compressionLevel); + ZSTD_parameters const params = ZSTD_getParams(compressionLevel, 0, 0); + return ZSTD_initCStream_internal(zcs, NULL, 0, params, 0); } size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs) From ab9162ebb49ff00e2df0431e3b2a9de8510604b6 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 11 Apr 2017 10:46:20 -0700 Subject: [PATCH 162/305] simplified call graph by calling ZSTD_compressBegin_internal() instead of ZSTD_compressBegin_advanced() --- lib/compress/zstd_compress.c | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index a4a3ee084..62c1c6cdd 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -2690,6 +2690,7 @@ static size_t ZSTD_compressBegin_internal(ZSTD_CCtx* cctx, ZSTD_parameters params, U64 pledgedSrcSize) { ZSTD_compResetPolicy_e const crp = dictSize ? ZSTDcrp_fullReset : ZSTDcrp_continue; + assert(!ZSTD_isError(ZSTD_checkCParams(params.cParams))); CHECK_F(ZSTD_resetCCtx_advanced(cctx, params, pledgedSrcSize, crp)); return ZSTD_compress_insertDictionary(cctx, dict, dictSize); } @@ -2911,7 +2912,7 @@ size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, u else { ZSTD_parameters params = cdict->refContext->params; params.fParams.contentSizeFlag = (pledgedSrcSize > 0); - CHECK_F(ZSTD_compressBegin_advanced(cctx, NULL, 0, params, pledgedSrcSize)); + CHECK_F(ZSTD_compressBegin_internal(cctx, NULL, 0, params, pledgedSrcSize)); } return 0; } @@ -3017,7 +3018,7 @@ static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, unsigned long long p if (zcs->inBuffSize==0) return ERROR(stage_wrong); /* zcs has not been init at least once => can't reset */ if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict(zcs->cctx, zcs->cdict, pledgedSrcSize)) - else CHECK_F(ZSTD_compressBegin_advanced(zcs->cctx, NULL, 0, zcs->params, pledgedSrcSize)); + else CHECK_F(ZSTD_compressBegin_internal(zcs->cctx, NULL, 0, zcs->params, pledgedSrcSize)); zcs->inToCompress = 0; zcs->inBuffPos = 0; From b4dd3378f10b7ba83077826129f59805176ef728 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 11 Apr 2017 11:35:36 -0700 Subject: [PATCH 163/305] Add BtrFS patch for 4.10 kernel --- contrib/linux-kernel/README.md | 10 +- contrib/linux-kernel/btrfs.diff | 633 +++++++++++++++++++++++++++ contrib/linux-kernel/fs/btrfs/zstd.c | 129 +++--- 3 files changed, 703 insertions(+), 69 deletions(-) create mode 100644 contrib/linux-kernel/btrfs.diff diff --git a/contrib/linux-kernel/README.md b/contrib/linux-kernel/README.md index 11938ad62..1cc74cca6 100644 --- a/contrib/linux-kernel/README.md +++ b/contrib/linux-kernel/README.md @@ -1,8 +1,7 @@ # Linux Kernel Patch There are two pieces, the `zstd_compress` and `zstd_decompress` kernel modules, and the BtrFS patch. -The patches are based off of the linux kernel version 4.9. -The BtrFS patch is not present in its entirety yet. +The patches are based off of the linux kernel master branch (version 4.10). ## Zstd Kernel modules @@ -21,6 +20,7 @@ The BtrFS patch is not present in its entirety yet. ## BtrFS -* `fs/btrfs/zstd.c` is provided. -* Some more glue is required to integrate it with BtrFS, but I haven't included the patches yet. - In the meantime see https://github.com/terrelln/linux/commit/1914f7d4ca6c539369c84853eafa4ac104883047 if you're interested. +* The patch is located in `btrfs.diff`. +* Additionally `fs/btrfs/zstd.c` is provided as a source for convenience. +* The patch seems to be working, it doesn't crash the kernel, and compresses at speeds and ratios athat are expected. + It can still use some more testing for fringe features, like printing options. diff --git a/contrib/linux-kernel/btrfs.diff b/contrib/linux-kernel/btrfs.diff new file mode 100644 index 000000000..b0f8b924b --- /dev/null +++ b/contrib/linux-kernel/btrfs.diff @@ -0,0 +1,633 @@ +diff --git a/fs/btrfs/Kconfig b/fs/btrfs/Kconfig +index 80e9c18..a26c63b 100644 +--- a/fs/btrfs/Kconfig ++++ b/fs/btrfs/Kconfig +@@ -6,6 +6,8 @@ config BTRFS_FS + select ZLIB_DEFLATE + select LZO_COMPRESS + select LZO_DECOMPRESS ++ select ZSTD_COMPRESS ++ select ZSTD_DECOMPRESS + select RAID6_PQ + select XOR_BLOCKS + select SRCU +diff --git a/fs/btrfs/Makefile b/fs/btrfs/Makefile +index 128ce17..962a95a 100644 +--- a/fs/btrfs/Makefile ++++ b/fs/btrfs/Makefile +@@ -6,7 +6,7 @@ btrfs-y += super.o ctree.o extent-tree.o print-tree.o root-tree.o dir-item.o \ + transaction.o inode.o file.o tree-defrag.o \ + extent_map.o sysfs.o struct-funcs.o xattr.o ordered-data.o \ + extent_io.o volumes.o async-thread.o ioctl.o locking.o orphan.o \ +- export.o tree-log.o free-space-cache.o zlib.o lzo.o \ ++ export.o tree-log.o free-space-cache.o zlib.o lzo.o zstd.o \ + compression.o delayed-ref.o relocation.o delayed-inode.o scrub.o \ + reada.o backref.o ulist.o qgroup.o send.o dev-replace.o raid56.o \ + uuid-tree.o props.o hash.o free-space-tree.o +diff --git a/fs/btrfs/compression.c b/fs/btrfs/compression.c +index c7721a6..66d4ced 100644 +--- a/fs/btrfs/compression.c ++++ b/fs/btrfs/compression.c +@@ -761,6 +761,7 @@ static struct { + static const struct btrfs_compress_op * const btrfs_compress_op[] = { + &btrfs_zlib_compress, + &btrfs_lzo_compress, ++ &btrfs_zstd_compress, + }; + + void __init btrfs_init_compress(void) +diff --git a/fs/btrfs/compression.h b/fs/btrfs/compression.h +index 39ec43a..d99fc21 100644 +--- a/fs/btrfs/compression.h ++++ b/fs/btrfs/compression.h +@@ -60,8 +60,9 @@ enum btrfs_compression_type { + BTRFS_COMPRESS_NONE = 0, + BTRFS_COMPRESS_ZLIB = 1, + BTRFS_COMPRESS_LZO = 2, +- BTRFS_COMPRESS_TYPES = 2, +- BTRFS_COMPRESS_LAST = 3, ++ BTRFS_COMPRESS_ZSTD = 3, ++ BTRFS_COMPRESS_TYPES = 3, ++ BTRFS_COMPRESS_LAST = 4, + }; + + struct btrfs_compress_op { +@@ -92,5 +93,6 @@ struct btrfs_compress_op { + + extern const struct btrfs_compress_op btrfs_zlib_compress; + extern const struct btrfs_compress_op btrfs_lzo_compress; ++extern const struct btrfs_compress_op btrfs_zstd_compress; + + #endif +diff --git a/fs/btrfs/ctree.h b/fs/btrfs/ctree.h +index 29b7fc2..878b23b9 100644 +--- a/fs/btrfs/ctree.h ++++ b/fs/btrfs/ctree.h +@@ -270,6 +270,7 @@ struct btrfs_super_block { + BTRFS_FEATURE_INCOMPAT_MIXED_GROUPS | \ + BTRFS_FEATURE_INCOMPAT_BIG_METADATA | \ + BTRFS_FEATURE_INCOMPAT_COMPRESS_LZO | \ ++ BTRFS_FEATURE_INCOMPAT_COMPRESS_ZSTD | \ + BTRFS_FEATURE_INCOMPAT_RAID56 | \ + BTRFS_FEATURE_INCOMPAT_EXTENDED_IREF | \ + BTRFS_FEATURE_INCOMPAT_SKINNY_METADATA | \ +diff --git a/fs/btrfs/disk-io.c b/fs/btrfs/disk-io.c +index 08b74da..0c43e4e 100644 +--- a/fs/btrfs/disk-io.c ++++ b/fs/btrfs/disk-io.c +@@ -2853,6 +2853,8 @@ int open_ctree(struct super_block *sb, + features |= BTRFS_FEATURE_INCOMPAT_MIXED_BACKREF; + if (fs_info->compress_type == BTRFS_COMPRESS_LZO) + features |= BTRFS_FEATURE_INCOMPAT_COMPRESS_LZO; ++ else if (tree_root->fs_info->compress_type == BTRFS_COMPRESS_ZSTD) ++ features |= BTRFS_FEATURE_INCOMPAT_COMPRESS_ZSTD; + + if (features & BTRFS_FEATURE_INCOMPAT_SKINNY_METADATA) + btrfs_info(fs_info, "has skinny extents"); +diff --git a/fs/btrfs/ioctl.c b/fs/btrfs/ioctl.c +index dabfc7a..d8ea727 100644 +--- a/fs/btrfs/ioctl.c ++++ b/fs/btrfs/ioctl.c +@@ -327,8 +327,10 @@ static int btrfs_ioctl_setflags(struct file *file, void __user *arg) + + if (fs_info->compress_type == BTRFS_COMPRESS_LZO) + comp = "lzo"; +- else ++ else if (fs_info->compress_type == BTRFS_COMPRESS_ZLIB) + comp = "zlib"; ++ else ++ comp = "zstd"; + ret = btrfs_set_prop(inode, "btrfs.compression", + comp, strlen(comp), 0); + if (ret) +@@ -1463,6 +1465,8 @@ int btrfs_defrag_file(struct inode *inode, struct file *file, + + if (range->compress_type == BTRFS_COMPRESS_LZO) { + btrfs_set_fs_incompat(fs_info, COMPRESS_LZO); ++ } else if (range->compress_type == BTRFS_COMPRESS_ZSTD) { ++ btrfs_set_fs_incompat(fs_info, COMPRESS_ZSTD); + } + + ret = defrag_count; +diff --git a/fs/btrfs/props.c b/fs/btrfs/props.c +index d6cb155..162105f 100644 +--- a/fs/btrfs/props.c ++++ b/fs/btrfs/props.c +@@ -383,6 +383,8 @@ static int prop_compression_validate(const char *value, size_t len) + return 0; + else if (!strncmp("zlib", value, len)) + return 0; ++ else if (!strncmp("zstd", value, len)) ++ return 0; + + return -EINVAL; + } +@@ -405,6 +407,8 @@ static int prop_compression_apply(struct inode *inode, + type = BTRFS_COMPRESS_LZO; + else if (!strncmp("zlib", value, len)) + type = BTRFS_COMPRESS_ZLIB; ++ else if (!strncmp("zstd", value, len)) ++ type = BTRFS_COMPRESS_ZSTD; + else + return -EINVAL; + +@@ -422,6 +426,8 @@ static const char *prop_compression_extract(struct inode *inode) + return "zlib"; + case BTRFS_COMPRESS_LZO: + return "lzo"; ++ case BTRFS_COMPRESS_ZSTD: ++ return "zstd"; + } + + return NULL; +diff --git a/fs/btrfs/super.c b/fs/btrfs/super.c +index da687dc..b064456 100644 +--- a/fs/btrfs/super.c ++++ b/fs/btrfs/super.c +@@ -513,6 +513,14 @@ int btrfs_parse_options(struct btrfs_fs_info *info, char *options, + btrfs_clear_opt(info->mount_opt, NODATASUM); + btrfs_set_fs_incompat(info, COMPRESS_LZO); + no_compress = 0; ++ } else if (strcmp(args[0].from, "zstd") == 0) { ++ compress_type = "zstd"; ++ info->compress_type = BTRFS_COMPRESS_ZSTD; ++ btrfs_set_opt(info->mount_opt, COMPRESS); ++ btrfs_clear_opt(info->mount_opt, NODATACOW); ++ btrfs_clear_opt(info->mount_opt, NODATASUM); ++ btrfs_set_fs_incompat(info, COMPRESS_ZSTD); ++ no_compress = 0; + } else if (strncmp(args[0].from, "no", 2) == 0) { + compress_type = "no"; + btrfs_clear_opt(info->mount_opt, COMPRESS); +@@ -1230,8 +1238,10 @@ static int btrfs_show_options(struct seq_file *seq, struct dentry *dentry) + if (btrfs_test_opt(info, COMPRESS)) { + if (info->compress_type == BTRFS_COMPRESS_ZLIB) + compress_type = "zlib"; +- else ++ else if (info->compress_type == BTRFS_COMPRESS_LZO) + compress_type = "lzo"; ++ else ++ compress_type = "zstd"; + if (btrfs_test_opt(info, FORCE_COMPRESS)) + seq_printf(seq, ",compress-force=%s", compress_type); + else +diff --git a/fs/btrfs/sysfs.c b/fs/btrfs/sysfs.c +index 1f157fb..b0dec90 100644 +--- a/fs/btrfs/sysfs.c ++++ b/fs/btrfs/sysfs.c +@@ -200,6 +200,7 @@ BTRFS_FEAT_ATTR_INCOMPAT(mixed_backref, MIXED_BACKREF); + BTRFS_FEAT_ATTR_INCOMPAT(default_subvol, DEFAULT_SUBVOL); + BTRFS_FEAT_ATTR_INCOMPAT(mixed_groups, MIXED_GROUPS); + BTRFS_FEAT_ATTR_INCOMPAT(compress_lzo, COMPRESS_LZO); ++BTRFS_FEAT_ATTR_INCOMPAT(compress_zstd, COMPRESS_ZSTD); + BTRFS_FEAT_ATTR_INCOMPAT(big_metadata, BIG_METADATA); + BTRFS_FEAT_ATTR_INCOMPAT(extended_iref, EXTENDED_IREF); + BTRFS_FEAT_ATTR_INCOMPAT(raid56, RAID56); +@@ -212,6 +213,7 @@ static struct attribute *btrfs_supported_feature_attrs[] = { + BTRFS_FEAT_ATTR_PTR(default_subvol), + BTRFS_FEAT_ATTR_PTR(mixed_groups), + BTRFS_FEAT_ATTR_PTR(compress_lzo), ++ BTRFS_FEAT_ATTR_PTR(compress_zstd), + BTRFS_FEAT_ATTR_PTR(big_metadata), + BTRFS_FEAT_ATTR_PTR(extended_iref), + BTRFS_FEAT_ATTR_PTR(raid56), +diff --git a/fs/btrfs/zstd.c b/fs/btrfs/zstd.c +new file mode 100644 +index 0000000..b7f319e +--- /dev/null ++++ b/fs/btrfs/zstd.c +@@ -0,0 +1,415 @@ ++#include ++#include ++#include ++#include ++#include ++#include ++#include ++#include ++#include ++#include "compression.h" ++ ++#define ZSTD_BTRFS_MAX_WINDOWLOG 17 ++#define ZSTD_BTRFS_MAX_INPUT (1 << ZSTD_BTRFS_MAX_WINDOWLOG) ++ ++static ZSTD_parameters zstd_get_btrfs_parameters(size_t src_len) ++{ ++ ZSTD_parameters params = ZSTD_getParams(3, src_len, 0); ++ ++ if (params.cParams.windowLog > ZSTD_BTRFS_MAX_WINDOWLOG) ++ params.cParams.windowLog = ZSTD_BTRFS_MAX_WINDOWLOG; ++ WARN_ON(src_len > ZSTD_BTRFS_MAX_INPUT); ++ return params; ++} ++ ++struct workspace { ++ void *mem; ++ size_t size; ++ char *buf; ++ struct list_head list; ++}; ++ ++static void zstd_free_workspace(struct list_head *ws) ++{ ++ struct workspace *workspace = list_entry(ws, struct workspace, list); ++ ++ vfree(workspace->mem); ++ kfree(workspace->buf); ++ kfree(workspace); ++} ++ ++static struct list_head *zstd_alloc_workspace(void) ++{ ++ ZSTD_parameters params = ++ zstd_get_btrfs_parameters(ZSTD_BTRFS_MAX_INPUT); ++ struct workspace *workspace; ++ ++ workspace = kzalloc(sizeof(*workspace), GFP_NOFS); ++ if (!workspace) ++ return ERR_PTR(-ENOMEM); ++ ++ workspace->size = max_t(size_t, ++ ZSTD_CStreamWorkspaceBound(params.cParams), ++ ZSTD_DStreamWorkspaceBound(ZSTD_BTRFS_MAX_INPUT)); ++ workspace->mem = vmalloc(workspace->size); ++ workspace->buf = kmalloc(PAGE_SIZE, GFP_NOFS); ++ if (!workspace->mem || !workspace->buf) ++ goto fail; ++ ++ INIT_LIST_HEAD(&workspace->list); ++ ++ return &workspace->list; ++fail: ++ zstd_free_workspace(&workspace->list); ++ return ERR_PTR(-ENOMEM); ++} ++ ++static int zstd_compress_pages(struct list_head *ws, ++ struct address_space *mapping, ++ u64 start, ++ struct page **pages, ++ unsigned long *out_pages, ++ unsigned long *total_in, ++ unsigned long *total_out) ++{ ++ struct workspace *workspace = list_entry(ws, struct workspace, list); ++ ZSTD_CStream *stream; ++ int ret = 0; ++ int nr_pages = 0; ++ struct page *in_page = NULL; /* The current page to read */ ++ struct page *out_page = NULL; /* The current page to write to */ ++ ZSTD_inBuffer in_buf = { NULL, 0, 0 }; ++ ZSTD_outBuffer out_buf = { NULL, 0, 0 }; ++ unsigned long tot_in = 0; ++ unsigned long tot_out = 0; ++ unsigned long len = *total_out; ++ const unsigned long nr_dest_pages = *out_pages; ++ unsigned long max_out = nr_dest_pages * PAGE_SIZE; ++ ZSTD_parameters params = zstd_get_btrfs_parameters(len); ++ ++ *out_pages = 0; ++ *total_out = 0; ++ *total_in = 0; ++ ++ /* Initialize the stream */ ++ stream = ZSTD_createCStream(params, len, workspace->mem, ++ workspace->size); ++ if (!stream) { ++ pr_warn("BTRFS: ZSTD_createStream failed\n"); ++ ret = -EIO; ++ goto out; ++ } ++ ++ /* map in the first page of input data */ ++ in_page = find_get_page(mapping, start >> PAGE_SHIFT); ++ in_buf.src = kmap(in_page); ++ in_buf.pos = 0; ++ in_buf.size = min_t(size_t, len, PAGE_SIZE); ++ ++ ++ /* Allocate and map in the output buffer */ ++ out_page = alloc_page(GFP_NOFS | __GFP_HIGHMEM); ++ if (out_page == NULL) { ++ ret = -ENOMEM; ++ goto out; ++ } ++ pages[nr_pages++] = out_page; ++ out_buf.dst = kmap(out_page); ++ out_buf.pos = 0; ++ out_buf.size = min_t(size_t, max_out, PAGE_SIZE); ++ ++ while (1) { ++ size_t ret2; ++ ++ ret2 = ZSTD_compressStream(stream, &out_buf, &in_buf); ++ if (ZSTD_isError(ret2)) { ++ pr_debug("BTRFS: ZSTD_compressStream returned %d\n", ++ ZSTD_getErrorCode(ret2)); ++ ret = -EIO; ++ goto out; ++ } ++ ++ /* Check to see if we are making it bigger */ ++ if (tot_in + in_buf.pos > 8192 && ++ tot_in + in_buf.pos < ++ tot_out + out_buf.pos) { ++ ret = -E2BIG; ++ goto out; ++ } ++ ++ /* We've reached the end of our output range */ ++ if (out_buf.pos >= max_out) { ++ tot_out += out_buf.pos; ++ ret = -E2BIG; ++ goto out; ++ } ++ ++ /* Check if we need more output space */ ++ if (out_buf.pos == out_buf.size) { ++ tot_out += PAGE_SIZE; ++ max_out -= PAGE_SIZE; ++ kunmap(out_page); ++ if (nr_pages == nr_dest_pages) { ++ out_page = NULL; ++ ret = -E2BIG; ++ goto out; ++ } ++ out_page = alloc_page(GFP_NOFS | __GFP_HIGHMEM); ++ if (out_page == NULL) { ++ ret = -ENOMEM; ++ goto out; ++ } ++ pages[nr_pages++] = out_page; ++ out_buf.dst = kmap(out_page); ++ out_buf.pos = 0; ++ out_buf.size = min_t(size_t, max_out, PAGE_SIZE); ++ } ++ ++ /* We've reached the end of the input */ ++ if (in_buf.pos >= len) { ++ tot_in += in_buf.pos; ++ break; ++ } ++ ++ /* Check if we need more input */ ++ if (in_buf.pos == in_buf.size) { ++ tot_in += PAGE_SIZE; ++ kunmap(in_page); ++ put_page(in_page); ++ ++ start += PAGE_SIZE; ++ len -= PAGE_SIZE; ++ in_page = find_get_page(mapping, start >> PAGE_SHIFT); ++ in_buf.src = kmap(in_page); ++ in_buf.pos = 0; ++ in_buf.size = min_t(size_t, len, PAGE_SIZE); ++ } ++ } ++ while (1) { ++ size_t ret2; ++ ++ ret2 = ZSTD_endStream(stream, &out_buf); ++ if (ZSTD_isError(ret2)) { ++ pr_debug("BTRFS: ZSTD_endStream returned %d\n", ++ ZSTD_getErrorCode(ret2)); ++ ret = -EIO; ++ goto out; ++ } ++ if (ret2 == 0) { ++ tot_out += out_buf.pos; ++ break; ++ } ++ if (out_buf.pos >= max_out) { ++ tot_out += out_buf.pos; ++ ret = -E2BIG; ++ goto out; ++ } ++ ++ tot_out += PAGE_SIZE; ++ max_out -= PAGE_SIZE; ++ kunmap(out_page); ++ if (nr_pages == nr_dest_pages) { ++ out_page = NULL; ++ ret = -E2BIG; ++ goto out; ++ } ++ out_page = alloc_page(GFP_NOFS | __GFP_HIGHMEM); ++ if (out_page == NULL) { ++ ret = -ENOMEM; ++ goto out; ++ } ++ pages[nr_pages++] = out_page; ++ out_buf.dst = kmap(out_page); ++ out_buf.pos = 0; ++ out_buf.size = min_t(size_t, max_out, PAGE_SIZE); ++ } ++ ++ if (tot_out >= tot_in) { ++ ret = -E2BIG; ++ goto out; ++ } ++ ++ ret = 0; ++ *total_in = tot_in; ++ *total_out = tot_out; ++out: ++ *out_pages = nr_pages; ++ /* Cleanup */ ++ if (in_page) { ++ kunmap(in_page); ++ put_page(in_page); ++ } ++ if (out_page) ++ kunmap(out_page); ++ return ret; ++} ++ ++static int zstd_decompress_bio(struct list_head *ws, struct page **pages_in, ++ u64 disk_start, ++ struct bio *orig_bio, ++ size_t srclen) ++{ ++ struct workspace *workspace = list_entry(ws, struct workspace, list); ++ ZSTD_DStream *stream; ++ int ret = 0; ++ unsigned long page_in_index = 0; ++ unsigned long total_pages_in = DIV_ROUND_UP(srclen, PAGE_SIZE); ++ unsigned long buf_start; ++ unsigned long total_out = 0; ++ ZSTD_inBuffer in_buf = { NULL, 0, 0 }; ++ ZSTD_outBuffer out_buf = { NULL, 0, 0 }; ++ ++ stream = ZSTD_createDStream( ++ ZSTD_BTRFS_MAX_INPUT, workspace->mem, workspace->size); ++ if (!stream) { ++ pr_debug("BTRFS: ZSTD_createDStream failed\n"); ++ ret = -EIO; ++ goto done; ++ } ++ ++ in_buf.src = kmap(pages_in[page_in_index]); ++ in_buf.pos = 0; ++ in_buf.size = min_t(size_t, srclen, PAGE_SIZE); ++ ++ out_buf.dst = workspace->buf; ++ out_buf.pos = 0; ++ out_buf.size = PAGE_SIZE; ++ ++ while (1) { ++ size_t ret2; ++ ++ ret2 = ZSTD_decompressStream(stream, &out_buf, &in_buf); ++ if (ZSTD_isError(ret2)) { ++ pr_debug("BTRFS: ZSTD_decompressStream returned %d\n", ++ ZSTD_getErrorCode(ret2)); ++ ret = -EIO; ++ goto done; ++ } ++ buf_start = total_out; ++ total_out += out_buf.pos; ++ out_buf.pos = 0; ++ ++ ret = btrfs_decompress_buf2page(out_buf.dst, buf_start, ++ total_out, disk_start, orig_bio); ++ if (ret == 0) ++ break; ++ ++ if (in_buf.pos >= srclen) ++ break; ++ ++ /* Check if we've hit the end of a frame */ ++ if (ret2 == 0) ++ break; ++ ++ if (in_buf.pos == in_buf.size) { ++ kunmap(pages_in[page_in_index++]); ++ if (page_in_index >= total_pages_in) { ++ in_buf.src = NULL; ++ ret = -EIO; ++ goto done; ++ } ++ srclen -= PAGE_SIZE; ++ in_buf.src = kmap(pages_in[page_in_index]); ++ in_buf.pos = 0; ++ in_buf.size = min_t(size_t, srclen, PAGE_SIZE); ++ } ++ } ++ ret = 0; ++ zero_fill_bio(orig_bio); ++done: ++ if (in_buf.src) ++ kunmap(pages_in[page_in_index]); ++ return ret; ++} ++ ++static int zstd_decompress(struct list_head *ws, unsigned char *data_in, ++ struct page *dest_page, ++ unsigned long start_byte, ++ size_t srclen, size_t destlen) ++{ ++ struct workspace *workspace = list_entry(ws, struct workspace, list); ++ ZSTD_DStream *stream; ++ int ret = 0; ++ size_t ret2; ++ ZSTD_inBuffer in_buf = { NULL, 0, 0 }; ++ ZSTD_outBuffer out_buf = { NULL, 0, 0 }; ++ unsigned long total_out = 0; ++ unsigned long pg_offset = 0; ++ char *kaddr; ++ ++ stream = ZSTD_createDStream( ++ ZSTD_BTRFS_MAX_INPUT, workspace->mem, workspace->size); ++ if (!stream) { ++ pr_warn("BTRFS: ZSTD_createDStream failed\n"); ++ ret = -EIO; ++ goto finish; ++ } ++ ++ destlen = min_t(size_t, destlen, PAGE_SIZE); ++ ++ in_buf.src = data_in; ++ in_buf.pos = 0; ++ in_buf.size = srclen; ++ ++ out_buf.dst = workspace->buf; ++ out_buf.pos = 0; ++ out_buf.size = PAGE_SIZE; ++ ++ ret2 = 1; ++ while (pg_offset < destlen && in_buf.pos < in_buf.size) { ++ unsigned long buf_start; ++ unsigned long buf_offset; ++ unsigned long bytes; ++ ++ /* Check if the frame is over and we still need more input */ ++ if (ret2 == 0) { ++ pr_debug("BTRFS: ZSTD_decompressStream ended early\n"); ++ ret = -EIO; ++ goto finish; ++ } ++ ret2 = ZSTD_decompressStream(stream, &out_buf, &in_buf); ++ if (ZSTD_isError(ret2)) { ++ pr_debug("BTRFS: ZSTD_decompressStream returned %d\n", ++ ZSTD_getErrorCode(ret2)); ++ ret = -EIO; ++ goto finish; ++ } ++ ++ buf_start = total_out; ++ total_out += out_buf.pos; ++ out_buf.pos = 0; ++ ++ if (total_out <= start_byte) ++ continue; ++ ++ if (total_out > start_byte && buf_start < start_byte) ++ buf_offset = start_byte - buf_start; ++ else ++ buf_offset = 0; ++ ++ bytes = min_t(unsigned long, destlen - pg_offset, ++ out_buf.size - buf_offset); ++ ++ kaddr = kmap_atomic(dest_page); ++ memcpy(kaddr + pg_offset, out_buf.dst + buf_offset, bytes); ++ kunmap_atomic(kaddr); ++ ++ pg_offset += bytes; ++ } ++ ret = 0; ++finish: ++ if (pg_offset < destlen) { ++ kaddr = kmap_atomic(dest_page); ++ memset(kaddr + pg_offset, 0, destlen - pg_offset); ++ kunmap_atomic(kaddr); ++ } ++ return ret; ++} ++ ++const struct btrfs_compress_op btrfs_zstd_compress = { ++ .alloc_workspace = zstd_alloc_workspace, ++ .free_workspace = zstd_free_workspace, ++ .compress_pages = zstd_compress_pages, ++ .decompress_bio = zstd_decompress_bio, ++ .decompress = zstd_decompress, ++}; +diff --git a/include/uapi/linux/btrfs.h b/include/uapi/linux/btrfs.h +index db4c253..f26c34f 100644 +--- a/include/uapi/linux/btrfs.h ++++ b/include/uapi/linux/btrfs.h +@@ -255,13 +255,7 @@ struct btrfs_ioctl_fs_info_args { + #define BTRFS_FEATURE_INCOMPAT_DEFAULT_SUBVOL (1ULL << 1) + #define BTRFS_FEATURE_INCOMPAT_MIXED_GROUPS (1ULL << 2) + #define BTRFS_FEATURE_INCOMPAT_COMPRESS_LZO (1ULL << 3) +-/* +- * some patches floated around with a second compression method +- * lets save that incompat here for when they do get in +- * Note we don't actually support it, we're just reserving the +- * number +- */ +-#define BTRFS_FEATURE_INCOMPAT_COMPRESS_LZOv2 (1ULL << 4) ++#define BTRFS_FEATURE_INCOMPAT_COMPRESS_ZSTD (1ULL << 4) + + /* + * older kernels tried to do bigger metadata blocks, but the diff --git a/contrib/linux-kernel/fs/btrfs/zstd.c b/contrib/linux-kernel/fs/btrfs/zstd.c index 23a3692ad..b7f319e7a 100644 --- a/contrib/linux-kernel/fs/btrfs/zstd.c +++ b/contrib/linux-kernel/fs/btrfs/zstd.c @@ -15,9 +15,10 @@ static ZSTD_parameters zstd_get_btrfs_parameters(size_t src_len) { ZSTD_parameters params = ZSTD_getParams(3, src_len, 0); - BUG_ON(src_len > ZSTD_BTRFS_MAX_INPUT); - BUG_ON(params.cParams.windowLog > ZSTD_BTRFS_MAX_WINDOWLOG); - params.fParams.checksumFlag = 1; + + if (params.cParams.windowLog > ZSTD_BTRFS_MAX_WINDOWLOG) + params.cParams.windowLog = ZSTD_BTRFS_MAX_WINDOWLOG; + WARN_ON(src_len > ZSTD_BTRFS_MAX_INPUT); return params; } @@ -39,17 +40,21 @@ static void zstd_free_workspace(struct list_head *ws) static struct list_head *zstd_alloc_workspace(void) { - ZSTD_parameters params = zstd_get_btrfs_parameters(ZSTD_BTRFS_MAX_INPUT); + ZSTD_parameters params = + zstd_get_btrfs_parameters(ZSTD_BTRFS_MAX_INPUT); struct workspace *workspace; workspace = kzalloc(sizeof(*workspace), GFP_NOFS); - if (!workspace) return ERR_PTR(-ENOMEM); + if (!workspace) + return ERR_PTR(-ENOMEM); - workspace->size = max_t(size_t, ZSTD_CStreamWorkspaceBound(params.cParams), + workspace->size = max_t(size_t, + ZSTD_CStreamWorkspaceBound(params.cParams), ZSTD_DStreamWorkspaceBound(ZSTD_BTRFS_MAX_INPUT)); workspace->mem = vmalloc(workspace->size); workspace->buf = kmalloc(PAGE_SIZE, GFP_NOFS); - if (!workspace->mem || !workspace->buf) goto fail; + if (!workspace->mem || !workspace->buf) + goto fail; INIT_LIST_HEAD(&workspace->list); @@ -61,16 +66,13 @@ fail: static int zstd_compress_pages(struct list_head *ws, struct address_space *mapping, - u64 start, unsigned long len, + u64 start, struct page **pages, - unsigned long nr_dest_pages, unsigned long *out_pages, unsigned long *total_in, - unsigned long *total_out, - unsigned long max_out) + unsigned long *total_out) { struct workspace *workspace = list_entry(ws, struct workspace, list); - ZSTD_parameters params = zstd_get_btrfs_parameters(len); ZSTD_CStream *stream; int ret = 0; int nr_pages = 0; @@ -80,13 +82,18 @@ static int zstd_compress_pages(struct list_head *ws, ZSTD_outBuffer out_buf = { NULL, 0, 0 }; unsigned long tot_in = 0; unsigned long tot_out = 0; + unsigned long len = *total_out; + const unsigned long nr_dest_pages = *out_pages; + unsigned long max_out = nr_dest_pages * PAGE_SIZE; + ZSTD_parameters params = zstd_get_btrfs_parameters(len); *out_pages = 0; *total_out = 0; *total_in = 0; /* Initialize the stream */ - stream = ZSTD_createCStream(params, len, workspace->mem, workspace->size); + stream = ZSTD_createCStream(params, len, workspace->mem, + workspace->size); if (!stream) { pr_warn("BTRFS: ZSTD_createStream failed\n"); ret = -EIO; @@ -112,10 +119,12 @@ static int zstd_compress_pages(struct list_head *ws, out_buf.size = min_t(size_t, max_out, PAGE_SIZE); while (1) { - const size_t rc = ZSTD_compressStream(stream, &out_buf, &in_buf); - if (ZSTD_isError(rc)) { + size_t ret2; + + ret2 = ZSTD_compressStream(stream, &out_buf, &in_buf); + if (ZSTD_isError(ret2)) { pr_debug("BTRFS: ZSTD_compressStream returned %d\n", - ZSTD_getErrorCode(rc)); + ZSTD_getErrorCode(ret2)); ret = -EIO; goto out; } @@ -177,14 +186,16 @@ static int zstd_compress_pages(struct list_head *ws, } } while (1) { - const size_t rc = ZSTD_endStream(stream, &out_buf); - if (ZSTD_isError(rc)) { + size_t ret2; + + ret2 = ZSTD_endStream(stream, &out_buf); + if (ZSTD_isError(ret2)) { pr_debug("BTRFS: ZSTD_endStream returned %d\n", - ZSTD_getErrorCode(rc)); + ZSTD_getErrorCode(ret2)); ret = -EIO; goto out; } - if (rc == 0) { + if (ret2 == 0) { tot_out += out_buf.pos; break; } @@ -228,24 +239,22 @@ out: kunmap(in_page); put_page(in_page); } - if (out_page) { kunmap(out_page); } + if (out_page) + kunmap(out_page); return ret; } -static int zstd_decompress_biovec(struct list_head *ws, struct page **pages_in, +static int zstd_decompress_bio(struct list_head *ws, struct page **pages_in, u64 disk_start, - struct bio_vec *bvec, - int vcnt, + struct bio *orig_bio, size_t srclen) { struct workspace *workspace = list_entry(ws, struct workspace, list); ZSTD_DStream *stream; int ret = 0; unsigned long page_in_index = 0; - unsigned long page_out_index = 0; unsigned long total_pages_in = DIV_ROUND_UP(srclen, PAGE_SIZE); unsigned long buf_start; - unsigned long pg_offset; unsigned long total_out = 0; ZSTD_inBuffer in_buf = { NULL, 0, 0 }; ZSTD_outBuffer out_buf = { NULL, 0, 0 }; @@ -266,13 +275,13 @@ static int zstd_decompress_biovec(struct list_head *ws, struct page **pages_in, out_buf.pos = 0; out_buf.size = PAGE_SIZE; - pg_offset = 0; - while (1) { - const size_t rc = ZSTD_decompressStream(stream, &out_buf, &in_buf); - if (ZSTD_isError(rc)) { + size_t ret2; + + ret2 = ZSTD_decompressStream(stream, &out_buf, &in_buf); + if (ZSTD_isError(ret2)) { pr_debug("BTRFS: ZSTD_decompressStream returned %d\n", - ZSTD_getErrorCode(rc)); + ZSTD_getErrorCode(ret2)); ret = -EIO; goto done; } @@ -280,23 +289,17 @@ static int zstd_decompress_biovec(struct list_head *ws, struct page **pages_in, total_out += out_buf.pos; out_buf.pos = 0; - { - int ret2 = btrfs_decompress_buf2page(out_buf.dst, buf_start, - total_out, disk_start, bvec, vcnt, - &page_out_index, &pg_offset); - if (ret2 == 0) { - break; - } - } - - if (in_buf.pos >= srclen) { + ret = btrfs_decompress_buf2page(out_buf.dst, buf_start, + total_out, disk_start, orig_bio); + if (ret == 0) + break; + + if (in_buf.pos >= srclen) break; - } /* Check if we've hit the end of a frame */ - if (rc == 0) { + if (ret2 == 0) break; - } if (in_buf.pos == in_buf.size) { kunmap(pages_in[page_in_index++]); @@ -311,10 +314,11 @@ static int zstd_decompress_biovec(struct list_head *ws, struct page **pages_in, in_buf.size = min_t(size_t, srclen, PAGE_SIZE); } } - btrfs_clear_biovec_end(bvec, vcnt, page_out_index, pg_offset); ret = 0; + zero_fill_bio(orig_bio); done: - if (in_buf.src) { kunmap(pages_in[page_in_index]); } + if (in_buf.src) + kunmap(pages_in[page_in_index]); return ret; } @@ -326,6 +330,7 @@ static int zstd_decompress(struct list_head *ws, unsigned char *data_in, struct workspace *workspace = list_entry(ws, struct workspace, list); ZSTD_DStream *stream; int ret = 0; + size_t ret2; ZSTD_inBuffer in_buf = { NULL, 0, 0 }; ZSTD_outBuffer out_buf = { NULL, 0, 0 }; unsigned long total_out = 0; @@ -350,41 +355,37 @@ static int zstd_decompress(struct list_head *ws, unsigned char *data_in, out_buf.pos = 0; out_buf.size = PAGE_SIZE; - ret = 1; + ret2 = 1; while (pg_offset < destlen && in_buf.pos < in_buf.size) { unsigned long buf_start; unsigned long buf_offset; unsigned long bytes; /* Check if the frame is over and we still need more input */ - if (ret == 0) { - pr_debug("BTRFS: ZSTD_decompressStream frame ended to early\n"); + if (ret2 == 0) { + pr_debug("BTRFS: ZSTD_decompressStream ended early\n"); ret = -EIO; goto finish; } - { - const size_t rc = ZSTD_decompressStream(stream, &out_buf, &in_buf); - if (ZSTD_isError(rc)) { - pr_debug("BTRFS: ZSTD_decompressStream returned %d\n", - ZSTD_getErrorCode(rc)); - ret = -EIO; - goto finish; - } - ret = rc > 0; + ret2 = ZSTD_decompressStream(stream, &out_buf, &in_buf); + if (ZSTD_isError(ret2)) { + pr_debug("BTRFS: ZSTD_decompressStream returned %d\n", + ZSTD_getErrorCode(ret2)); + ret = -EIO; + goto finish; } + buf_start = total_out; total_out += out_buf.pos; out_buf.pos = 0; - if (total_out <= start_byte) { + if (total_out <= start_byte) continue; - } - if (total_out > start_byte && buf_start < start_byte) { + if (total_out > start_byte && buf_start < start_byte) buf_offset = start_byte - buf_start; - } else { + else buf_offset = 0; - } bytes = min_t(unsigned long, destlen - pg_offset, out_buf.size - buf_offset); @@ -409,6 +410,6 @@ const struct btrfs_compress_op btrfs_zstd_compress = { .alloc_workspace = zstd_alloc_workspace, .free_workspace = zstd_free_workspace, .compress_pages = zstd_compress_pages, - .decompress_biovec = zstd_decompress_biovec, + .decompress_bio = zstd_decompress_bio, .decompress = zstd_decompress, }; From 4ee6b15dac055e7e653b1fbfd7f65755f42597c8 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 11 Apr 2017 11:59:44 -0700 Subject: [PATCH 164/305] force contentSizeFlag=0 when using ZSTD_initCStream_usingCDict() because by definition srcSize is not known when using this prototype. added relevant test Note : this use was already working, because at a later stage (both ZSTD_compressBegin_usingCDict() and ZSTD_copyCCtx()) pledgedSrcSize=0 is translated into "unknown", no matter the frame parameter. This is not correct, but of little importance, as the medium term plan is to no longer set fParams within CDict --- lib/compress/zstd_compress.c | 3 ++- tests/zstreamtest.c | 14 +++++++++----- 2 files changed, 11 insertions(+), 6 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 62c1c6cdd..0ce2054fc 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -3077,7 +3077,8 @@ static size_t ZSTD_initCStream_stage2(ZSTD_CStream* zcs, size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) { if (!cdict) return ERROR(GENERIC); /* cannot handle NULL cdict (does not know what to do) */ - { ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict); + { ZSTD_parameters params = ZSTD_getParamsFromCDict(cdict); + params.fParams.contentSizeFlag = 0; zcs->cdict = cdict; return ZSTD_initCStream_stage2(zcs, params, 0); } diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index 6f26ea555..ac200a7a6 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -448,10 +448,11 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo if (!ZSTD_isError(r)) goto _output_error; /* must fail : frame requires > 100 bytes */ DISPLAYLEVEL(3, "OK (%s)\n", ZSTD_getErrorName(r)); } - DISPLAYLEVEL(3, "test%3i : check dictionary used : ", testNb++); + DISPLAYLEVEL(3, "test%3i : dictionary compression with masked dictID : ", testNb++); { ZSTD_parameters params = ZSTD_getParams(1, CNBufferSize, dictionary.filled); ZSTD_CDict* cdict; params.fParams.noDictIDFlag = 1; + params.fParams.contentSizeFlag = 1; /* test contentSize, should be disabled with initCStream_usingCDict */ cdict = ZSTD_createCDict_advanced(dictionary.start, dictionary.filled, 1, params, customMem); { size_t const initError = ZSTD_initCStream_usingCDict(zc, cdict); if (ZSTD_isError(initError)) goto _output_error; } @@ -470,17 +471,20 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo cSize = outBuff.pos; ZSTD_freeCDict(cdict); DISPLAYLEVEL(3, "OK (%u bytes : %.2f%%)\n", (U32)cSize, (double)cSize/CNBufferSize*100); + } - { size_t const r = ZSTD_decompress(decodedBuffer, CNBufferSize, compressedBuffer, cSize); - if (!ZSTD_isError(r)) goto _output_error; /* must fail : dictionary not used */ - DISPLAYLEVEL(3, "OK (%s)\n", ZSTD_getErrorName(r)); } + DISPLAYLEVEL(3, "test%3i : decompress without dictionary : ", testNb++); + { size_t const r = ZSTD_decompress(decodedBuffer, CNBufferSize, compressedBuffer, cSize); + if (!ZSTD_isError(r)) goto _output_error; /* must fail : dictionary not used */ + DISPLAYLEVEL(3, "OK (%s)\n", ZSTD_getErrorName(r)); } /* Unknown srcSize */ DISPLAYLEVEL(3, "test%3i : pledgedSrcSize == 0 behaves properly : ", testNb++); { ZSTD_parameters params = ZSTD_getParams(5, 0, 0); params.fParams.contentSizeFlag = 1; - ZSTD_initCStream_advanced(zc, NULL, 0, params, 0); } /* cstream advanced should write the 0 size field */ + ZSTD_initCStream_advanced(zc, NULL, 0, params, 0); + } /* cstream advanced shall write content size = 0 */ inBuff.src = CNBuffer; inBuff.size = 0; inBuff.pos = 0; From b633377d0e6e2857e2ad2ffaa57f3015b7bc0b8f Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 11 Apr 2017 12:40:53 -0700 Subject: [PATCH 165/305] Add BtrFS benchmarks --- contrib/linux-kernel/README.md | 26 ++++++ contrib/linux-kernel/btrfs-benchmark.sh | 104 ++++++++++++++++++++++++ 2 files changed, 130 insertions(+) create mode 100755 contrib/linux-kernel/btrfs-benchmark.sh diff --git a/contrib/linux-kernel/README.md b/contrib/linux-kernel/README.md index 1cc74cca6..a62838517 100644 --- a/contrib/linux-kernel/README.md +++ b/contrib/linux-kernel/README.md @@ -24,3 +24,29 @@ The patches are based off of the linux kernel master branch (version 4.10). * Additionally `fs/btrfs/zstd.c` is provided as a source for convenience. * The patch seems to be working, it doesn't crash the kernel, and compresses at speeds and ratios athat are expected. It can still use some more testing for fringe features, like printing options. + +### Benchmarks + +Benchmarks run on a Ubuntu 14.04 with 2 cores and 4 GiB of RAM. +The VM is running on a Macbook Pro with a 3.1 GHz Intel Core i7 processor, +16 GB of ram, and a SSD. + +The compression benchmark is copying 10 copies of the +unzipped [silesia corpus](http://mattmahoney.net/dc/silesia.html) into a BtrFS +filesystem mounted with `-o compress-force={none, lzo, zlib, zstd}`. +The decompression benchmark is timing how long it takes to `tar` all 10 copies +into `/dev/null`. +The compression ratio is measured by comparing the output of `df` and `du`. +See `btrfs-benchmark.sh` for details. + +| Algorithm | Compression ratio | Compression speed | Decompression speed | +|-----------|-------------------|-------------------|---------------------| +| None | 0.99 | 504 MB/s | 686 MB/s | +| lzo | 1.66 | 398 MB/s | 442 MB/s | +| zlib | 2.58 | 65 MB/s | 241 MB/s | +| zstd 1 | 2.57 | 260 MB/s | 383 MB/s | +| zstd 3 | 2.71 | 174 MB/s | 408 MB/s | +| zstd 6 | 2.87 | 70 MB/s | 398 MB/s | +| zstd 9 | 2.92 | 43 MB/s | 406 MB/s | +| zstd 12 | 2.93 | 21 MB/s | 408 MB/s | +| zstd 15 | 3.01 | 11 MB/s | 354 MB/s | diff --git a/contrib/linux-kernel/btrfs-benchmark.sh b/contrib/linux-kernel/btrfs-benchmark.sh new file mode 100755 index 000000000..5e28da9c6 --- /dev/null +++ b/contrib/linux-kernel/btrfs-benchmark.sh @@ -0,0 +1,104 @@ +# !/bin/sh +set -e + +# Benchmarks run on a Ubuntu 14.04 VM with 2 cores and 4 GiB of RAM. +# The VM is running on a Macbook Pro with a 3.1 GHz Intel Core i7 processor and +# 16 GB of RAM and an SSD. + +# silesia is a directory that can be downloaded from +# http://mattmahoney.net/dc/silesia.html +# ls -l silesia/ +# total 203M +# -rwxr-xr-x 1 terrelln 9.8M Apr 12 2002 dickens +# -rwxr-xr-x 1 terrelln 49M May 31 2002 mozilla +# -rwxr-xr-x 1 terrelln 9.6M Mar 20 2003 mr +# -rwxr-xr-x 1 terrelln 32M Apr 2 2002 nci +# -rwxr-xr-x 1 terrelln 5.9M Jul 4 2002 ooffice +# -rwxr-xr-x 1 terrelln 9.7M Apr 11 2002 osdb +# -rwxr-xr-x 1 terrelln 6.4M Apr 2 2002 reymont +# -rwxr-xr-x 1 terrelln 21M Mar 25 2002 samba +# -rwxr-xr-x 1 terrelln 7.0M Mar 24 2002 sao +# -rwxr-xr-x 1 terrelln 40M Mar 25 2002 webster +# -rwxr-xr-x 1 terrelln 8.1M Apr 4 2002 x-ray +# -rwxr-xr-x 1 terrelln 5.1M Nov 30 2000 xml + +# $HOME is on a ext4 filesystem +BENCHMARK_DIR="$HOME/silesia/" +N=10 + +# Normalize the environment +sudo umount /mnt/btrfs 2> /dev/null > /dev/null || true +sudo mount -t btrfs $@ /dev/sda3 /mnt/btrfs +sudo rm -rf /mnt/btrfs/* +sync +sudo umount /mnt/btrfs +sudo mount -t btrfs $@ /dev/sda3 /mnt/btrfs + +# Run the benchmark +echo "Compression" +time sh -c "for i in \$(seq $N); do sudo cp -r $BENCHMARK_DIR /mnt/btrfs/\$i; done; sync" + +echo "Approximate compression ratio" +printf "%d / %d\n" \ + $(df /mnt/btrfs --output=used -B 1 | tail -n 1) \ + $(sudo du /mnt/btrfs -b -d 0 | tr '\t' '\n' | head -n 1); + +# Unmount and remount to avoid any caching +sudo umount /mnt/btrfs +sudo mount -t btrfs $@ /dev/sda3 /mnt/btrfs + +echo "Decompression" +time sudo tar -c /mnt/btrfs 2> /dev/null | wc -c > /dev/null + +sudo rm -rf /mnt/btrfs/* +sudo umount /mnt/btrfs + +# Run for each of -o compress-force={none, lzo, zlib, zstd} 5 times and take the +# min time and ratio. +# Ran zstd with compression levels {1, 3, 6, 9, 12, 15}. +# Original size: 2119415342 B (using du /mnt/btrfs) + +# none +# compress: 4.205 s +# decompress: 3.090 s +# ratio: 0.99 + +# lzo +# compress: 5.328 s +# decompress: 4.793 s +# ratio: 1.66 + +# zlib +# compress: 32.588 s +# decompress: 8.791 s +# ratio : 2.58 + +# zstd 1 +# compress: 8.147 s +# decompress: 5.527 s +# ratio : 2.57 + +# zstd 3 +# compress: 12.207 s +# decompress: 5.195 s +# ratio : 2.71 + +# zstd 6 +# compress: 30.253 s +# decompress: 5.324 s +# ratio : 2.87 + +# zstd 9 +# compress: 49.659 s +# decompress: 5.220 s +# ratio : 2.92 + +# zstd 12 +# compress: 99.245 s +# decompress: 5.193 s +# ratio : 2.93 + +# zstd 15 +# compress: 196.997 s +# decompress: 5.992 s +# ratio : 3.01 From c3ba15e48f89071b2f2ab5aa137b5c5774709e3a Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 10 Apr 2017 16:22:35 -0700 Subject: [PATCH 166/305] Seekable compression demo --- doc/zstd_manual.html | 107 +++++++++ examples/.gitignore | 2 + examples/Makefile | 9 +- examples/seekable_compression.c | 126 ++++++++++ examples/seekable_decompression.c | 175 ++++++++++++++ lib/common/error_private.c | 2 + lib/common/seekable.h | 23 ++ lib/common/zstd_errors.h | 2 + lib/compress/zstdseek_compress.c | 265 ++++++++++++++++++++ lib/decompress/zstdseek_decompress.c | 347 +++++++++++++++++++++++++++ lib/zstd.h | 117 +++++++++ 11 files changed, 1174 insertions(+), 1 deletion(-) create mode 100644 examples/seekable_compression.c create mode 100644 examples/seekable_decompression.c create mode 100644 lib/common/seekable.h create mode 100644 lib/compress/zstdseek_compress.c create mode 100644 lib/decompress/zstdseek_decompress.c diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index a66f0a49c..a47b7142a 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -28,6 +28,9 @@
  1. Buffer-less streaming compression (synchronous mode)
  2. Buffer-less streaming decompression (synchronous mode)
  3. Block functions
  4. +
  5. Seekable Format
  6. +
  7. Seekable compression - HowTo
  8. +
  9. Seekable decompression - HowTo

Introduction

@@ -660,5 +663,109 @@ size_t ZSTD_compressBlock  (ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, cons
 size_t ZSTD_decompressBlock(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
 size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize);  /**< insert block into `dctx` history. Useful for uncompressed blocks */
 

+

Seekable Format

+  The seekable format splits the compressed data into a series of "chunks",
+  each compressed individually so that decompression of a section in the
+  middle of an archive only requires zstd to decompress at most a chunk's
+  worth of extra data, instead of the entire archive.
+
+ +

Seekable compression - HowTo

  A ZSTD_seekable_CStream object is required to tracking streaming operation.
+  Use ZSTD_seekable_createCStream() and ZSTD_seekable_freeCStream() to create/
+  release resources.
+
+  Streaming objects are reusable to avoid allocation and deallocation,
+  to start a new compression operation call ZSTD_seekable_initCStream() on the
+  compressor.
+
+  Data streamed to the seekable compressor will automatically be split into
+  chunks of size `maxChunkSize` (provided in ZSTD_seekable_initCStream()),
+  or if none is provided, will be cut off whenver ZSTD_endChunk() is called
+  or when the default maximum chunk size is reached (approximately 4GB).
+
+  Use ZSTD_seekable_initCStream() to initialize a ZSTD_seekable_CStream object
+  for a new compression operation.
+  `maxChunkSize` indicates the size at which to automatically start a new
+  seekable frame.  `maxChunkSize == 0` implies the default maximum size.
+  @return : a size hint for input to provide for compression, or an error code
+            checkable with ZSTD_isError()
+
+  Use ZSTD_seekable_compressStream() repetitively to consume input stream.
+  The function will automatically update both `pos` fields.
+  Note that it may not consume the entire input, in which case `pos < size`,
+  and it's up to the caller to present again remaining data.
+  @return : a size hint, preferred nb of bytes to use as input for next
+            function call or an error code, which can be tested using
+            ZSTD_isError().
+            Note 1 : it's just a hint, to help latency a little, any other
+                     value will work fine.
+            Note 2 : size hint is guaranteed to be <= ZSTD_CStreamInSize()
+
+  At any time, call ZSTD_seekable_endChunk() to end the current chunk and
+  start a new one.
+
+  ZSTD_endStream() will end the current chunk, and then write the seek table
+  so that decompressors can efficiently find compressed chunks.
+  ZSTD_endStream() may return a number > 0 if it was unable to flush all the
+  necessary data to `output`.  In this case, it should be called again until
+  all remaining data is flushed out and 0 is returned.
+
+ +

Seekable compressor management

ZSTD_seekable_CStream* ZSTD_seekable_createCStream(void);
+size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs);
+

+

Seekable compression functions

size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, unsigned maxChunkSize);
+size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input);
+size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output);
+size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output);
+

+

Seekable decompression - HowTo

  A ZSTD_seekable_DStream object is required to tracking streaming operation.
+  Use ZSTD_seekable_createDStream() and ZSTD_seekable_freeDStream() to create/
+  release resources.
+
+  Streaming objects are reusable to avoid allocation and deallocation,
+  to start a new compression operation call ZSTD_seekable_initDStream() on the
+  compressor.
+
+  Use ZSTD_seekable_loadSeekTable() to load the seek table from a file.
+  `src` should point to a block of data read from the end of the file,
+  i.e. `src + srcSize` should always be the end of the file.
+  @return : 0 if the table was loaded successfully, or if `srcSize` was too
+            small, a size hint for how much data to provide.
+            An error code may also be returned, checkable with ZSTD_isError()
+
+  Use ZSTD_initDStream to prepare for a new decompression operation using the
+  seektable loaded with ZSTD_seekable_loadSeekTable().
+  Data in the range [rangeStart, rangeEnd) will be decompressed.
+
+  Call ZSTD_seekable_decompressStream() repetitively to consume input stream.
+  @return : There are a number of possible return codes for this function
+           - 0, the decompression operation has completed.
+           - An error code checkable with ZSTD_isError
+             + If this error code is ZSTD_error_needSeek, the user should seek
+               to the file position provided by ZSTD_seekable_getSeekOffset()
+               and indicate this to the stream with
+               ZSTD_seekable_updateOffset(), before resuming decompression
+             + Otherwise, this is a regular decompression error and the input
+               file is likely corrupted or the API was incorrectly used.
+           - A size hint, the preferred nb of bytes to provide as input to the
+             next function call to improve latency.
+
+  ZSTD_seekable_getSeekOffset() and ZSTD_seekable_updateOffset() are helper
+  functions to indicate where the user should seek their file stream to, when
+  a different position is required to continue decompression.
+  Note that ZSTD_seekable_updateOffset will error if given an offset other
+  than the one requested from ZSTD_seekable_getSeekOffset().
+
+ +

Seekable decompressor management

ZSTD_seekable_DStream* ZSTD_seekable_createDStream(void);
+size_t ZSTD_seekable_freeDStream(ZSTD_seekable_DStream* zds);
+

+

Seekable decompression functions

size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, size_t srcSize);
+size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, unsigned long long rangeStart, unsigned long long rangeEnd);
+size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input);
+unsigned long long ZSTD_seekable_getSeekOffset(ZSTD_seekable_DStream* zds);
+size_t ZSTD_seekable_updateOffset(ZSTD_seekable_DStream* zds, unsigned long long offset);
+

diff --git a/examples/.gitignore b/examples/.gitignore index 0711813d3..c6b2a9f42 100644 --- a/examples/.gitignore +++ b/examples/.gitignore @@ -6,6 +6,8 @@ dictionary_decompression streaming_compression streaming_decompression multiple_streaming_compression +seekable_compression +seekable_decompression #test artefact tmp* diff --git a/examples/Makefile b/examples/Makefile index b84983f08..88cc10d24 100644 --- a/examples/Makefile +++ b/examples/Makefile @@ -18,7 +18,8 @@ default: all all: simple_compression simple_decompression \ dictionary_compression dictionary_decompression \ streaming_compression streaming_decompression \ - multiple_streaming_compression + multiple_streaming_compression \ + seekable_compression simple_compression : simple_compression.c $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ @@ -41,6 +42,12 @@ multiple_streaming_compression : multiple_streaming_compression.c streaming_decompression : streaming_decompression.c $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ +seekable_compression : seekable_compression.c + $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ + +seekable_decompression : seekable_decompression.c + $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ + clean: @rm -f core *.o tmp* result* *.zst \ simple_compression simple_decompression \ diff --git a/examples/seekable_compression.c b/examples/seekable_compression.c new file mode 100644 index 000000000..5ab36c3cb --- /dev/null +++ b/examples/seekable_compression.c @@ -0,0 +1,126 @@ +/** + * Copyright 2017-present, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the license found in the + * LICENSE-examples file in the root directory of this source tree. + */ + +#include // malloc, free, exit, atoi +#include // fprintf, perror, feof, fopen, etc. +#include // strlen, memset, strcat +#define ZSTD_STATIC_LINKING_ONLY +#include // presumes zstd library is installed + +static void* malloc_orDie(size_t size) +{ + void* const buff = malloc(size); + if (buff) return buff; + /* error */ + perror("malloc:"); + exit(1); +} + +static FILE* fopen_orDie(const char *filename, const char *instruction) +{ + FILE* const inFile = fopen(filename, instruction); + if (inFile) return inFile; + /* error */ + perror(filename); + exit(3); +} + +static size_t fread_orDie(void* buffer, size_t sizeToRead, FILE* file) +{ + size_t const readSize = fread(buffer, 1, sizeToRead, file); + if (readSize == sizeToRead) return readSize; /* good */ + if (feof(file)) return readSize; /* good, reached end of file */ + /* error */ + perror("fread"); + exit(4); +} + +static size_t fwrite_orDie(const void* buffer, size_t sizeToWrite, FILE* file) +{ + size_t const writtenSize = fwrite(buffer, 1, sizeToWrite, file); + if (writtenSize == sizeToWrite) return sizeToWrite; /* good */ + /* error */ + perror("fwrite"); + exit(5); +} + +static size_t fclose_orDie(FILE* file) +{ + if (!fclose(file)) return 0; + /* error */ + perror("fclose"); + exit(6); +} + +static void compressFile_orDie(const char* fname, const char* outName, int cLevel, unsigned chunkSize) +{ + FILE* const fin = fopen_orDie(fname, "rb"); + FILE* const fout = fopen_orDie(outName, "wb"); + size_t const buffInSize = ZSTD_CStreamInSize(); /* can always read one full block */ + void* const buffIn = malloc_orDie(buffInSize); + size_t const buffOutSize = ZSTD_CStreamOutSize(); /* can always flush a full block */ + void* const buffOut = malloc_orDie(buffOutSize); + + ZSTD_seekable_CStream* const cstream = ZSTD_seekable_createCStream(); + if (cstream==NULL) { fprintf(stderr, "ZSTD_seekable_createCStream() error \n"); exit(10); } + size_t const initResult = ZSTD_seekable_initCStream(cstream, cLevel, chunkSize); + if (ZSTD_isError(initResult)) { fprintf(stderr, "ZSTD_seekable_initCStream() error : %s \n", ZSTD_getErrorName(initResult)); exit(11); } + + size_t read, toRead = buffInSize; + while( (read = fread_orDie(buffIn, toRead, fin)) ) { + ZSTD_inBuffer input = { buffIn, read, 0 }; + while (input.pos < input.size) { + ZSTD_outBuffer output = { buffOut, buffOutSize, 0 }; + toRead = ZSTD_seekable_compressStream(cstream, &output , &input); /* toRead is guaranteed to be <= ZSTD_CStreamInSize() */ + if (ZSTD_isError(toRead)) { fprintf(stderr, "ZSTD_compressStream() error : %s \n", ZSTD_getErrorName(toRead)); exit(12); } + if (toRead > buffInSize) toRead = buffInSize; /* Safely handle case when `buffInSize` is manually changed to a value < ZSTD_CStreamInSize()*/ + fwrite_orDie(buffOut, output.pos, fout); + } + } + + ZSTD_outBuffer output = { buffOut, buffOutSize, 0 }; + while (1) { + size_t const remainingToFlush = ZSTD_seekable_endStream(cstream, &output); /* close stream */ + fwrite_orDie(buffOut, output.pos, fout); + if (!remainingToFlush) break; + } + + ZSTD_seekable_freeCStream(cstream); + fclose_orDie(fout); + fclose_orDie(fin); + free(buffIn); + free(buffOut); +} + +static const char* createOutFilename_orDie(const char* filename) +{ + size_t const inL = strlen(filename); + size_t const outL = inL + 5; + void* outSpace = malloc_orDie(outL); + memset(outSpace, 0, outL); + strcat(outSpace, filename); + strcat(outSpace, ".zst"); + return (const char*)outSpace; +} + +int main(int argc, const char** argv) { + const char* const exeName = argv[0]; + if (argc!=3) { + printf("wrong arguments\n"); + printf("usage:\n"); + printf("%s FILE CHUNK_SIZE\n", exeName); + return 1; + } + + { const char* const inFileName = argv[1]; + unsigned const chunkSize = (unsigned)atoi(argv[2]); + + const char* const outFileName = createOutFilename_orDie(inFileName); + compressFile_orDie(inFileName, outFileName, 5, chunkSize); + } +} diff --git a/examples/seekable_decompression.c b/examples/seekable_decompression.c new file mode 100644 index 000000000..97563c3de --- /dev/null +++ b/examples/seekable_decompression.c @@ -0,0 +1,175 @@ +/** + * Copyright 2016-present, Yann Collet, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the license found in the + * LICENSE-examples file in the root directory of this source tree. + */ + + +#include // malloc, exit +#include // fprintf, perror, feof +#include // strerror +#include // errno +#define ZSTD_STATIC_LINKING_ONLY +#include // presumes zstd library is installed +#include + + +static void* malloc_orDie(size_t size) +{ + void* const buff = malloc(size); + if (buff) return buff; + /* error */ + perror("malloc"); + exit(1); +} + +static void* realloc_orDie(void* ptr, size_t size) +{ + ptr = realloc(ptr, size); + if (ptr) return ptr; + /* error */ + perror("realloc"); + exit(1); +} + +static FILE* fopen_orDie(const char *filename, const char *instruction) +{ + FILE* const inFile = fopen(filename, instruction); + if (inFile) return inFile; + /* error */ + perror(filename); + exit(3); +} + +static size_t fread_orDie(void* buffer, size_t sizeToRead, FILE* file) +{ + size_t const readSize = fread(buffer, 1, sizeToRead, file); + if (readSize == sizeToRead) return readSize; /* good */ + if (feof(file)) return readSize; /* good, reached end of file */ + /* error */ + perror("fread"); + exit(4); +} + +static size_t fwrite_orDie(const void* buffer, size_t sizeToWrite, FILE* file) +{ + size_t const writtenSize = fwrite(buffer, 1, sizeToWrite, file); + if (writtenSize == sizeToWrite) return sizeToWrite; /* good */ + /* error */ + perror("fwrite"); + exit(5); +} + +static size_t fclose_orDie(FILE* file) +{ + if (!fclose(file)) return 0; + /* error */ + perror("fclose"); + exit(6); +} + +static void fseek_orDie(FILE* file, long int offset, int origin) { + if (!fseek(file, offset, origin)) { + if (!fflush(file)) return; + } + /* error */ + perror("fseek"); + exit(7); +} + + +static void decompressFile_orDie(const char* fname, unsigned startOffset, unsigned endOffset) +{ + FILE* const fin = fopen_orDie(fname, "rb"); + size_t const buffInSize = ZSTD_DStreamInSize(); + void* const buffIn = malloc_orDie(buffInSize); + FILE* const fout = stdout; + size_t const buffOutSize = ZSTD_DStreamOutSize(); /* Guarantee to successfully flush at least one complete compressed block in all circumstances. */ + void* const buffOut = malloc_orDie(buffOutSize); + + ZSTD_seekable_DStream* const dstream = ZSTD_seekable_createDStream(); + if (dstream==NULL) { fprintf(stderr, "ZSTD_seekable_createDStream() error \n"); exit(10); } + + { size_t sizeNeeded = 0; + void* buffSeekTable = NULL; + + do { + sizeNeeded = ZSTD_seekable_loadSeekTable(dstream, buffSeekTable, sizeNeeded); + if (!sizeNeeded) break; + + if (ZSTD_isError(sizeNeeded)) { + fprintf(stderr, "ZSTD_seekable_loadSeekTable() error : %s \n", + ZSTD_getErrorName(sizeNeeded)); + exit(11); + } + + fseek_orDie(fin, -(long) sizeNeeded, SEEK_END); + buffSeekTable = realloc_orDie(buffSeekTable, sizeNeeded); + fread_orDie(buffSeekTable, sizeNeeded, fin); + } while (sizeNeeded > 0); + + free(buffSeekTable); + } + + /* In more complex scenarios, a file may consist of multiple appended frames (ex : pzstd). + * The following example decompresses only the first frame. + * It is compatible with other provided streaming examples */ + size_t const initResult = ZSTD_seekable_initDStream(dstream, startOffset, endOffset); + if (ZSTD_isError(initResult)) { fprintf(stderr, "ZSTD_seekable_initDStream() error : %s \n", ZSTD_getErrorName(initResult)); exit(11); } + + size_t result, read, toRead = 0; + + do { + read = fread_orDie(buffIn, toRead, fin); + { ZSTD_inBuffer input = { buffIn, read, 0 }; + ZSTD_outBuffer output = { buffOut, buffOutSize, 0 }; + result = ZSTD_seekable_decompressStream(dstream, &output, &input); + + if (ZSTD_isError(result)) { + if (ZSTD_getErrorCode(result) == ZSTD_error_needSeek) { + unsigned long long const offset = ZSTD_seekable_getSeekOffset(dstream); + fseek_orDie(fin, offset, SEEK_SET); + ZSTD_seekable_updateOffset(dstream, offset); + toRead = 0; + } else { + fprintf(stderr, + "ZSTD_seekable_decompressStream() error : %s \n", + ZSTD_getErrorName(result)); + exit(12); + } + } else { + toRead = result; + } + fwrite_orDie(buffOut, output.pos, fout); + } + } while (result > 0); + + ZSTD_seekable_freeDStream(dstream); + fclose_orDie(fin); + fclose_orDie(fout); + free(buffIn); + free(buffOut); +} + + +int main(int argc, const char** argv) +{ + const char* const exeName = argv[0]; + + if (argc!=4) { + fprintf(stderr, "wrong arguments\n"); + fprintf(stderr, "usage:\n"); + fprintf(stderr, "%s FILE\n", exeName); + return 1; + } + + { + const char* const inFilename = argv[1]; + unsigned const startOffset = (unsigned) atoi(argv[2]); + unsigned const endOffset = (unsigned) atoi(argv[3]); + decompressFile_orDie(inFilename, startOffset, endOffset); + } + return 0; +} diff --git a/lib/common/error_private.c b/lib/common/error_private.c index 44ae20104..4084edca6 100644 --- a/lib/common/error_private.c +++ b/lib/common/error_private.c @@ -38,6 +38,8 @@ const char* ERR_getErrorString(ERR_enum code) case PREFIX(dictionary_corrupted): return "Dictionary is corrupted"; case PREFIX(dictionary_wrong): return "Dictionary mismatch"; case PREFIX(dictionaryCreation_failed): return "Cannot create Dictionary from provided samples"; + case PREFIX(chunkIndex_tooLarge): return "Chunk index is too large"; + case PREFIX(needSeek): return "Wrong file position, a seek is required to continue"; case PREFIX(maxCode): default: return notErrorCode; } diff --git a/lib/common/seekable.h b/lib/common/seekable.h new file mode 100644 index 000000000..ab11a4388 --- /dev/null +++ b/lib/common/seekable.h @@ -0,0 +1,23 @@ +#ifndef SEEKABLE_H +#define SEEKABLE_H + +#if defined (__cplusplus) +extern "C" { +#endif + +#include "zstd_internal.h" + +static const unsigned ZSTD_seekTableFooterSize = 9; + +#define ZSTD_SEEKABLE_MAGICNUMBER 0x8F92EAB1 + +#define ZSTD_SEEKABLE_MAXCHUNKS 0x8000000U + +/* 0xFE03F607 is the largest number x such that ZSTD_compressBound(x) fits in a 32-bit integer */ +#define ZSTD_SEEKABLE_MAX_CHUNK_DECOMPRESSED_SIZE 0xFE03F607 + +#if defined (__cplusplus) +} +#endif + +#endif diff --git a/lib/common/zstd_errors.h b/lib/common/zstd_errors.h index 3d579d969..13a5608e0 100644 --- a/lib/common/zstd_errors.h +++ b/lib/common/zstd_errors.h @@ -58,6 +58,8 @@ typedef enum { ZSTD_error_dictionary_corrupted, ZSTD_error_dictionary_wrong, ZSTD_error_dictionaryCreation_failed, + ZSTD_error_chunkIndex_tooLarge, + ZSTD_error_needSeek, ZSTD_error_maxCode } ZSTD_ErrorCode; diff --git a/lib/compress/zstdseek_compress.c b/lib/compress/zstdseek_compress.c new file mode 100644 index 000000000..284724e58 --- /dev/null +++ b/lib/compress/zstdseek_compress.c @@ -0,0 +1,265 @@ +/** + * Copyright (c) 2017-present, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + +#include /* malloc, free */ + +#define XXH_STATIC_LINKING_ONLY +#include "xxhash.h" + +#include "zstd_internal.h" /* includes zstd.h */ +#include "seekable.h" + +typedef struct { + U32 cSize; + U32 dSize; + U32 checksum; +} chunklogEntry_t; + +typedef struct { + chunklogEntry_t* entries; + U32 size; + U32 capacity; +} chunklog_t; + +struct ZSTD_seekable_CStream_s { + ZSTD_CStream* cstream; + chunklog_t chunklog; + + U32 chunkCSize; + U32 chunkDSize; + + XXH64_state_t xxhState; + + U32 maxChunkSize; + + int checksumFlag; +}; + +ZSTD_seekable_CStream* ZSTD_seekable_createCStream() +{ + ZSTD_seekable_CStream* zcs = malloc(sizeof(ZSTD_seekable_CStream)); + + if (zcs == NULL) return NULL; + + memset(zcs, 0, sizeof(*zcs)); + + zcs->cstream = ZSTD_createCStream(); + if (zcs->cstream == NULL) goto failed1; + + { size_t const CHUNKLOG_STARTING_CAPACITY = 16; + zcs->chunklog.entries = + malloc(sizeof(chunklogEntry_t) * CHUNKLOG_STARTING_CAPACITY); + if (zcs->chunklog.entries == NULL) goto failed2; + zcs->chunklog.capacity = CHUNKLOG_STARTING_CAPACITY; + } + + return zcs; + +failed2: + ZSTD_freeCStream(zcs->cstream); +failed1: + free(zcs); + return NULL; +} + +size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs) +{ + if (zcs == NULL) return 0; /* support free on NULL */ + ZSTD_freeCStream(zcs->cstream); + free(zcs->chunklog.entries); + free(zcs); + + return 0; +} + +size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, + int compressionLevel, + U32 maxChunkSize) +{ + zcs->chunklog.size = 0; + zcs->chunkCSize = 0; + zcs->chunkDSize = 0; + + if (maxChunkSize > ZSTD_SEEKABLE_MAX_CHUNK_DECOMPRESSED_SIZE) { + return ERROR(compressionParameter_unsupported); + } + + zcs->maxChunkSize = maxChunkSize + ? maxChunkSize + : ZSTD_SEEKABLE_MAX_CHUNK_DECOMPRESSED_SIZE; + + zcs->checksumFlag = 0; + if (zcs->checksumFlag) { + XXH64_reset(&zcs->xxhState, 0); + } + + return ZSTD_initCStream(zcs->cstream, compressionLevel); +} + +static size_t ZSTD_seekable_logChunk(ZSTD_seekable_CStream* zcs) +{ + if (zcs->chunklog.size == ZSTD_SEEKABLE_MAXCHUNKS) + return ERROR(chunkIndex_tooLarge); + + zcs->chunklog.entries[zcs->chunklog.size] = (chunklogEntry_t) + { + .cSize = zcs->chunkCSize, + .dSize = zcs->chunkDSize, + }; + if (zcs->checksumFlag) + zcs->chunklog.entries[zcs->chunklog.size].checksum = + XXH64_digest(&zcs->xxhState) & 0xFFFFFFFFU; + + zcs->chunklog.size++; + if (zcs->chunklog.size == zcs->chunklog.capacity) { + size_t const newCapacity = zcs->chunklog.capacity * 2; + chunklogEntry_t* const newEntries = realloc(zcs->chunklog.entries, + sizeof(chunklogEntry_t) * newCapacity); + + if (newEntries == NULL) return ERROR(memory_allocation); + + zcs->chunklog.entries = newEntries; + zcs->chunklog.capacity = newCapacity; + } + + return 0; +} + +size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output) +{ + size_t const prevOutPos = output->pos; + size_t ret = ZSTD_endStream(zcs->cstream, output); + + zcs->chunkCSize += output->pos - prevOutPos; + + /* need to flush before doing the rest */ + if (ret) return ret; + + /* frame done */ + + /* store the chunk data for later */ + ret = ZSTD_seekable_logChunk(zcs); + if (ret) return ret; + + /* reset for the next chunk */ + zcs->chunkCSize = 0; + zcs->chunkDSize = 0; + + ZSTD_resetCStream(zcs->cstream, 0); + if (zcs->checksumFlag) + XXH64_reset(&zcs->xxhState, 0); + + return 0; +} + +size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input) +{ + const BYTE* const inBase = (const BYTE*) input->src + input->pos; + size_t inLen = input->size - input->pos; + + inLen = MIN(inLen, (size_t)(zcs->maxChunkSize - zcs->chunkDSize)); + + if (inLen > 0) { + ZSTD_inBuffer inTmp = { inBase, inLen, 0 }; + size_t const prevOutPos = output->pos; + + size_t const ret = ZSTD_compressStream(zcs->cstream, output, &inTmp); + + if (zcs->checksumFlag) { + XXH64_update(&zcs->xxhState, inBase, inTmp.pos); + } + + zcs->chunkCSize += output->pos - prevOutPos; + zcs->chunkDSize += inTmp.pos; + + input->pos += inTmp.pos; + + if (ZSTD_isError(ret)) return ret; + } + + if (zcs->maxChunkSize == zcs->chunkDSize) { + size_t const ret = ZSTD_seekable_endChunk(zcs, output); + if (ZSTD_isError(ret)) return ret; + } + + return (size_t)(zcs->maxChunkSize - zcs->chunkDSize); +} + +static size_t ZSTD_seekable_seekTableSize(ZSTD_seekable_CStream* zcs) +{ + size_t const sizePerChunk = 8 + (zcs->checksumFlag?4:0); + size_t const seekTableLen = ZSTD_skippableHeaderSize + + sizePerChunk * zcs->chunklog.size + + ZSTD_seekTableFooterSize; + + return seekTableLen; +} + +static size_t ZSTD_seekable_writeSeekTable(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output) +{ + BYTE* op = (BYTE*) output->dst; + + /* repurpose + * zcs->chunkDSize: the current index in the table and + * zcs->chunkCSize: the amount of the table written so far */ + + size_t const sizePerChunk = 8 + (zcs->checksumFlag?4:0); + size_t const seekTableLen = ZSD_seekable_seekTableSize(zcs); + + if (zcs->chunkCSize == 0) { + if (output->size - output->pos < 4) return seekTableLen - zcs->chunkCSize; + MEM_writeLE32(op + output->pos, ZSTD_MAGIC_SKIPPABLE_START); + output->pos += 4; + zcs->chunkCSize += 4; + } + if (zcs->chunkCSize == 4) { + if (output->size - output->pos < 4) return seekTableLen - zcs->chunkCSize; + MEM_writeLE32(op + output->pos, seekTableLen - ZSTD_skippableHeaderSize); + output->pos += 4; + zcs->chunkCSize += 4; + } + + while (zcs->chunkDSize < zcs->chunklog.size) { + if (output->size - output->pos < sizePerChunk) return seekTableLen - zcs->chunkCSize; + MEM_writeLE32(op + output->pos + 0, zcs->chunklog.entries[zcs->chunkDSize].cSize); + MEM_writeLE32(op + output->pos + 4, zcs->chunklog.entries[zcs->chunkDSize].dSize); + if (zcs->checksumFlag) { + MEM_writeLE32(op + output->pos + 8, zcs->chunklog.entries[zcs->chunkDSize].checksum); + } + output->pos += sizePerChunk; + zcs->chunkCSize += sizePerChunk; + zcs->chunkDSize++; + } + + if (output->size - output->pos < ZSTD_seekTableFooterSize) return seekTableLen - zcs->chunkCSize; + MEM_writeLE32(op + output->pos, zcs->chunklog.size); + { BYTE sfd = 0; + sfd |= (zcs->checksumFlag) << 7; + + op[output->pos + 4] = sfd; + } + MEM_writeLE32(op + output->pos + 5, ZSTD_SEEKABLE_MAGICNUMBER); + + output->pos += ZSTD_seekTableFooterSize; + zcs->chunkCSize += ZSTD_seekTableFooterSize; + + if (zcs->chunkCSize != seekTableLen) return ERROR(GENERIC); + return 0; +} + +size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output) +{ + if (zcs->chunkDSize) { + const size_t endChunk = ZSTD_seekable_endChunk(zcs, output); + /* return an accurate size hint */ + if (endChunk) return endChunk + ZSTD_seekable_seekTableLen(zcs); + } + + return ZSTD_seekable_writeSeekTable(zcs, output); +} diff --git a/lib/decompress/zstdseek_decompress.c b/lib/decompress/zstdseek_decompress.c new file mode 100644 index 000000000..e76b4c612 --- /dev/null +++ b/lib/decompress/zstdseek_decompress.c @@ -0,0 +1,347 @@ +/* + * Copyright (c) 2017-present, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. An additional grant + * of patent rights can be found in the PATENTS file in the same directory. + */ + +#include /* malloc, free */ + +#define XXH_STATIC_LINKING_ONLY +#include "xxhash.h" + +#include "zstd_internal.h" /* includes zstd.h */ +#include "seekable.h" + +typedef struct { + U64 cOffset; + U64 dOffset; + U32 checksum; +} seekEntry_t; + +typedef struct { + seekEntry_t* entries; + size_t tableLen; + + int checksumFlag; +} seekTable_t; + +static U32 ZSTD_seekable_offsetToChunk(const seekTable_t* table, U64 pos) +{ + U32 lo = 0; + U32 hi = table->tableLen; + + while (lo + 1 < hi) { + U32 mid = lo + ((hi - lo) >> 1); + if (table->entries[mid].dOffset <= pos) { + lo = mid; + } else { + hi = mid; + } + } + return lo; +} + +enum ZSTD_seekable_DStream_stage { + zsds_init, + zsds_seek, + zsds_decompress, + zsds_done, +}; + +struct ZSTD_seekable_DStream_s { + ZSTD_DStream* dstream; + seekTable_t seekTable; + + U32 curChunk; + U64 compressedOffset; + U64 decompressedOffset; + + U64 targetStart; + U64 targetEnd; + + U64 nextSeek; + + enum ZSTD_seekable_DStream_stage stage; + + XXH64_state_t xxhState; +}; + +ZSTD_seekable_DStream* ZSTD_seekable_createDStream(void) +{ + ZSTD_seekable_DStream* zds = malloc(sizeof(ZSTD_seekable_DStream)); + + if (zds == NULL) return NULL; + + memset(zds, 0, sizeof(*zds)); + + zds->dstream = ZSTD_createDStream(); + if (zds->dstream == NULL) { + free(zds); + return NULL; + } + + return zds; +} + +size_t ZSTD_seekable_freeDStream(ZSTD_seekable_DStream* zds) +{ + if (zds == NULL) return 0; + ZSTD_freeDStream(zds->dstream); + free(zds->seekTable.entries); + free(zds); + + return 0; +} + +size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, size_t srcSize) +{ + const BYTE* ip = (const BYTE*)src + srcSize; + + U32 numChunks; + int checksumFlag; + + U32 sizePerEntry; + + if (srcSize < ZSTD_seekTableFooterSize) + return ZSTD_seekTableFooterSize; + + if (MEM_readLE32(ip - 4) != ZSTD_SEEKABLE_MAGICNUMBER) { + return ERROR(prefix_unknown); + } + + { + BYTE const sfd = ip[-5]; + checksumFlag = sfd >> 7; + + numChunks = MEM_readLE32(ip-9); + + sizePerEntry = 8 + (checksumFlag?4:0); + } + + { U32 const tableSize = sizePerEntry * numChunks; + U32 const frameSize = tableSize + ZSTD_seekTableFooterSize + ZSTD_skippableHeaderSize; + + const BYTE* base = ip - frameSize; + + if (srcSize < frameSize) return frameSize; + + if ((MEM_readLE32(base) & 0xFFFFFFF0U) != ZSTD_MAGIC_SKIPPABLE_START) { + return ERROR(prefix_unknown); + } + if (MEM_readLE32(base+4) + ZSTD_skippableHeaderSize != frameSize) { + return ERROR(prefix_unknown); + } + + { /* Allocate an extra entry at the end so that we can do size + * computations on the last element without special case */ + seekEntry_t* entries = malloc(sizeof(seekEntry_t) * (numChunks + 1)); + const BYTE* tableBase = base + ZSTD_skippableHeaderSize; + + U32 idx; + size_t pos; + + U64 cOffset = 0; + U64 dOffset = 0; + + if (!entries) { + free(entries); + return ERROR(memory_allocation); + } + + for (idx = 0, pos = 0; idx < numChunks; idx++) { + entries[idx].cOffset = cOffset; + entries[idx].dOffset = dOffset; + + cOffset += MEM_readLE32(tableBase + pos); pos += 4; + dOffset += MEM_readLE32(tableBase + pos); pos += 4; + if (checksumFlag) { + entries[idx].checksum = MEM_readLE32(tableBase + pos); + pos += 4; + } + } + entries[numChunks].cOffset = cOffset; + entries[numChunks].dOffset = dOffset; + + zds->seekTable.entries = entries; + zds->seekTable.tableLen = numChunks; + zds->seekTable.checksumFlag = checksumFlag; + return 0; + } + } +} + +size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, U64 rangeStart, U64 rangeEnd) +{ + /* restrict range to the end of the file, and not before the range start */ + rangeEnd = MIN(rangeEnd, zds->seekTable.entries[zds->seekTable.tableLen].dOffset); + rangeEnd = MAX(rangeEnd, rangeStart); + + zds->targetStart = rangeStart; + zds->targetEnd = rangeEnd; + zds->stage = zsds_seek; + + /* force a seek first */ + zds->curChunk = (U32) -1; + zds->compressedOffset = (U64) -1; + zds->decompressedOffset = (U64) -1; + + if (zds->seekTable.checksumFlag) { + XXH64_reset(&zds->xxhState, 0); + } + + { const size_t ret = ZSTD_initDStream(zds->dstream); + if (ZSTD_isError(ret)) return ret; } + return 0; +} + +U64 ZSTD_seekable_getSeekOffset(ZSTD_seekable_DStream* zds) +{ + return zds->nextSeek; +} + +size_t ZSTD_seekable_updateOffset(ZSTD_seekable_DStream* zds, U64 offset) +{ + if (zds->stage != zsds_seek) { + return ERROR(stage_wrong); + } + if (offset != zds->nextSeek) { + return ERROR(needSeek); + } + + zds->stage = zsds_decompress; + zds->compressedOffset = offset; + return 0; +} + +size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input) +{ + const seekTable_t* const jt = &zds->seekTable; + while (1) { + switch (zds->stage) { + case zsds_init: + return ERROR(init_missing); + case zsds_decompress: { + BYTE* const outBase = (BYTE*)output->dst + output->pos; + size_t const outLen = output->size - output->pos; + while (zds->decompressedOffset < zds->targetStart) { + U64 const toDecompress = + zds->targetStart - zds->decompressedOffset; + size_t const prevInputPos = input->pos; + + ZSTD_outBuffer outTmp = { + .dst = outBase, + .size = (size_t)MIN((U64)outLen, toDecompress), + .pos = 0}; + + size_t const ret = + ZSTD_decompressStream(zds->dstream, &outTmp, input); + + if (ZSTD_isError(ret)) return ret; + if (ret == 0) { + /* should not happen at this stage */ + return ERROR(corruption_detected); + } + + zds->compressedOffset += input->pos - prevInputPos; + zds->decompressedOffset += outTmp.pos; + + if (zds->seekTable.checksumFlag) { + XXH64_update(&zds->xxhState, outTmp.dst, outTmp.pos); + } + + if (input->pos == input->size) { + /* need more input */ + return MIN( + ZSTD_DStreamInSize(), + (size_t)(zds->seekTable.entries[zds->curChunk + 1] + .cOffset - + zds->compressedOffset)); + } + } + + /* do actual decompression */ + { + U64 const toDecompress = + MIN(zds->targetEnd, + jt->entries[zds->curChunk + 1].dOffset) - + zds->decompressedOffset; + size_t const prevInputPos = input->pos; + + ZSTD_outBuffer outTmp = { + .dst = outBase, + .size = (size_t)MIN((U64)outLen, toDecompress), + .pos = 0}; + + size_t const ret = + ZSTD_decompressStream(zds->dstream, &outTmp, input); + + if (ZSTD_isError(ret)) return ret; + + zds->compressedOffset += input->pos - prevInputPos; + zds->decompressedOffset += outTmp.pos; + + output->pos += outTmp.pos; + + if (zds->seekTable.checksumFlag) { + XXH64_update(&zds->xxhState, outTmp.dst, outTmp.pos); + if (ret == 0) { + /* verify the checksum */ + U32 const digest = XXH64_digest(&zds->xxhState); + if (digest != jt->entries[zds->curChunk].checksum) { + return ERROR(checksum_wrong); + } + + XXH64_reset(&zds->xxhState, 0); + } + } + + if (zds->decompressedOffset == zds->targetEnd) { + /* done */ + zds->stage = zsds_done; + return 0; + } + + if (ret == 0) { + /* frame is done */ + ZSTD_resetDStream(zds->dstream); + zds->stage = zsds_seek; + break; + } + + /* need more input */ + return MIN( + ZSTD_DStreamInSize(), + (size_t)(zds->seekTable.entries[zds->curChunk + 1] + .cOffset - + zds->compressedOffset)); + } + } + case zsds_seek: { + U32 targetChunk; + if (zds->decompressedOffset < zds->targetStart || + zds->decompressedOffset >= zds->targetEnd) { + /* haven't started yet */ + targetChunk = ZSTD_seekable_offsetToChunk(jt, zds->targetStart); + } else { + targetChunk = ZSTD_seekable_offsetToChunk(jt, zds->decompressedOffset); + } + + zds->curChunk = targetChunk; + + if (zds->compressedOffset == jt->entries[targetChunk].cOffset) { + zds->stage = zsds_decompress; + break; + } + + zds->nextSeek = jt->entries[targetChunk].cOffset; + zds->decompressedOffset = jt->entries[targetChunk].dOffset; + return ERROR(needSeek); + } + case zsds_done: + return 0; + } + } +} diff --git a/lib/zstd.h b/lib/zstd.h index 6066db45e..5bbd1b742 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -776,6 +776,123 @@ ZSTDLIB_API size_t ZSTD_decompressBlock(ZSTD_DCtx* dctx, void* dst, size_t dstCa ZSTDLIB_API size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize); /**< insert block into `dctx` history. Useful for uncompressed blocks */ +/*-**************************************************************************** +* Seekable Format +* +* The seekable format splits the compressed data into a series of "chunks", +* each compressed individually so that decompression of a section in the +* middle of an archive only requires zstd to decompress at most a chunk's +* worth of extra data, instead of the entire archive. +******************************************************************************/ + +typedef struct ZSTD_seekable_CStream_s ZSTD_seekable_CStream; +typedef struct ZSTD_seekable_DStream_s ZSTD_seekable_DStream; + +/*-**************************************************************************** +* Seekable compression - HowTo +* A ZSTD_seekable_CStream object is required to tracking streaming operation. +* Use ZSTD_seekable_createCStream() and ZSTD_seekable_freeCStream() to create/ +* release resources. +* +* Streaming objects are reusable to avoid allocation and deallocation, +* to start a new compression operation call ZSTD_seekable_initCStream() on the +* compressor. +* +* Data streamed to the seekable compressor will automatically be split into +* chunks of size `maxChunkSize` (provided in ZSTD_seekable_initCStream()), +* or if none is provided, will be cut off whenver ZSTD_endChunk() is called +* or when the default maximum chunk size is reached (approximately 4GB). +* +* Use ZSTD_seekable_initCStream() to initialize a ZSTD_seekable_CStream object +* for a new compression operation. +* `maxChunkSize` indicates the size at which to automatically start a new +* seekable frame. `maxChunkSize == 0` implies the default maximum size. +* @return : a size hint for input to provide for compression, or an error code +* checkable with ZSTD_isError() +* +* Use ZSTD_seekable_compressStream() repetitively to consume input stream. +* The function will automatically update both `pos` fields. +* Note that it may not consume the entire input, in which case `pos < size`, +* and it's up to the caller to present again remaining data. +* @return : a size hint, preferred nb of bytes to use as input for next +* function call or an error code, which can be tested using +* ZSTD_isError(). +* Note 1 : it's just a hint, to help latency a little, any other +* value will work fine. +* Note 2 : size hint is guaranteed to be <= ZSTD_CStreamInSize() +* +* At any time, call ZSTD_seekable_endChunk() to end the current chunk and +* start a new one. +* +* ZSTD_endStream() will end the current chunk, and then write the seek table +* so that decompressors can efficiently find compressed chunks. +* ZSTD_endStream() may return a number > 0 if it was unable to flush all the +* necessary data to `output`. In this case, it should be called again until +* all remaining data is flushed out and 0 is returned. +******************************************************************************/ + +/*===== Seekable compressor management =====*/ +ZSTDLIB_API ZSTD_seekable_CStream* ZSTD_seekable_createCStream(void); +ZSTDLIB_API size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs); + +/*===== Seekable compression functions =====*/ +ZSTDLIB_API size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, unsigned maxChunkSize); +ZSTDLIB_API size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input); +ZSTDLIB_API size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); +ZSTDLIB_API size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); + +/*-**************************************************************************** +* Seekable decompression - HowTo +* A ZSTD_seekable_DStream object is required to tracking streaming operation. +* Use ZSTD_seekable_createDStream() and ZSTD_seekable_freeDStream() to create/ +* release resources. +* +* Streaming objects are reusable to avoid allocation and deallocation, +* to start a new compression operation call ZSTD_seekable_initDStream() on the +* compressor. +* +* Use ZSTD_seekable_loadSeekTable() to load the seek table from a file. +* `src` should point to a block of data read from the end of the file, +* i.e. `src + srcSize` should always be the end of the file. +* @return : 0 if the table was loaded successfully, or if `srcSize` was too +* small, a size hint for how much data to provide. +* An error code may also be returned, checkable with ZSTD_isError() +* +* Use ZSTD_initDStream to prepare for a new decompression operation using the +* seektable loaded with ZSTD_seekable_loadSeekTable(). +* Data in the range [rangeStart, rangeEnd) will be decompressed. +* +* Call ZSTD_seekable_decompressStream() repetitively to consume input stream. +* @return : There are a number of possible return codes for this function +* - 0, the decompression operation has completed. +* - An error code checkable with ZSTD_isError +* + If this error code is ZSTD_error_needSeek, the user should seek +* to the file position provided by ZSTD_seekable_getSeekOffset() +* and indicate this to the stream with +* ZSTD_seekable_updateOffset(), before resuming decompression +* + Otherwise, this is a regular decompression error and the input +* file is likely corrupted or the API was incorrectly used. +* - A size hint, the preferred nb of bytes to provide as input to the +* next function call to improve latency. +* +* ZSTD_seekable_getSeekOffset() and ZSTD_seekable_updateOffset() are helper +* functions to indicate where the user should seek their file stream to, when +* a different position is required to continue decompression. +* Note that ZSTD_seekable_updateOffset will error if given an offset other +* than the one requested from ZSTD_seekable_getSeekOffset(). +******************************************************************************/ + +/*===== Seekable decompressor management =====*/ +ZSTDLIB_API ZSTD_seekable_DStream* ZSTD_seekable_createDStream(void); +ZSTDLIB_API size_t ZSTD_seekable_freeDStream(ZSTD_seekable_DStream* zds); + +/*===== Seekable decompression functions =====*/ +ZSTDLIB_API size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, size_t srcSize); +ZSTDLIB_API size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, unsigned long long rangeStart, unsigned long long rangeEnd); +ZSTDLIB_API size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input); +ZSTDLIB_API unsigned long long ZSTD_seekable_getSeekOffset(ZSTD_seekable_DStream* zds); +ZSTDLIB_API size_t ZSTD_seekable_updateOffset(ZSTD_seekable_DStream* zds, unsigned long long offset); + #endif /* ZSTD_H_ZSTD_STATIC_LINKING_ONLY */ #if defined (__cplusplus) From a3b7c2260404a85d0d72db37d4a627b9c2caf058 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 11 Apr 2017 11:40:34 -0700 Subject: [PATCH 167/305] Make seekable streams work w/ small buffers, misc fixes --- doc/zstd_manual.html | 4 +- examples/seekable_compression.c | 9 ++- examples/seekable_decompression.c | 2 + lib/compress/zstdseek_compress.c | 94 +++++++++++++++++++--------- lib/decompress/zstdseek_decompress.c | 52 +++++++++------ lib/zstd.h | 4 +- 6 files changed, 110 insertions(+), 55 deletions(-) diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index a47b7142a..0fa5c5d78 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -687,6 +687,8 @@ size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSiz for a new compression operation. `maxChunkSize` indicates the size at which to automatically start a new seekable frame. `maxChunkSize == 0` implies the default maximum size. + `checksumFlag` indicates whether or not the seek table should include chunk + checksums on the uncompressed data for verification. @return : a size hint for input to provide for compression, or an error code checkable with ZSTD_isError() @@ -714,7 +716,7 @@ size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSiz

Seekable compressor management

ZSTD_seekable_CStream* ZSTD_seekable_createCStream(void);
 size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs);
 

-

Seekable compression functions

size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, unsigned maxChunkSize);
+

Seekable compression functions

size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, int checksumFlag, unsigned maxChunkSize);
 size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input);
 size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output);
 size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output);
diff --git a/examples/seekable_compression.c b/examples/seekable_compression.c
index 5ab36c3cb..f4bceb10c 100644
--- a/examples/seekable_compression.c
+++ b/examples/seekable_compression.c
@@ -68,7 +68,7 @@ static void compressFile_orDie(const char* fname, const char* outName, int cLeve
 
     ZSTD_seekable_CStream* const cstream = ZSTD_seekable_createCStream();
     if (cstream==NULL) { fprintf(stderr, "ZSTD_seekable_createCStream() error \n"); exit(10); }
-    size_t const initResult = ZSTD_seekable_initCStream(cstream, cLevel, chunkSize);
+    size_t const initResult = ZSTD_seekable_initCStream(cstream, cLevel, 1, chunkSize);
     if (ZSTD_isError(initResult)) { fprintf(stderr, "ZSTD_seekable_initCStream() error : %s \n", ZSTD_getErrorName(initResult)); exit(11); }
 
     size_t read, toRead = buffInSize;
@@ -77,15 +77,16 @@ static void compressFile_orDie(const char* fname, const char* outName, int cLeve
         while (input.pos < input.size) {
             ZSTD_outBuffer output = { buffOut, buffOutSize, 0 };
             toRead = ZSTD_seekable_compressStream(cstream, &output , &input);   /* toRead is guaranteed to be <= ZSTD_CStreamInSize() */
-            if (ZSTD_isError(toRead)) { fprintf(stderr, "ZSTD_compressStream() error : %s \n", ZSTD_getErrorName(toRead)); exit(12); }
+            if (ZSTD_isError(toRead)) { fprintf(stderr, "ZSTD_seekable_compressStream() error : %s \n", ZSTD_getErrorName(toRead)); exit(12); }
             if (toRead > buffInSize) toRead = buffInSize;   /* Safely handle case when `buffInSize` is manually changed to a value < ZSTD_CStreamInSize()*/
             fwrite_orDie(buffOut, output.pos, fout);
         }
     }
 
-    ZSTD_outBuffer output = { buffOut, buffOutSize, 0 };
     while (1) {
+        ZSTD_outBuffer output = { buffOut, buffOutSize, 0 };
         size_t const remainingToFlush = ZSTD_seekable_endStream(cstream, &output);   /* close stream */
+        if (ZSTD_isError(remainingToFlush)) { fprintf(stderr, "ZSTD_seekable_endStream() error : %s \n", ZSTD_getErrorName(remainingToFlush)); exit(13); }
         fwrite_orDie(buffOut, output.pos, fout);
         if (!remainingToFlush) break;
     }
@@ -123,4 +124,6 @@ int main(int argc, const char** argv) {
         const char* const outFileName = createOutFilename_orDie(inFileName);
         compressFile_orDie(inFileName, outFileName, 5, chunkSize);
     }
+
+    return 0;
 }
diff --git a/examples/seekable_decompression.c b/examples/seekable_decompression.c
index 97563c3de..641e0429f 100644
--- a/examples/seekable_decompression.c
+++ b/examples/seekable_decompression.c
@@ -143,6 +143,7 @@ static void decompressFile_orDie(const char* fname, unsigned startOffset, unsign
                 toRead = result;
             }
             fwrite_orDie(buffOut, output.pos, fout);
+            if (toRead > buffInSize) toRead = buffInSize;
         }
     } while (result > 0);
 
@@ -171,5 +172,6 @@ int main(int argc, const char** argv)
         unsigned const endOffset = (unsigned) atoi(argv[3]);
         decompressFile_orDie(inFilename, startOffset, endOffset);
     }
+
     return 0;
 }
diff --git a/lib/compress/zstdseek_compress.c b/lib/compress/zstdseek_compress.c
index 284724e58..f9e108afc 100644
--- a/lib/compress/zstdseek_compress.c
+++ b/lib/compress/zstdseek_compress.c
@@ -39,6 +39,8 @@ struct ZSTD_seekable_CStream_s {
     U32 maxChunkSize;
 
     int checksumFlag;
+
+    int writingSeekTable;
 };
 
 ZSTD_seekable_CStream* ZSTD_seekable_createCStream()
@@ -52,6 +54,7 @@ ZSTD_seekable_CStream* ZSTD_seekable_createCStream()
     zcs->cstream = ZSTD_createCStream();
     if (zcs->cstream == NULL) goto failed1;
 
+    /* allocate some initial space */
     {   size_t const CHUNKLOG_STARTING_CAPACITY = 16;
         zcs->chunklog.entries =
                 malloc(sizeof(chunklogEntry_t) * CHUNKLOG_STARTING_CAPACITY);
@@ -70,7 +73,7 @@ failed1:
 
 size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs)
 {
-    if (zcs == NULL) return 0; /* support free on NULL */
+    if (zcs == NULL) return 0; /* support free on null */
     ZSTD_freeCStream(zcs->cstream);
     free(zcs->chunklog.entries);
     free(zcs);
@@ -80,12 +83,14 @@ size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs)
 
 size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs,
                                  int compressionLevel,
+                                 int checksumFlag,
                                  U32 maxChunkSize)
 {
     zcs->chunklog.size = 0;
     zcs->chunkCSize = 0;
     zcs->chunkDSize = 0;
 
+    /* make sure maxChunkSize has a reasonable value */
     if (maxChunkSize > ZSTD_SEEKABLE_MAX_CHUNK_DECOMPRESSED_SIZE) {
         return ERROR(compressionParameter_unsupported);
     }
@@ -94,11 +99,13 @@ size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs,
                                 ? maxChunkSize
                                 : ZSTD_SEEKABLE_MAX_CHUNK_DECOMPRESSED_SIZE;
 
-    zcs->checksumFlag = 0;
+    zcs->checksumFlag = checksumFlag;
     if (zcs->checksumFlag) {
         XXH64_reset(&zcs->xxhState, 0);
     }
 
+    zcs->writingSeekTable = 0;
+
     return ZSTD_initCStream(zcs->cstream, compressionLevel);
 }
 
@@ -114,10 +121,13 @@ static size_t ZSTD_seekable_logChunk(ZSTD_seekable_CStream* zcs)
         };
     if (zcs->checksumFlag)
         zcs->chunklog.entries[zcs->chunklog.size].checksum =
+                /* take lower 32 bits of digest */
                 XXH64_digest(&zcs->xxhState) & 0xFFFFFFFFU;
 
     zcs->chunklog.size++;
+    /* grow the buffer if required */
     if (zcs->chunklog.size == zcs->chunklog.capacity) {
+        /* exponential size increase for constant amortized runtime */
         size_t const newCapacity = zcs->chunklog.capacity * 2;
         chunklogEntry_t* const newEntries = realloc(zcs->chunklog.entries,
                 sizeof(chunklogEntry_t) * newCapacity);
@@ -134,6 +144,7 @@ static size_t ZSTD_seekable_logChunk(ZSTD_seekable_CStream* zcs)
 size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output)
 {
     size_t const prevOutPos = output->pos;
+    /* end the frame */
     size_t ret = ZSTD_endStream(zcs->cstream, output);
 
     zcs->chunkCSize += output->pos - prevOutPos;
@@ -165,6 +176,7 @@ size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer*
 
     inLen = MIN(inLen, (size_t)(zcs->maxChunkSize - zcs->chunkDSize));
 
+    /* if we haven't finished flushing the last chunk, don't start writing a new one */
     if (inLen > 0) {
         ZSTD_inBuffer inTmp = { inBase, inLen, 0 };
         size_t const prevOutPos = output->pos;
@@ -184,8 +196,12 @@ size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer*
     }
 
     if (zcs->maxChunkSize == zcs->chunkDSize) {
+        /* log the chunk and start over */
         size_t const ret = ZSTD_seekable_endChunk(zcs, output);
         if (ZSTD_isError(ret)) return ret;
+
+        /* get the client ready for the next chunk */
+        return (size_t)zcs->maxChunkSize;
     }
 
     return (size_t)(zcs->maxChunkSize - zcs->chunkDSize);
@@ -204,62 +220,78 @@ static size_t ZSTD_seekable_seekTableSize(ZSTD_seekable_CStream* zcs)
 static size_t ZSTD_seekable_writeSeekTable(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output)
 {
     BYTE* op = (BYTE*) output->dst;
+    BYTE tmp[4]; /* so that we can work with buffers too small to write a whole word to */
 
     /* repurpose
      * zcs->chunkDSize: the current index in the table and
-     * zcs->chunkCSize: the amount of the table written so far */
+     * zcs->chunkCSize: the amount of the table written so far
+     *
+     * This function is written this way so that if it has to return early
+     * because of a small buffer, it can keep going where it left off.
+     */
 
     size_t const sizePerChunk = 8 + (zcs->checksumFlag?4:0);
-    size_t const seekTableLen = ZSD_seekable_seekTableSize(zcs);
+    size_t const seekTableLen = ZSTD_seekable_seekTableSize(zcs);
 
-    if (zcs->chunkCSize == 0) {
-        if (output->size - output->pos < 4) return seekTableLen - zcs->chunkCSize;
-        MEM_writeLE32(op + output->pos, ZSTD_MAGIC_SKIPPABLE_START);
-        output->pos += 4;
-        zcs->chunkCSize += 4;
-    }
-    if (zcs->chunkCSize == 4) {
-        if (output->size - output->pos < 4) return seekTableLen - zcs->chunkCSize;
-        MEM_writeLE32(op + output->pos, seekTableLen - ZSTD_skippableHeaderSize);
-        output->pos += 4;
-        zcs->chunkCSize += 4;
-    }
+#define st_write32(x, o)                                                       \
+    do {                                                                       \
+        if (zcs->chunkCSize < (o) + 4) {                                       \
+            size_t const lenWrite = MIN(output->size - output->pos,            \
+                                        (o) + 4 - zcs->chunkCSize);            \
+            MEM_writeLE32(tmp, (x));                                           \
+            memcpy(op + output->pos, tmp + (zcs->chunkCSize - (o)), lenWrite); \
+            zcs->chunkCSize += lenWrite;                                       \
+            output->pos += lenWrite;                                           \
+            if (lenWrite < 4) return seekTableLen - zcs->chunkCSize;           \
+        }                                                                      \
+    } while (0)
+
+    st_write32(ZSTD_MAGIC_SKIPPABLE_START, 0);
+    st_write32(seekTableLen - ZSTD_skippableHeaderSize, 4);
 
     while (zcs->chunkDSize < zcs->chunklog.size) {
-        if (output->size - output->pos < sizePerChunk) return seekTableLen - zcs->chunkCSize;
-        MEM_writeLE32(op + output->pos + 0, zcs->chunklog.entries[zcs->chunkDSize].cSize);
-        MEM_writeLE32(op + output->pos + 4, zcs->chunklog.entries[zcs->chunkDSize].dSize);
+        st_write32(zcs->chunklog.entries[zcs->chunkDSize].cSize,
+                   ZSTD_skippableHeaderSize + sizePerChunk * zcs->chunkDSize);
+        st_write32(zcs->chunklog.entries[zcs->chunkDSize].dSize,
+                   ZSTD_skippableHeaderSize + sizePerChunk * zcs->chunkDSize + 4);
         if (zcs->checksumFlag) {
-            MEM_writeLE32(op + output->pos + 8, zcs->chunklog.entries[zcs->chunkDSize].checksum);
+            st_write32(zcs->chunklog.entries[zcs->chunkDSize].checksum,
+                       ZSTD_skippableHeaderSize + sizePerChunk * zcs->chunkDSize + 8);
         }
-        output->pos += sizePerChunk;
-        zcs->chunkCSize += sizePerChunk;
+
         zcs->chunkDSize++;
     }
 
-    if (output->size - output->pos < ZSTD_seekTableFooterSize) return seekTableLen - zcs->chunkCSize;
-    MEM_writeLE32(op + output->pos, zcs->chunklog.size);
-    {   BYTE sfd = 0;
+    st_write32(zcs->chunklog.size, seekTableLen - ZSTD_seekTableFooterSize);
+
+    if (output->size - output->pos < 1) return seekTableLen - zcs->chunkCSize;
+    if (zcs->chunkCSize < seekTableLen - 4) {
+        BYTE sfd = 0;
         sfd |= (zcs->checksumFlag) << 7;
 
-        op[output->pos + 4] = sfd;
+        op[output->pos] = sfd;
+        output->pos++;
+        zcs->chunkCSize++;
     }
-    MEM_writeLE32(op + output->pos + 5, ZSTD_SEEKABLE_MAGICNUMBER);
 
-    output->pos += ZSTD_seekTableFooterSize;
-    zcs->chunkCSize += ZSTD_seekTableFooterSize;
+    st_write32(ZSTD_SEEKABLE_MAGICNUMBER, seekTableLen - 4);
 
     if (zcs->chunkCSize != seekTableLen) return ERROR(GENERIC);
     return 0;
+
+#undef st_write32
 }
 
 size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output)
 {
-    if (zcs->chunkDSize) {
+    if (!zcs->writingSeekTable && zcs->chunkDSize) {
         const size_t endChunk = ZSTD_seekable_endChunk(zcs, output);
+        if (ZSTD_isError(endChunk)) return endChunk;
         /* return an accurate size hint */
-        if (endChunk) return endChunk + ZSTD_seekable_seekTableLen(zcs);
+        if (endChunk) return endChunk + ZSTD_seekable_seekTableSize(zcs);
     }
 
+    zcs->writingSeekTable = 1;
+
     return ZSTD_seekable_writeSeekTable(zcs, output);
 }
diff --git a/lib/decompress/zstdseek_decompress.c b/lib/decompress/zstdseek_decompress.c
index e76b4c612..5e303f26b 100644
--- a/lib/decompress/zstdseek_decompress.c
+++ b/lib/decompress/zstdseek_decompress.c
@@ -28,6 +28,10 @@ typedef struct {
     int checksumFlag;
 } seekTable_t;
 
+/** ZSTD_seekable_offsetToChunk() :
+ *  Performs a binary search to find the last chunk with a decompressed offset
+ *  <= pos
+ *  @return : the chunk's index */
 static U32 ZSTD_seekable_offsetToChunk(const seekTable_t* table, U64 pos)
 {
     U32 lo = 0;
@@ -44,8 +48,9 @@ static U32 ZSTD_seekable_offsetToChunk(const seekTable_t* table, U64 pos)
     return lo;
 }
 
+/* Stream decompressor state machine stages */
 enum ZSTD_seekable_DStream_stage {
-    zsds_init,
+    zsds_init = 0,
     zsds_seek,
     zsds_decompress,
     zsds_done,
@@ -75,6 +80,7 @@ ZSTD_seekable_DStream* ZSTD_seekable_createDStream(void)
 
     if (zds == NULL) return NULL;
 
+    /* also initializes stage to zsds_init */
     memset(zds, 0, sizeof(*zds));
 
     zds->dstream = ZSTD_createDStream();
@@ -88,7 +94,7 @@ ZSTD_seekable_DStream* ZSTD_seekable_createDStream(void)
 
 size_t ZSTD_seekable_freeDStream(ZSTD_seekable_DStream* zds)
 {
-    if (zds == NULL) return 0;
+    if (zds == NULL) return 0; /* support free on null */
     ZSTD_freeDStream(zds->dstream);
     free(zds->seekTable.entries);
     free(zds);
@@ -105,6 +111,7 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src,
 
     U32 sizePerEntry;
 
+    /* footer is fixed size */
     if (srcSize < ZSTD_seekTableFooterSize)
         return ZSTD_seekTableFooterSize;
 
@@ -112,15 +119,13 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src,
         return ERROR(prefix_unknown);
     }
 
-    {
-        BYTE const sfd = ip[-5];
+    {   BYTE const sfd = ip[-5];
         checksumFlag = sfd >> 7;
-
-        numChunks = MEM_readLE32(ip-9);
-
-        sizePerEntry = 8 + (checksumFlag?4:0);
     }
 
+    numChunks = MEM_readLE32(ip-9);
+    sizePerEntry = 8 + (checksumFlag?4:0);
+
     {   U32 const tableSize = sizePerEntry * numChunks;
         U32 const frameSize = tableSize + ZSTD_seekTableFooterSize + ZSTD_skippableHeaderSize;
 
@@ -151,6 +156,7 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src,
                 return ERROR(memory_allocation);
             }
 
+            /* compute cumulative positions */
             for (idx = 0, pos = 0; idx < numChunks; idx++) {
                 entries[idx].cOffset = cOffset;
                 entries[idx].dOffset = dOffset;
@@ -175,7 +181,8 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src,
 
 size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, U64 rangeStart, U64 rangeEnd)
 {
-    /* restrict range to the end of the file, and not before the range start */
+    /* restrict range to the end of the file, of non-negative size */
+    rangeStart = MIN(rangeStart, zds->seekTable.entries[zds->seekTable.tableLen].dOffset);
     rangeEnd = MIN(rangeEnd, zds->seekTable.entries[zds->seekTable.tableLen].dOffset);
     rangeEnd = MAX(rangeEnd, rangeStart);
 
@@ -192,6 +199,8 @@ size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, U64 rangeStart, U64
         XXH64_reset(&zds->xxhState, 0);
     }
 
+    if (rangeStart == rangeEnd) zds->stage = zsds_done;
+
     {   const size_t ret = ZSTD_initDStream(zds->dstream);
         if (ZSTD_isError(ret)) return ret; }
     return 0;
@@ -222,7 +231,7 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer
     while (1) {
         switch (zds->stage) {
         case zsds_init:
-            return ERROR(init_missing);
+            return ERROR(init_missing); /* ZSTD_seekable_initDStream should be called first */
         case zsds_decompress: {
             BYTE* const outBase = (BYTE*)output->dst + output->pos;
             size_t const outLen = output->size - output->pos;
@@ -248,7 +257,7 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer
                 zds->compressedOffset += input->pos - prevInputPos;
                 zds->decompressedOffset += outTmp.pos;
 
-                if (zds->seekTable.checksumFlag) {
+                if (jt->checksumFlag) {
                     XXH64_update(&zds->xxhState, outTmp.dst, outTmp.pos);
                 }
 
@@ -256,7 +265,7 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer
                     /* need more input */
                     return MIN(
                             ZSTD_DStreamInSize(),
-                            (size_t)(zds->seekTable.entries[zds->curChunk + 1]
+                            (size_t)(jt->entries[zds->curChunk + 1]
                                              .cOffset -
                                      zds->compressedOffset));
                 }
@@ -285,11 +294,11 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer
 
                 output->pos += outTmp.pos;
 
-                if (zds->seekTable.checksumFlag) {
+                if (jt->checksumFlag) {
                     XXH64_update(&zds->xxhState, outTmp.dst, outTmp.pos);
                     if (ret == 0) {
                         /* verify the checksum */
-                        U32 const digest = XXH64_digest(&zds->xxhState);
+                        U32 const digest = XXH64_digest(&zds->xxhState) & 0xFFFFFFFFU;
                         if (digest != jt->entries[zds->curChunk].checksum) {
                             return ERROR(checksum_wrong);
                         }
@@ -306,17 +315,21 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer
 
                 if (ret == 0) {
                     /* frame is done */
+                    /* make sure this lines up with the expected frame border */
+                    if (zds->decompressedOffset !=
+                                jt->entries[zds->curChunk + 1].dOffset ||
+                        zds->compressedOffset !=
+                                jt->entries[zds->curChunk + 1].cOffset)
+                        return ERROR(corruption_detected);
                     ZSTD_resetDStream(zds->dstream);
                     zds->stage = zsds_seek;
                     break;
                 }
 
                 /* need more input */
-                return MIN(
-                        ZSTD_DStreamInSize(),
-                        (size_t)(zds->seekTable.entries[zds->curChunk + 1]
-                                         .cOffset -
-                                 zds->compressedOffset));
+                return MIN(ZSTD_DStreamInSize(), (size_t)(
+                        jt->entries[zds->curChunk + 1].cOffset -
+                        zds->compressedOffset));
             }
         }
         case zsds_seek: {
@@ -338,6 +351,7 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer
 
             zds->nextSeek = jt->entries[targetChunk].cOffset;
             zds->decompressedOffset = jt->entries[targetChunk].dOffset;
+            /* signal to user that a seek is required */
             return ERROR(needSeek);
         }
         case zsds_done:
diff --git a/lib/zstd.h b/lib/zstd.h
index 5bbd1b742..748a98270 100644
--- a/lib/zstd.h
+++ b/lib/zstd.h
@@ -807,6 +807,8 @@ typedef struct ZSTD_seekable_DStream_s ZSTD_seekable_DStream;
 *  for a new compression operation.
 *  `maxChunkSize` indicates the size at which to automatically start a new
 *  seekable frame.  `maxChunkSize == 0` implies the default maximum size.
+*  `checksumFlag` indicates whether or not the seek table should include chunk
+*  checksums on the uncompressed data for verification.
 *  @return : a size hint for input to provide for compression, or an error code
 *            checkable with ZSTD_isError()
 *
@@ -836,7 +838,7 @@ ZSTDLIB_API ZSTD_seekable_CStream* ZSTD_seekable_createCStream(void);
 ZSTDLIB_API size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs);
 
 /*===== Seekable compression functions =====*/
-ZSTDLIB_API size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, unsigned maxChunkSize);
+ZSTDLIB_API size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, int checksumFlag, unsigned maxChunkSize);
 ZSTDLIB_API size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input);
 ZSTDLIB_API size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output);
 ZSTDLIB_API size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output);

From 45f3bc4801ebb27dc118346a7bc761b715e478b9 Mon Sep 17 00:00:00 2001
From: Sean Purcell 
Date: Tue, 11 Apr 2017 13:50:05 -0700
Subject: [PATCH 168/305] Add format specification

---
 doc/zstd_seekable_compression_format.md | 107 ++++++++++++++++++++++++
 lib/decompress/zstdseek_decompress.c    |   5 ++
 2 files changed, 112 insertions(+)
 create mode 100644 doc/zstd_seekable_compression_format.md

diff --git a/doc/zstd_seekable_compression_format.md b/doc/zstd_seekable_compression_format.md
new file mode 100644
index 000000000..fd0593f9a
--- /dev/null
+++ b/doc/zstd_seekable_compression_format.md
@@ -0,0 +1,107 @@
+# Zstandard Seekable Format
+
+### Notices
+
+Copyright (c) 2017-present Facebook, Inc.
+
+Permission is granted to copy and distribute this document
+for any purpose and without charge,
+including translations into other languages
+and incorporation into compilations,
+provided that the copyright notice and this notice are preserved,
+and that any substantive changes or deletions from the original
+are clearly marked.
+Distribution of this document is unlimited.
+
+### Version
+0.1.0 (11/04/17)
+
+## Introduction
+This document defines a format for compressed data to be stored so that subranges of the data can be efficiently decompressed without requiring the entire document to be decompressed.
+This is done by splitting up the input data into chunks,
+each of which are compressed independently,
+and so can be decompressed independently.
+Decompression then takes advantage of a provided 'seek table', which allows the decompressor to immediately jump to the desired data.  This is done in a way that is compatible with the original Zstandard format by placing the seek table in a Zstandard skippable frame.
+
+### Overall conventions
+In this document:
+- square brackets i.e. `[` and `]` are used to indicate optional fields or parameters.
+- the naming convention for identifiers is `Mixed_Case_With_Underscores`
+- All numeric fields are little-endian unless specified otherwise
+
+## Format
+
+The format consists of a number of chunks (Zstandard compressed frames and skippable frames), followed by a final skippable frame at the end containing the seek table.
+
+### Seek Table Format
+The structure of the seek table frame is as follows:
+
+|`Skippable_Magic_Number`|`Frame_Size`|`[Seek_Table_Entries]`|`Seek_Table_Footer`|
+|------------------------|------------|----------------------|-------------------|
+| 4 bytes                | 4 bytes    | 8-12 bytes each      | 9 bytes           |
+
+__`Skippable_Magic_Number`__
+
+Value : 0x184D2A5?, which means any value from 0x184D2A50 to 0x184D2A5F.
+All 16 values are valid to identify a skippable frame.
+This is for compatibility with [Zstandard skippable frames].
+
+__`Frame_Size`__
+
+The total size of the skippable frame, not including the `Skippable_Magic_Number` or `Frame_Size`.  This is for compatibility with [Zstandard skippable frames].
+
+[Zstandard skippable frames]: https://github.com/facebook/zstd/blob/master/doc/zstd_compression_format.md#skippable-frames
+
+#### `Seek_Table_Footer`
+The seek table footer format is as follows:
+
+|`Number_Of_Chunks`|`Seek_Table_Descriptor`|`Seekable_Magic_Number`|
+|------------------|-----------------------|-----------------------|
+| 4 bytes          | 1 byte                | 4 bytes               |
+
+__`Number_Of_Chunks`__
+
+The number of stored chunks in the data.
+
+__`Seek_Table_Descriptor`__
+
+A bitfield describing the format of the seek table.
+
+| Bit number | Field name                |
+| ---------- | ----------                |
+| 7          | `Checksum_Flag`           |
+| 6-2        | `Reserved_Bits`           |
+| 1-0        | `Unused_Bits`             |
+
+While only `Checksum_Flag` currently exists, there are 7 other bits in this field that can be used for future changes to the format,
+for example the addition of inline dictionaries.
+
+__`Checksum_Flag`__
+
+If the checksum flag is set, each of the seek table entries contains a 4 byte checksum of the uncompressed data contained in its chunk.
+
+`Reserved_Bits` are not currently used but may be used in the future for breaking changes, so a compliant decoder should ensure they are set to 0.  `Unused_Bits` may be used in the future for non-breaking changes, so a compliant decoder should not interpret these bits.
+
+#### __`Seek_Table_Entries`__
+
+`Seek_Table_Entries` consists of `Number_Of_Chunks` (one for each chunk in the data, not including the seek table frame) entries of the following form, in sequence:
+
+|`Compressed_Size`|`Decompressed_Size`|`[Checksum]`|
+|-----------------|-------------------|------------|
+| 4 bytes         | 4 bytes           | 4 bytes    |
+
+__`Compressed_Size`__
+
+The compressed size of the chunk.
+The cumulative sum of the `Compressed_Size` fields of chunks `0` to `i` gives the offset in the compressed file of chunk `i+1`.
+
+__`Decompressed_Size`__
+
+The size of the decompressed data contained in the chunk.  For skippable or otherwise empty frames, this value is 0.
+
+__`Checksum`__
+
+Only present if `Checksum_Flag` is set in the `Seek_Table_Descriptor`.  Value : the least significant 32 bits of the XXH64 digest of the uncompressed data, stored in little-endian format.
+
+## Version Changes
+- 0.1.0: initial version
diff --git a/lib/decompress/zstdseek_decompress.c b/lib/decompress/zstdseek_decompress.c
index 5e303f26b..92901befa 100644
--- a/lib/decompress/zstdseek_decompress.c
+++ b/lib/decompress/zstdseek_decompress.c
@@ -121,6 +121,11 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src,
 
     {   BYTE const sfd = ip[-5];
         checksumFlag = sfd >> 7;
+
+        /* check reserved bits */
+        if ((checksumFlag >> 2) & 0x1f) {
+            return ERROR(corruption_detected);
+        }
     }
 
     numChunks = MEM_readLE32(ip-9);

From b13da709e854f8639061da48c2d58fa1ba4ded54 Mon Sep 17 00:00:00 2001
From: Sean Purcell 
Date: Tue, 11 Apr 2017 13:53:43 -0700
Subject: [PATCH 169/305] Fixes

---
 doc/README.md        |  2 ++
 doc/zstd_manual.html | 14 +++++++++++---
 examples/Makefile    |  2 +-
 3 files changed, 14 insertions(+), 4 deletions(-)

diff --git a/doc/README.md b/doc/README.md
index 47cfe3617..0aee04280 100644
--- a/doc/README.md
+++ b/doc/README.md
@@ -17,4 +17,6 @@ __`zstd_manual.html`__ : Documentation on the functions found in `zstd.h`.
 See [http://zstd.net/zstd_manual.html](http://zstd.net/zstd_manual.html) for
 the manual released with the latest official `zstd` release.
 
+__`zstd_seekable_compression_format.md`__ : This document defines the Zstandard
+format for seekable compression.
 
diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html
index 0fa5c5d78..c7017d12c 100644
--- a/doc/zstd_manual.html
+++ b/doc/zstd_manual.html
@@ -500,14 +500,22 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v
 

Advanced streaming functions


 
 

Advanced Streaming compression functions

ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem);
+size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);   /**< size of CStream is variable, depending primarily on compression level */
 size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize);   /**< pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
 size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
 size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize,
                                              ZSTD_parameters params, unsigned long long pledgedSrcSize);  /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */
 size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict);  /**< note : cdict will just be referenced, and must outlive compression session */
-size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize);  /**< re-use compression parameters from previous init; skip dictionary loading stage; zcs must be init at least once before. note: pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
-size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);
 

+
size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize);
+

start a new compression job, using same parameters from previous job. + This is typically useful to skip dictionary loading stage, since it will re-use it in-place.. + Note that zcs must be init at least once before using ZSTD_resetCStream(). + pledgedSrcSize==0 means "srcSize unknown". + If pledgedSrcSize > 0, its value must be correct, as it will be written in header, and controlled at the end. + @return : 0, or an error code (which can be tested using ZSTD_isError()) +


+

Advanced Streaming decompression functions

typedef enum { DStream_p_maxWindowSize } ZSTD_DStreamParameter_e;
 ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem);
 size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
@@ -556,7 +564,7 @@ size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds);
 size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel);
 size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */
 size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**<  note: if pledgedSrcSize can be 0, indicating unknown size.  if it is non-zero, it must be accurate.  for 0 size frames, use compressBegin_advanced */
-size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: if pledgedSrcSize can be 0, indicating unknown size.  if it is non-zero, it must be accurate.  for 0 size frames, use compressBegin_advanced */
+size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: fail if cdict==NULL. pledgedSrcSize can be 0, indicating unknown size.  For 0 size frames, use compressBegin_advanced */
 size_t ZSTD_compressContinue(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
 size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
 

diff --git a/examples/Makefile b/examples/Makefile index 88cc10d24..2f649eaef 100644 --- a/examples/Makefile +++ b/examples/Makefile @@ -19,7 +19,7 @@ all: simple_compression simple_decompression \ dictionary_compression dictionary_decompression \ streaming_compression streaming_decompression \ multiple_streaming_compression \ - seekable_compression + seekable_compression seekable_decompression simple_compression : simple_compression.c $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ From d048fefef72d9664a6ca90ae62ee7a07eef3fcba Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 11 Apr 2017 14:38:56 -0700 Subject: [PATCH 170/305] Move seekable format content to /contrib --- contrib/seekable_format/examples/.gitignore | 2 + contrib/seekable_format/examples/Makefile | 35 +++++ .../examples/seekable_compression | Bin 0 -> 2557127 bytes .../examples}/seekable_compression.c | 2 + .../examples}/seekable_decompression.c | 2 + contrib/seekable_format/zstd_seekable.h | 140 ++++++++++++++++++ .../zstd_seekable_compression_format.md | 0 .../seekable_format}/zstdseek_compress.c | 3 +- .../seekable_format}/zstdseek_decompress.c | 3 +- doc/README.md | 3 - doc/zstd_manual.html | 109 -------------- examples/.gitignore | 2 - examples/Makefile | 9 +- lib/common/seekable.h | 23 --- lib/zstd.h | 119 --------------- 15 files changed, 186 insertions(+), 266 deletions(-) create mode 100644 contrib/seekable_format/examples/.gitignore create mode 100644 contrib/seekable_format/examples/Makefile create mode 100755 contrib/seekable_format/examples/seekable_compression rename {examples => contrib/seekable_format/examples}/seekable_compression.c (99%) rename {examples => contrib/seekable_format/examples}/seekable_decompression.c (99%) create mode 100644 contrib/seekable_format/zstd_seekable.h rename {doc => contrib/seekable_format}/zstd_seekable_compression_format.md (100%) rename {lib/compress => contrib/seekable_format}/zstdseek_compress.c (99%) rename {lib/decompress => contrib/seekable_format}/zstdseek_decompress.c (99%) delete mode 100644 lib/common/seekable.h diff --git a/contrib/seekable_format/examples/.gitignore b/contrib/seekable_format/examples/.gitignore new file mode 100644 index 000000000..4ded45619 --- /dev/null +++ b/contrib/seekable_format/examples/.gitignore @@ -0,0 +1,2 @@ +seekable_compression +seekable_decompression diff --git a/contrib/seekable_format/examples/Makefile b/contrib/seekable_format/examples/Makefile new file mode 100644 index 000000000..c560eb075 --- /dev/null +++ b/contrib/seekable_format/examples/Makefile @@ -0,0 +1,35 @@ +# ################################################################ +# Copyright (c) 2017-present, Facebook, Inc. +# All rights reserved. +# +# This source code is licensed under the BSD-style license found in the +# LICENSE file in the root directory of this source tree. An additional grant +# of patent rights can be found in the PATENTS file in the same directory. +# ################################################################ + +# This Makefile presumes libzstd is built, using `make` in / or /lib/ + +LDFLAGS += -L../../../lib/ -lzstd +CPPFLAGS += -I../ -I../../../lib -I../../../lib/common + +CFLAGS = -O3 +CFLAGS += -g + +SEEKABLE_OBJS = ../zstdseek_compress.c ../zstdseek_decompress.c + +.PHONY: default all clean test + +default: all + +all: seekable_compression seekable_decompression + +seekable_compression : seekable_compression.c + $(CC) $(CPPFLAGS) $(CFLAGS) $(SEEKABLE_OBJS) $^ $(LDFLAGS) -o $@ + +seekable_decompression : seekable_decompression.c + $(CC) $(CPPFLAGS) $(CFLAGS) $(SEEKABLE_OBJS) $^ $(LDFLAGS) -o $@ + +clean: + @rm -f core *.o tmp* result* *.zst \ + seekable_compression seekable_decompression + @echo Cleaning completed diff --git a/contrib/seekable_format/examples/seekable_compression b/contrib/seekable_format/examples/seekable_compression new file mode 100755 index 0000000000000000000000000000000000000000..88e1d2949b157168b81cc7c090cc29dd439dd425 GIT binary patch literal 2557127 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

@3hU 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

`-evC*HutB?YDj=}8j`$HJd$f8s7i&Y5y6j2cc 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⋙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

    WO&T%e1ybc{ zkG{ zGtKl9Zh(2i%(o5gaL+mpJV)~e?UfP|~LTQ3*kb78mhKWc7DvQ)q1kpxXAeHO*SOPn_p}jRzgFx;``O&JPSv&Vqz-iD{ z#}XK;8{R$ck<_hWHD-~ISDCbwkJE1l_g0^eICO}By`AvICbX{wFfnCxgi&>4jeXoI1JLvgU`hg*WFjP*J(JAffJ4kg53g3F z3pN7Maxf$aW^7(U-5Sp(E(88we3NLJ87afcc7p;#I!8dMV8jrVjL~!U@QfU=2L?4+ zcX)aJ+=OQ+3@!1Q+a6bLXcUtk$Hw7MA^YjL@)#MkgcUtQ!?XRwM@$*sBgYsdTE``z zZm=fT+AhoKstrL->Jl^Lc8sxlth@5!DJ=ccOjwvD;X+~aoflv*a%d=x1BP&BAIvzD z%Q!PZct4)p=oDek0?=lhdtBS4m2vtIiVo8gLpa_;-U?SLINpwsWp3k}sC5hyF(~K< zyN3GbE@c%PF^iP@jAX>v%+lISa3;(G4OcH@^m$;l5~u}VyBR!e8z?`}>O%Odop)u- z`U|6u$-S+S=|fO}=jOFwf+dpE5vrjEEH}m_m}3HEva!nsAp zf#6Agc}(_H%o@1C7O2*6JRjO7;>C<+&~VxzDaErZG0z+;E0#7a5VMbs?U5kXz+G^Xx}FcLB^-J*nwyF&*TCJ}qV z+!aAB724a?%wVK3j3%}uHk7WUViLkq9tHuAC2Jy=C1F8C9O%N7&Pp>Xmq7?YjQdK@ z?`E=qYsOASLTRF#(OSU(J9%uGD}6~l@j)<@;fOGA_DXvtR!DKW>2jPG7u1;Q8}^yG z89C#V6pCmu8~x#MgR#R5b%n9Q`#U87wBpH3FM2u-ZZuMdcIb!}as;8a!nTJIAB78S z8Q6n5Iol45PO*<{UzMJyi6lmfqVG8#jb7vI0*$38B}8%Ig#u8>h0|#AQE1#fL@icI z?TNp{puI?6*V0voo^%4v8gN?lh7HbJP^`8 zh?We&m>6*Nl{N3d;K zH^Pi(ai*;$x78@P|*+p=qattq2jU8f$I3}#ZK}2cY_0qF|GI8D9s1;>+ z%q=01xmkw48_jkU5A+yo_8JQDK!>W>@HW~G2&p8uOST%rwuRdYlSM)&XeAy)Cu#~= zI=I?9#2{nXfU{9wu@;rWMrXtBlTmTlRMa04R3S zsR3IwhvbFCwB)eqq$@PhHk}c(n!ysu9TMaC<)A6+BehR45(aje(}6XhOA*Vpmn`WW zaW2)4OK5G}s`3 zM$9*%D^_&O3e}ksU7ZkFbGfg0*e&=$w=B_t8(i@Q!D7ovx4FP}ZyPBfYE{;5crP}v z5IaIh@@%Q@b0a&okpLMKRTR!j%4u zV$=eP1A`Z~p~uK(KLHYUtBeVPY6W#yFUd|9d%le0kXkV2WrRH~^u}6?kRfJZ^W=kN zhxLq@W}TI?1iTuxPc{%6zA>#Ba}y}@g$^Jwq(#JMU`m=zbzV;BTTA$`NnrxY?cx1Q z{IPBbI|Joz(tGW6C?=wOm<(f`ulgh-rha2hL`@UvHDDumPO-`(JnUn2>4Z=c7EPH$ zoz9|jpo@iNR3(*|*H+1n4H?p*3D<3>q|r=u9mVE=WM&N-SLTM2Z=AiHK#>WFT-_3@ z9EmcDy^nktAj1_lU!)S$jiD;5DVK)~21dvYo~9rKKQjKVTxbj7qjNMjq}-&H@G^rL zUxJ__QbX(r-(*V%?jOiBwW@y1G$Yh#HqM1WK=E#sW_A${Dj)2(7AB4RDS{3U+R^%K&Ioo3(&B z>8C{RhP`{K8>bezx}&$9IITtvBUb=va*?9Jem{SJL6)u9vd|H-9EjMmk(PP`yM(zY zeTGd#w>fLPi0Fo9Yg(5IPdkL=g`qSYj=pdU)7c~T(?Mf&jIHOLIol#1Z*wx8%VagX z+d^sERp4|Sev%^`NRBkk(R*;r@*jMq83IDo$`#YZ{VX9a7v_2yI#k3Ndjo5U521)| z(Y=Kd^(53Mj=snO#SO{VGJ>3G*vtoxVQOKg8UgT9Nz4@iAp8_kZo zQv~5Po>F>qc!|Foweh6lY94DpjA8_&R%ihpQPw%*k{0+&*hvm?WC~ynpkAubRA41i z#>fp}QhbxqPsK`%oy?MPq>Z^NycRb4#s)^8C3R$fz`$xuUGq*QHJF^7Y> zz7*GnrjbDJMTjq+6xLF%-Dqq&$8ID+3&KBzE+O`k_{u<;_v-B#YdmFUM`s27$i!hPwb27A0@u+V zoi>1@>2rJu7>Y+Oq@+XL}%25nGf(!=EYB&?0-(JUF9<*GTGfm#dCdKxlh*8I{I zU~;OWE&Vq@5@p^3O%9Z-KHl!K1w}0w{e3y8sL2l{%lERCMSXZ0_uJDy-^*1N_2KW5 zX}Zv-uyo z8DL!(#d?fnOp@PmUd~G#KVZOTkD7G!pwAz<)JvRXA3k@Mmlz#9o$V!t22Vw8VQ%S` zmVNw|w1s+U>G&{Ag+$+v4U{%}eHKh9fI)Z9%0o|?Q_vT*gsCl+Oh{VMN>#F18` zxnlQ)tarpg*WY;eE9>t3`bDb_FKUN6ESdjW#|J;L<5&0na@%dsPX4$62*UP9`DBvh zd~*KQmQ)tXz2QYocS!4pdC=yiDOJ=mOQq})kfA+Lwy5RKw~#N?q*D2#n$U7O8plwV zA5GJu^k{!>lGJVT8|FXF{hqu9i8ZMmpZhOngPa#-OpA*oeki~9%5NAY&AngV0wql< zN*n&CQ9A2mPRlYOLmHhgxyAGys@1 zsTE=QpWSPRTBsCI=gI{C2RrvK(B%BR3?OE$pihmGi?xC#e-Q?NH^&+enp&e!B&4x00e`lQX3dGmW~ z>!pCUn*66^`Cj3S;&`FVVYbf&nmop4RcZ1Rn^mXDGi+AXcO|*aW-ZX<4#`5FYl`}y zXka)sd5j(5DoswcS#_FJ3WKtm%-B*3GW^ zn!zf`Sr>JN*F~$Pg>T3&)YEg4RQ4*pLBQDc(x2Z(`i= zL&_854)G8FJna2L%8x#BV87vC`eI*i+~EUjEgfo98_Nc;X!1PCD!n?P z%9lxrO}297cO`kh&8pF)vcy2I5L=y4Yx@3isfu#3)d`hTE_yTQ!X_;-qW@}lYz24S} zt=_cOC|j#i`?|x{icOEN2!wA~ycTOxr5l#w86QJjijC5XY7;CBJxwcRk#|NET?N4* zttQ_Ss15)#IJR_A^HuSbd3D;mGa8y)CxAs;(R>9Ctwdqd`AXS^CS~nOtCgClwVuu( zP}rJOva*uZwvZq#gZ%BismccFbSeV`6cPV zCD)dZL9Sz^-f#=xVof^aTJcH9rA<59O_O7#w*#;caml6m_giQx6$wWnYVs~iu5c@w zACN1iLlw6g+fhu9M2mcKOG!IX)i?i>+FbBc2cj7A;}o^+dO_%vXdj>3lQFl4hPA;K zX?Oy0cwiNVjlz}Q=g#)Xo!%Rm+Dhub8_XM|!Mo*mKksbE$ELYImbcJ1ntZNgg{XKb zGbou$%sW~{N1N8rnqDiN9Om^(o8qNX=6Va`;wThRB#&{C3~AUfb11I&*rsbW`G92Y z=dEz)rMbTdD)a`+7AbOAc%pCtC~6s{b(t4K?Qv;lM@e`z_gM>1l_s^#Fal9%b-8zx z*wM#)y!Ax!SSdf;+TWF$to%Z82qzbNRFJ|5`3FNI=A5vM9v zx;VqSK$E%_i+OwWg;L_b4q22&N9=h`QB?X3u{0fC)U>W)fQcq`$r8gvU+6e|RKLBA zk@^Y9Fi~8z%~Hn7$0{RyrmPFqqn--w4q4igau<~nq9_0*bXH2*)|G527`2RgY2b&^ zq*fzO&-|EoOe?Onva>*wTG98G#=UA!3fz5V=NbmKCZE{bP@X!BZ{TaiqODk?$q`P) znxa14D7jdNYSQhYCWeyspzy#D)1+Ims@IBJENJsJslXT_>XX6)YHQN%Ax>=tZIJC@ zfhLDZ7M3NQ#WnJ9S5Quq_t~r}O+I6@=4D1T6s{6G4h)Ho~>G?$p>xLd`&8eN*Ku33XIATPz##;nFYT_liMY0fAW;yi_N;T zkH~wUD{A(`@_Lw;^{(hM18%7qrM*%wl2^)UKr22Kbg6WGE6CEM67yj7BJFo)>%xHy ziN5+!YN3d6AO%JARk@{8jSj?sAWM^n*{n)U9%ZwtH2Ec)RjJ9LHmgdL!zD|dPWr6O z4cf=YwMNJF$5Q)1Cayy~eeqMr6)z3pxOf2P_XTg_@EZHcv#wxwzTlCMdHipxbwcbK$$q;3CHO=|m}ht~UO z8MQ>vZpjs<>3U1gI!!86DAw5$xX|>2CONWJ;oN7tCvUjAlP#~ann!F%rQh{}3R+Al;w99($MGGm75P}m$M0<5f zo0GP+S}D?My?{pZKTfszf?SgKBHFT}yfoQlM`UU5w9-PCSjegr zAxHUX@&Zf#Jv6C>l#V5=ft2*-lK<-;gZx@gi3ZO8IMv#vyPQR`PLtXa$ba@HAiuWk z%mSKRDV===HcYQtKnq=AA*)n`oLNMZms|4hp-C+ikUusHsm1xK^h%tr#npx@#r<0E zcT&&T%>S91TP>dR)y7pI98>!?!4p|(l_*%qii$vI!sM)xUrb(_JXf;Jf<>Rc8Whyz zMWKQp-9t^O9%_p59%@SU5MSBEai}WQLsf|$s!H_`UjlUthQ$DRO>PPyuTQrG1vPnV zs9=;z@g8bQ^-xoc_fS)+hd7nuJyezIp{hg=Ri%1}Q|W%eaC?BfCZF6R^3TcJzXTOE z`PLpQ>f3$xG4urYYO>j8#jH=Qb&M@iqe-n*?rn$({##PKa$nn-CdYkBv#K?DqGTEA z(Wldcf|_g!71XD7K|xJ^HB?ZaHtlb_Hi_V`0$gVSuF<4+L(tU}^{LZ#TBFJJHfy0K zwP#aTpKh`RYc%-_o3&7rL-#Y_nY#KkU2@^e(BuF?%M{e7V=b~Znmom3E!3ov#MBMw zVNyz!!z@**G^x}n^G?^Kl7`x}8pMHynNEfO<{w1@+XCNGg%pTUMvuV$$hQiMwc2q^D4xM=bQ z+eM-GF0@e4MT{2Tm3ACjXz~ZPi+t~0XrZ8s7%hJy?Krg5)S ziY$G~o7CwYZrlw;%~TmIu~yRhL6JR4tTMKd*yzNeZ5<`BB(TA=)cf2~hZZ#_=p{BB zL!?r*{NmHOLyP*bB*;9xs1NIc%)v!{P%VP$N5@nOGQSv;EbVxzG*|BB5}KYRdGKj! zvQo14CFk*-yMO=o$|H)Jw=gKKNu?OI8l{(FS0D7epQamwrZm}BqG!!Z2gNn%^ens< z^sgjb9`t;WrdI_`Y4Y3>J!{_iL2*qwJs%YGX;L~)mj*o#(DY?NQ<}V{M9-S1-C%H{ zy&Ny5%dw))X%keHv(Af@*IPX`ZX7#Dn=;0RHl{WU3Tjf@Hw8bs2WxV=J=9om)mFq* zs!6pKF+*z7?KHl!QF5p?j#YprwNrS>6(?=0gyUQr3g}~@Pg!emp(dBxtV&H@Dp_n( zJRcb)45Wf{yq$*^>OB6HAWI=tg6x?#M+y5IrKL{+mNEC8V)urHKc=ltlUIrlAc&51 zy%f34wh~)swaBY&wn%K#L23DGTO>9GXc47dDaO_^Iu0PUCbb$5NfZgKr-e%SZB_|r z>n(^in!Hf5_UIAULIEvL)ueV?g1pfprLc~d84(lts|3VN7Wyhp-ejTQgVR|H1;bpY z$pHG;$VZEmf<9g*4*mB9#Ct6Ci#2(_g??ip*9Di;2VGq7|20Kx#C(-e&dUo#+jS4yhQ}aGIk{V-ks0Rxme7 z<&ZI|nSs)dNrl!AihS}pPusdKhKS|_y~Ggtwp9A1{PH;J!)rn2;YEGuE0YXL4le40 z&Naw9x~LC2@gOs9jB3YPsFY+y%P?@!p@RXk9clZy$Mw!PjJf zLM~?0E4v3};%G+80L`OgXhsW_Li1As+W`3mngcXFD1cj&|0Y>3nwqEZ0?nxS=-T3! zc1f~8lM@dxg%PsQT-1kp$s`56pr}vZl3b`tP4+v`sjUy=C9{M|T_;Uw{r^H!Hw!Mm zvp7|1QqcfTXItnuTj(n_`9qskqsjYhR^?-o{G-jP(d4@}t5TB*O9>nxqYnWkVngyx zX+k@OGeeUK1BHHG(MN)c;CU$Oi=Ro|{SkHKy#BdlZ?n)KLv9402l?{N=7jHZZs=~ua>)U=`Xl#jtl zn$)J4rK4??V9LjP=cGSd=Bm-;Uu;(8hmt(lQnyBvU$9w~nj9xte8iAGD6A!fRxu2U z)W(sCxlJDxnBK5pR%!Ab$#Mb(X(<0H+3y846tTe;iz-bjHYM0bUk#I5CD=w^jSL!& zQ=)3Byq#rBRcf-zW>sl&k9!KKK$A0M&VoxrlXE4@Xl#Az2nuRa=Q~qSpDqsyYVz7pi~97F zeg>Bk-jq1NH43nHL!H(X_32#OX^keYuvrT=sXd#z`gE5qSfj~5+pL9}TyU^~*3{Lf zGbIq!i)Jl~WJB(GDRO(=5q)8Ec@zGkX3fdiWO%V;n8p>5Z11 z7HaZ#n^mdFZIT7;e@5|d^6+U(-AYX=;o%L{r1Bp4Rg=nlD5c3eq%9i1nxa1GV3pW@ zoGlzV+cM2*npFJ}>;-C0bx>H7%CkXX&ABQltV!ixyZxvc%ECb_niHT9^L4BjI(n9l zi#4gl!}jADF(%b|KeFQ#n@6?C!?s9lS$(0r`n@d@^O}B5UcF_D%#SJv{I*U6Kx;ABIRc%ZsXB5<=wr>gs=$ll$2WwloJ=9om)hNQ8p~*j6 zZQP4}^wFJK3sgIW8K^iZ0wpZNJwShtHH#K%@)4U=smZ@e7Ehj^5j+AIDm6JXKv5^G zbdaTxDna&4oAIK0j6myCfMsl-?Ro+7DBI)dnpF5oPDWbq1&ep=QvxmWJ6j}n1i%gQ z>YuhqY)bpKyy|ZOoF5(Xr=*AjizZ){R|nv85a*luqUI}(WnK(?9X>5mN`~TP;!7gs zkUh1dWt4TWuG6GeV=jtfuJyD~DQDf^g#=or#8;}(BBfA`mx-gQWt`#ESgU4 z#Z=o`EfoW!LUXj~5>77#^A4#THjgwnP?c5D-nD_C$S0p^w5{7>m}pMWOUyjFTPmq) zF2&T*GCbH(jG~1~ses=~XD`Sv8BCgfMe?9>H2H>P#lGU#7k>@vXfnVfZj0@b;z61C z-bAzvaQ)~)rEq;uVUwAV48Jl-?k~ zq{$~GiyJMIS0!x>_bf+5@bbd4VH{y z@(XMbTMs&yb2u@{5X^Y?G`#S8TNvYc%Orj7QvcN`;)Q zxImLD_SmCVTy87YXwt11@9|uz@O4{pfhI4pSv8uxRI*B=-7RmwWh*Yw=3Q@U0bh8lUk4PiMv>HQsEw3u~L&- z5ubX;bc{BBn^fxy|G?#aF7NX_dDgmnA-CpoIS9pK>Afh$KCrzix7srCA~9k`i_oX* z(SO>*r>9TiS0Ly|7m4*0ZKYICpO+p^?7gRWkyuaBBBgrL9_IJnQ@lv5r)ZH{j{R%R3#cb+kTyU$PUFy%28( zYmS2UDS7i1ygXbp|F={-$a&dk0ZI(eH&ahBfF2Sp(dRT2+fQFtO0j`bY(oNXxg}Fh z-_MtFgS-t80stB+JMq1`dDcj3X>g(f$4;67@ zm_CL=JmmU*yp*$uM+<6!iBj-TrI1t5&|FKLBqamHV@0)yO3R)Q_agsPa6c5ThuYJm z(#ZEVbg$8er-fiBr%COW%SD>^wEZV-`&F7$4#c?{np8$JQ;}7!^^O zpx7WFA`otZfW{j--l(dsZo8Ad2!n$ta=(O2ssbWLMGYFAi0Fxkni*jxqt1wmPE=G> zMl(L3nUQgVuQ~ylFLr!?X|1BxICYC-uKNP>EB+p*4n?d_S*aGbE#9g z2F{sCSkfjAZ^ZB5v<&`R;(78)s(lIiu*dlrKEusOy#Zsu#%EZE+wyd$+E)x z+x4%M6YvWrY5;#^qB?LunadgLrUAOz!=`itc)W^4Z^rRp4*0K4wlo8O%4A1mhBeItwIee+BZOQQ)Bk}c?5Fy7zuNg*Wh-$0-T8mprMW|z4QVYvnMG~?IgGAdl3St85o<}I^f0oE{>|$G{6v#2 z&A?AF*%29-9#S<>SN+rVubf{4uP{*^c$JFwsGV0k@__QIs^`dR|7MlHHJH3j;a7v` zYYOiRqBu{`IoniPz+mOhkg&SNtQtW2l(W)ZG8(~^TM}l}#i{yt7m4)3+R%X+g43;! zCh%ev{)hTz;!ChE4P(d>lLP0P_6ujua}yy2JpibUyY2`_ktpDzPe9u zqIOe*OU-cn9SYyAe;1T{n3I=eNMJ6B142MHVP97}I!-)*La$F!)qGUI2dPY)H4kAd0?++pNQ)ggC@hYcoGh03WE5!Pi9#r^Z?`Rd>?!df2RK4Sk1x7LQr z!1;*o^R=n^dF+ym|5#n_Der9#>AY0a$*2FNTJl3?;+0R4QWJZXJ}jN+#eFZ<;_ZYv z0({CVmlfJ&9%wyJrIIk<3%e9)%>+f@U0sU3sNh!a?*j3eQ<=zUebjoG0RDZYhlz|9 z{U+`h6F=d}tB7jgVCL?rKS|St>QlvR-TcE3l}gCLf-9Gi+Lwjz_{D)CqlDsNWqL$a&Pch0dZ5<~(L zizq6j4Og@=%{Mw1)`pMuU%lL>Bz|FublW~u?l%>;>mEFLS;2m>{*{dpa9(!7Ka8`0>jTVn&)yfA3ilYz+2Q)pk*3lDK3_$HwF^2KjV1Lm zp!8M*Z)?q419-QJa>GVO>z6?hI8f=KksrLrsnAFCqSxOXxdRet_X8T>wC-8O+R z%K0^oAMO$h=4$n^*?f%y@%5D21+pO;+`RpTMZRS(CYSv#ylURM`SOc4!?L?Zkth@U zW+*c%G8*$T8jOl;eSk2|pHynep4UcN!K;<->-Fzr^3}T7EpBJG$VaexOujGZR*Ah{ zson@^u(n|T?58TkPa%vgsgLVbSwaD3CTaljFIv4EtyinZNb*b8TL_Gg@RtALm289U zbL8~0M=^fk^SH|$+7+cM#R(+lvfN|CxKQ$~#9TUwH8pm#u$FK6i&LH08+@2N#n~4MKbU9?IMgh)@-&WSwC)=* z+rmfXKX}oN6bcZl+?RjA0K*p<)dsL?=+_hs34ea=!nFe#Eiy2_c9BSFK}oLHij)?4 zm|yFNloqKdJWR_U#sB3kVAt5a6@IfV^GdCKHVSAu3-f zIzfl)v-(#=z*|k!1pb$a#(=lj$c_LBDDjOYqJeK^u=fc(q5k_If@knO5@+hAnxR)% zrs_bf`EZbH=n>@gA#UJ37SFVgXF85Yd=t$=BQ*_# zuV?{38MJ`#6)j+(?rxmL)bnEs+!+%&y8z-zhW|!a#P8PdsTE0HoGCsU(6r>;fj?%c zssnFUk&N@8)jiWtXaUdFddS^@SsX+SxHFW^U`7f_k}LyvYBq^dOnU{epb`KqF0HTt z)3qFj>5v**SX-r+OEN@oStwf!eO;yJQ6QQGK=SBG$h=3%w=z<1|b!FXNXm#220l53HPts(bX%*1&9Ng9otH6BJTa7XRjD2E5grmyp z9}hb!9KFheZQNZcrAG#{l=ku%I(3tUPM=|E837)qBDtTtGNUmO^kDWhEy={!sqlWF zSv7z-lnA^{R_kp#RGaUtyGj6h|D z0tROiGtLOGNJQ-HwHp1cmW4V){$(rwx-b8F?C6=Qjdf~YkIvhJXCP*Bj_+D5K3%Uk`mi|~0|o+}jrv>Ip%3hiGrJKWoMfmL^dye5EG|I$l|^VlMgtPN)P%&) ze%|4B>fDCQ9mjVl#|vA_iU4G-0Di(m)4+|(ORe%OBSc&xeC=t5Eg%l$2o0?I@@Emf7fK@|(mVx1}X4nL-Fi{;?_2qwV4_{E0=?lbA4k*8B_eunrEYEak;bMNREhiqd zaA@Z#y_}L#Id}tCTVPYbgGQGX4&)@|6AJG#(G>9TG1FSA@Y*1{P2sMuGA-a@6HNl! zCYk|WYoZb0znExdbi8!CV#3@5AFXvJ`qpF>R2)jtk`XLc7LgMXIOuAV8w2mvwMT;pA+nWOtAcf3k;WmTKfd>30ioP zhplZIFE`<(?TiM4CA(ta8?{;s<2PyJMl0bl;D@zd3%#g1YcO6Pf&jut?Cv*mbt5N? zX;3tUCXI`X$tVz#g&qxP7W8N^*W|zhq~Z!=8kx<-F(6}>I01@-&5!j&%aBNdfq0T0 z{llp5(1f9!=Cj0|feUPm>cF~*#u!OPN(>pFA`<4WR7Xv?ITX&x3uO$eT|Sq3P?8bKnh5X%|MbQT0oK{T0p`VEg<2G z7Vs68i4kh)wLwJqr(5_VK76-fK29f22(U#VPz5rsbQen3l^c$3x0yH!#JB8mS7qM& zXzabvi$?hM^Af929!=61%A*>EG)3Bg0$<@Vo~x{xO@ow2SZSUm1mY8I2#PP;v&O+d?mP z_6$uEqn8Nn2&dABy`XZms!{{X%!GU&A7y%xlA}NtA?IBsZ-MDVjRQ!{r zWE8krMd?d#-msN2^w-(EVIyVeud~7DjzAUo&$CkXhOLyLr|JzGDML?ni}JZUPzC<` ztW>>WD`n`Zdc#J_&{N%}e103K0^h%4Suv?(^AG&6iprXxwcn&E#^rXjHxVi@C)9GK+m22*aGl zxbeL;C<5{cr1b0$A-~!u4>h z8eC?EW5DZFv`BW9m)Dla&s(60{^ee`esmiOYJy7TR6K|Ue_ zkIA2B&7T$wDtvUNdO7Yv#1gr}1?hSAgGp-SE6!4ad7QO}(_Ng~XKOYn(&F*RPpFWw zNj@{l1V3Gs6I;Aj>qn@BfwQL_V?MX239EwK76Ouna)F2qnQe@T>{NloR<4A&A-Ke4 zPVdW>@~W>^=6$)P9A6m9A;w#|YeKQg!u-p->4z$bgLPlx0Wzn`z~hWz$O2cKqtMvZ z4k!~qyd}?N4p;q~&8`K!OGP=0Rz~Y_Q_BiMa%l@VtfEqp)>T0xV%R_=^g=Mn>y_pa_Id>5_dqlCMmqBEfyVhr zeK$0&g~Y0e%BQr8#8*7|h~DS=iYLtGthIQs7=_bx6uMK)w>tUsOw}q=%%@1HiM`}0 zIr*l~oZk&4xc}5VR33Oobf11C$3!42OGP^l3KdqIVxhH$)mvjwA=yhV$ZJ(;li7^| z-#)#p=vod%z(Y?dMfoB#n#IKMb+h?u0Y7h|3E+2CG*G*^23|iymlMF-R3!HzceH67 z{c6(#HdG{f+uAg)2ztN|2ECna8pM_Ob{0WFTPhxiReA;YhRxVexNd0o$|B0LECCq? zxeW)b>4?Z>noc&T4Tu%OUjE;u`Io0$HxKrqv@O=DeSs0QLUC){4^-tI{VU5jkO<_q z88Gm^)<5Nds~@Q=d1LV1@(tmoOEM()Stm{4WlAk5$5H%FjixP#7O+BRs~e`rsD8~# zb_}>cMf2r4Z+nQ=@2nQb;&GDO#4#=0lbZaP0{(?jZ35{;<|>d0Bxg#%ABDNJxXr}=6E zvCrRt46gWE+vznV%$@vo!r+SGMmHs%s+}JgNB}E#{<){)RsYmr2dvoncPzb6^%vRr zPhF@m6ycN~oT)hC7pbEc8-WIpjLE_Ugp*uDzEN)DB)*T^Qa@wC9}r_Hxw~BgPgnid znOzIWk|B1R+BA04`6c@Y;2|n1^x`pWBTF=}!v7Y>|7!KS zCin+d>~8C6M~3BG0$8#0?{=dO?c6=!}0z!}TD6g05nY(vVK-#~e1%W+mYd~CPF$JaE1A^B#E zye1-}{e6`mk{{iMRu=#7x0-+RR@N$wLk^l(+v{<1%{Idd-b?$omaoMO4PP!U$p1yClN!E^&#&; zw(LVG2G~*nhsryI5`U!F%YA^nD*AU4m(Jm}%ksU=gYZC|F>Fi417VzB zTT&f~UXftRh#y{zI8OTU5Ld?gbSjc#Ss#d$&O`*HUs<>O!^?;30dkccJ4b;xnP>)h z=4+M}T#6--#wryiu1R_CExdm*9GTa*Y2Y$Hg=}TCLI>_TvG1Tk2Ti4wAECfgf&bbz z4Z6t{!J$j{c!lq>Ntaklde(Gk1Xu1BjdwMUtLslKGcD(~Kpf`D8BZj~ZL+=l3EUrQ z?#F;t_Y(>C?wsruu5|tnyMIKJ1N&1niquBtXcyrlT)qwM;OqKV&IlQXP&RIta)zT- zXIz;`@-Pb!-}!?uEm^V>L+h?n&;z9>58!xSQ^itH`Hvn^O}$YI9jZ>w^?{h>_3dvJ zV;Cse#8_^#S}yv;vpRXv2WNCyOxQ{!1w@I(<3TMXy_^O5Gr<4;i)DqjJYWvo`?OLd zytcP#T%$tK18xj@TiZ0Y20h@fgWird4T4X+&WX@bDc1X5A>F0rDDOsC-x=C(^WfrB zOZHFIzlp2jbb#$pY5VG0^zXas`^o0Y)#{&^JEqc{r;rTGv;tNl+U!oea5Z1;jf2R^o{M`O7PcPm4hEUwAbqJMj-(%$-4t|9?DCK?4+Nc)oorccYtXbd>1BDwOm zu{}hqrPD>GUOZV6SKe?>0`n_x;FGM&aUh*Y%7GNMTxkM6JJjROHVuke^njFfF%M`^ z)S?HZq(yJ1)%SqKIqu_hi|TA~LR+qf1DTZ4;|`mlFSD690X)S-BS3-^Z$Ls5Z`%^{ zcp?$!wnUsGKF*1#@M^?asj1bSUP?z>EjEFSz0TOhv@8DnHMHldpBDyyz>1x}$bGWv zj|Dqm#m-*|!LFj@zqxg$Iz7vrH-HtpJKZXQ-DPIi0#@v9b#~XNJ{gn!GVpa;sxtom zIwE$BFljcmX;4&hO$t~cvB{Bmw2e_* ztZ&oU&(@F;V5NigE&$lR+U#boRTw7xjGqWIwdkLBG=ifu!=wei$C57>muQ3z6En}P zFn_J749utQaLlJXbGOf8Z0=Qu#hpwVKU1L`QGkD@y%c&ho~c4POl_3QwihM8w($#f z#>gxzUVoH)OW9rOkJVfTc|(VFt+GD*$%^Gx^&r~?V8z+(wk}qj`4t0atkZHH48&Q! zkBf<=cK_a06IYehf19oT`)G-9DT?+|mCL0gxzck<@qgut%_ZpzXIQ~nE}ZpgZ5Pf6 z^4nIsa&)0V7GYb9{!yBQEUOi;68+Xh^p_SF&WIlV{YNiP8e~xCz$X{ZfDo4pXFJjt z&KNkVD8F!aUY{4vup!p`!kO604cLAzoMA|oOI|lKQysKt%5}T^ zT2V}(hk0B!M~JyyHtW*|$){8f%deU}m>4i*$nwi>#1LIH>k$T)4EABgVB@2D&8&xm zsG4g()5V4K%DO<9^d%x7-O3{7Z*{NMgY`e^Uzr)ePpc?Dg&oakeK#lqzt^Qm>%O1} z{ArhB9F?ciN>OPgsI*d4TA8QPN>TA&Fz{3wDJqQwl}3t6BSEE+qT-*@_f(oGD$N9y zW{OHPL8Y0Z;vez%RK`-6tH?HG#-%Ul+R5ZlQaARNenh=6#?$@+A7Urq0;&r$Zi8s1l* z#p`{wystjzr1v%IeT}B_<)3ETpkwuE8?&R__@G0H%U}R8OAm%O>@j2GitvU#R-L&G zd#s#u8}?ZJV?#2)2;?uu@1?`kmjnVaOD7QLXv^*&ST*zulsD|L8qZC&#}Q?2!_gE~ zUx7Qc@^1diWra^U?gHO%dMV1!EITybct$DAGuGSNE6rOA_{-PQ!6fi^D$0GeGFtSI zc=;=2pK9(KK&*u9EuA8g%755B$?TfIiro%(T8!P1W;YJ}--xt{>O&D=MDILfo-SATe+EkzFe_v$#(fVAkvOpEcZWUqinE61~qYaK{GgSAWN zWi;qVPAKv-Vfh|95_&GwOkkOIOb2p-9f)TkzOh4tZbUEcq|lq&|B8hF4GI4% z=<3H>&*Fb)n{uY}LP&$L6rn$Lf3ZgTQcHCc*fP-w@P6$yKiLM;x47SKMCdbhR!*0{ zwMjaI56t9RSe&xIP&O-6@;zdSEl)5ApCBRTg>r5v!@6$1|Kj&54f$FN zy$PgVEMG+99h=4>ZeEp?q zsL^%)D}vy!8I37m=&=s`bM~S2fb0BuJGmjnQ%wrF;(xgVnVUi|FRFBXEVmCX@*g;M+ z_Ao2WFX)Wuy)zy8I8}XP#z_c=X{S^k|Fqlez3dTTPoK!x$Zw~ zJKcTsd{(ygz#~<(hgszJ&xSMPC+p|d6#|a_QCIOy{NFkQi8jB{eVnp{Z7u_|TdZse zpR5#eH8Au<4G5uJEoMPB=_Nmj@pk^i2D{6Zu^gj-xRNSc(}{f*!d@g8MLCH9GJayW zzC(i^L=XS3vH_d`61JH7%OvopAGtpWq!+PU>$-fjRYP7s@>)CyF#3fR4JbIPxic^d z9fB3dn^NiAD^e)W z{fl8f6_*J4ZV&mheyQjFu_f+j7e1J^r60-tlB2qwoli#-4~{%J-;+<;10zXMJsdXp zpej!2D>5nGWS-&@TB|dW{8bi!6K!(TfpC$!2SQ5<&3`v=t-AQ6xf%t2Mn(C_+-OGY z8$l7crAv|4zXe6$wk|~%sD4WSvStJ6ygaX@MgO7*r1Mgd7X6DNuu8?>vs~+_G*VO= ziL5kIR2rG5(nwM9_bfe?W{OHPL8Y0Z(o9fkrl|N^0-nlvipqF`%6N*(c!J7!ii*EU z2bI6GBAWnG9eD{%7?rOEMIb{`D$@FHPy|+M#b33iKitZ}3y9CMhq&G`BsFrO09ZBj z!F$6VBRzMIJ*stX!yYqkZo|=3jQ%bRIUju8vcjia7y>R-QJ&S&jMm;k5x8%cBCUgi zBJl7o#W)5}rIn)6N>FK~sI)RqrIn)M*CkJ-k)qN_P-G!j%ADJp(l@>H5BD$N9y zW{OHPL8Y0Z;@2flWjsY?JV9kVMP)oeWjsa2Pfe)&DXbzu3bkAeXzgXyI02+ON<~@+ z1Vv!AZ2h{_cMoyBV@PUbT>?TTr{;tAhCN1l?jCzo>)eJtX58F{qp2AExe2P8em<>`@Ec}l{U z$bn^8T9vb^$_EvxeyJK4VSL|vo%%ZYb7oDlF_yB)yqG{m2o>wkW<*3A=;_Jy~*aG4}Ze0SazWn|LU(YvR z4Pe#K?=~>}3o~p2=~+(hfK^|9yMQmW<@g4~P)?QoEqUz7WcgEhv~W@WR30rH_I%m= z5)FuyN)$kR^qt*Ve0|^YHU<32ndPXUyg-2^O3NPSuO*DD3!Y>37kX$F8@|YzvyILBifJ;nN2mZ!H zQ@{bukkrbC&MfKbSZ?aTbtamM(^Ytj6IKX)%BD!!J1vV-KvI{;V&N^$V;Foj_kXqf^rg4a_k|JE)p%IpbQ6Lk)Fs5;5)wmY@`>KZYjSw1; za3sH*Ex&|QFrfkWqFWkpFEnW!9=L36(}2BX0SNcPn8uc>aTJFUWbv2B!}iq0{&o5{ z->`viumiyuaFdDZz|-tNItF~BiR!>T>@YCKdbYxjnq$ClAgu#;we@~1J}Tr~IjC}Y zkdT0vt7WbR=8z$3q?>fgZUp!#O`>R#S889h=FaL0vidibtb#eF%3TW}d6W0t0I#*U*k z{LxrMyV^(jw=wt4-%WRXBP4piZ_~IW2ybmOaPPFStOMU$)uXW?=m9x#C^}()zo=tg2x*+9^RA%Hcrt{A zkOr$5X^JOXB~6L(zO|hg+~2z+JI5+UaF0Bj>*g0f!kYhF;j=ZNuM3FH8_;NG*8~iAcU)|%WFF2s}{58y9k7Grm$^<@?pWErX-6~Lcye?}IC{A{I9dNze{>QVojk}N zos0=RNb`8ze30gG#t4*PWM2ruUs}dmEc!Rt4y*MH=zS|C9^3$C(h(ze(d2J?atyu8 zSk-|XR%Nr7hh83)G7fB~*zT|nWFymU5RcO*wI|%mT*K$M`D_C5J^M0_77U^_rNY1K z=+ukamy}}n!iDrJ`f0xI%F}NdZSvM_g_Lj~U(II8CSTO!6m&a@vk}F|Dp?-XXbop^ zTAtOt)b>b=6Z>}nU!()IDhwE~G6Y@gWhkLphu~`Uh?~TU7OV959q4P-6Q_>l??B6Y zJ`%qJjmv8+o(7P!Ai194e+PPBy%KU*xFP~PN<|CGx4i5(kRdrLzv;mo~Is8JGf_Dw1FGo|n-$E$Bglt`b8+XyNSF-PXN|sKZ*! z9-I9_VkPxfqg=P(>VNU;_TJSf2X2e0TZt}#$MHEvO5oGVM8ts{y^~RwNq}$?m<7sRT zdT=<M;9c$QN)o9iT;WnlEc!aTUmaoAyBUb+Gwt0qJIMF}D*v4lR22 z6!$-yZcJ>ys^1F7S&t4eIax8_@OPGj7LYFG1#_2YG)TwLqjK}+{>Y4Q`zr07)v3%vSUUvK26s-esB?(z&zP=vxfVeO8dif{3#0}4w zx-DrjfBT?UCtGY(-2sJ8zHTZrK&TDYI{w$qy+Qa+bA&CpOlJ1pU$B#j zI*{`Q*$an@BW|pI=zKQu`48r^3B>p8dttO-5bYH!bY`7uF}3auY0R?w)0l8Xp?ejf zHJruFGC_Gb;belXmeS=Pq zciQ1;1js`Sk}4TDc>|xDTibM9`m-jVJl_msd~dk%Ft7-oxiMDs~72CdB^~`yNc3T^@gpKp{ME%8!1Cib+Ph!ZJ-ML zu8C%VQx}$j$>RyY)hd!31DiTDj=jkAfaj>F(4%pS=}iIeFwt`0t{3BN2DqDw$~4mY zb5mRnJY7Z7?Zys`Z>_}Pa^PDoK{N$iZ=xCCW)ry`J&AmUWvvAyd-4_zAbsTgTNy37 z5k(+9mWs9LAD!Y>)(jv%bHhfWheoQ0-t~c7IVu3@K~j51n+9G*4@d}wUKBv>xtgx$ zTiWVC>ah4#Pa1SCyOzy^H0WG<*yflNnh7rcS&);I!%G7fAl(-)9-%?^(!;hIjS81_ z9U63BXwseFIw z27M$<(fPGi;)NEowDdxYiCQwG#YD9&Edr)%>4X;ZRj6!~U#E~xkuYsbeOjcuBuI;V zmsDtx{ZcnBOVYoLY_y+zw1PGjO6*ftL{}@C!>5+;G z7xzG77x%#b6KR;x!kx4Nt^=6~QiJ|!!c$cLv|tCU*!i~uV8`+$&)NXVvGlhgvDH`+ zuVA9LvBTo^50?5dAS+m*M}u*Zei;vzBM|{vhYBh*7#Yz6QhM@87BE!d3g9Ly-!Wh% zNB+slRhrW<`$mEEBE;6X!;kDk>9XSA{{-(_)&D!f*a0hctKC8Cy{dm>ume_jZMB8% z8jHIQtoZkD@5VpD%7S`sSH?CZiOC=)j_zk zO@q)RyFjv>69j**Y8x*$*J;LxMu-4k6aw36fxVy_MIyAFF0>+!L~!zWvJ$&05WKTZ zgNWpW8Aw3F5%}q~Y&%C)flbcuX{P0AT28#5aoJndi&rZ~y6CXb%;6d}qBPg}t4br+gnx{!{i8tEoH=?! zjn&9{%ij?h^@a}hFo$c@!q|+)gB*`deo4K&042Y9n#IG^{b@X62h2KR_#JjT-2HkF zT`*z{AJB^Fx3hSFxUL;%@R2xpF|tZ*tGG=|ZHU4hY}W zLxdcqA6>ij_5nQSy!hZOxHMnvO0!969%09xCXmsJ*Htd#_c-=E zt-649jc@f@zMa1_teA0x3zgJmhB~8tP_)eE@0PTh=(5Cu`f!%jYU#41a`v;OqzR;M z`ectf<%C&robIZ)E&`}ieVEOWtGF)vG>a?t=~0Ti>$i(&T+4m-7+3j&=#l#Wtj->} zWCvtQ^hu)EWEs;*(|zL$>7;(jqq($M8oRBKvS996Suo2)=1+*lbF$ko`#kLpWS8(|Zh+ zE(uKRhnflOnd#kwQ#}+_=UZaM7`Cg6EEW%yQoMRRR6p@@@|X~Cyp+{~w zR)}7lqaSJSgStbK9C@m-QXhT>EXR!a00tEBq+jb5!;M0}~fVx>f1 zu}Z41ztukAmCRwRl;|r~N%h4zaTHDV6)PqBid9m5HMAz!D$DKz$hnMs&uF!LlO+E( zNq!9@aXi(1GzJ_LgW(+8JPg?29&9~gm^fT&r`Rxg61KgLni&&{1-@^>7%&r`;RvTz zeq|g_8PY}sa+(P-{EH~IV!6aBf4nX>OFeRj9lgq}jJaIP71h|Yh^pj{$Kg&uRzn>w{je@IGrQ zc7^;kc0SPRWkpQF5lAG1H3=_{r4aUVMhJ3`1PJjWn%M3#Jx+67M@VoC;y@`P4wIkM z3p0syUg;dwarAS`K^@5D{^ECc@@?gkEbgQ7QA~QR>#Idy|7Cr(fb>~BluH8w&&z-K zryx*nXxp{nZh_o79vZ$#FTZwwq$>f$x!d#zxWpCteokL3i#0t-F;3vUnkBL!#|t2d zktZnic=4cu@6#*}aM*h5KZu$ZW2n=_qeU}u=vWb_9F}+5u5#AfO7o-s@^zW9->jdd z)@tm+4dprz>hdVdD(9;Z#`JN|!b- zK*-51cR@x2rt$!gf9lygYv2qD$)}~cwZ~Rl26j`2MtBOK4rEy^^f*M{q!XhM#<@@2 z2#xLJHs0J;qu$CLhvU9v@fxs8w1K4vd-X@0iM1N%^K7wfdP>b+j!9g3sE-ZP8vuF? zq{e!=ze7(Kg*R}Eii#s24QPm7%wd+0Z#VB_KpYmwM;cp$UQCo2erGD<5O|Lz)V<6T zD`KjIMr$lk+n@;iU6&%Qg_kZXQXxwma90(TUB`*?oLec*t;}<7r8u`T z&$*T2EWeGa5kORy24I!5fBe{UZlpLj5}X?;&W!};Mv8MI!MTy*?4N9nIm^nvz2ADEuFGRWDQ^6p>dB2rE-8Lt*+?Bh2y(y(d9o~W_DZK zG^X37uza+b)&~NWt!)}mDphI0O3DOSA?P39@dO(X#HA!)O`8TzA*C=haqCqq58aj3mR$40hAK*;1hjvKWdI$oj( zgiNVO3qGO=zM^m_#Cq$wC8F`7N&+v0;|W;(xJSF@0jgXEhP;V(OJ zujw@gz9H4N51V5EvE7274}8J0vKN?Y48 zvqG3~=Z}WC!-H6^@6ZT=&HyWf*LUI`5>HFwzhbwc9otE#8{BaeOUUF(#;k2ep;2J9 z2gj^1?p1`J*Ev3v$3C>*t7E+{=;z`f)VgF&PF8Rh#+yTqRUtYef z_(;|v;8`k~pTEp_h}KIj{4pw(@WnPxZ(>=yPeb|d5E76O%WQtL%Nr9G+>3vT3b_g zv$AOcD@4}X=VkG7MX67+4@>oR@$rY__EG~bWHVsw=+0OvK z8M?hC75*)*hjTUjb{GsGOTEy&u|s3Q<>iVl!kgMO7O7Bj0%Qd)^k}eom9NB;&PmJN zav*6PtX;amPUv52dUYVHc)^cGr2uY9W#Gn6d=w=EjC#I?fdiFmH(K2-*CgkP;Hyox zGy_8^*8^ARZ8Unv={6%*1XC8N(~^s$k{zfWWqG2bf!dK-Tw=K_j-E&vN=fE=8`l$U zTt|Q$VGBLd^Fob&cR6dzcPiz*)(kNTk@5gSGqONjNz;0lrpPAFSmH4ZiuXK=XXe8S ziAN?;+gE^whco1%?@8bHWLz5CwIi%$$TxP_bKt5F_Okq`<+}l_INNM{mWnffAcF>bC>dX1 z#o3yavo-Br%YeM6}KJW`}E%hvW|Z`SQ#(Z8CmHzKb1NS^Qu5A+?pYN~3&^ zAb*QGR`7mS_c-J^-$Q}= zJyyOefDo5^R6E*3dAeij)B@FXu)GWN(%KUF+FbP~!_P_nWH>g&n*U_D*yeBh>BAZ; zG9*74zE$4dysa%i86JC(%ks&e!p4>|I8+|XmpgHN+ha%Kj}v<)dg&#f?p?n(-ir#O zK+POpyDTsJ2N4XG4Bk@lKrr%t!0LGPB7rI6EQM1#;43 zx^X{m(tg(1rm6}2zKU{wZuP_tGV+B$U?olA${9pN56Heo{QHY(Z`01V7%z7T>mB+L z8-CfY1EC^oIu|0oLi;GV$;j;R-39vyE)S>M|3Tsa z>#+r-XSsh3#8K|v?-TL$9P`xxRt^1T4MV~&Usy_u&dcv>&>{)tn>lHbhf08hShSz#s5Z7kGLXxRXb0AFvSF-UwSi0)VT9viR` z;9Vvf1KzJ9dG@`N(IBM6<0G@gv);&$Kz_Z^8u3g=U>dq}*Xp03Ir=lpRUL?xoDTXo z2tGsg4-Ixe246-3SaG$(U7*2LX0c2Y3wDXeys&z|Sxo}5mP@kv?=8vumXHL0lSLU+ z{yRzoc~%GrxZFfdAb}0kRxJVJT=I2Qr{cMQFHj#S%OxOS#m-+{JK5QdChWGmq_)&< z%(&NqmHzyjtjnRml;LB7d9RkUdl(ewgUf{e4I7>*Abn#^n>6eqKKH#yu9hNzBnFkSE8rKE=^|rV2f}!_+Nyk3man}- zNErDo-||BVrm{>EPM0x(k=Cvc5&o-1T?Z1(qx0y>-XuopjSq-2W)nwyyGHvfoA*s1 zQS&;u%k{h2TvXO?WurzOc>5OWAyw5k8D*%GHr~#LSyvLfEaUp_17p#`$|xx7{1aUqFWR-}F;nZbsWmqRYypI#YNt z8G&^d3a#NRu9#z_i1Tm%W0$U&SUT)Vi1&Ztr)LQzJ{We(&O>Dt*V#k$6Bk~%L6nbJ zT!a1QJ}yRHfr| zA|%IiR+o>5Ekrz`qDAja%*00EX!ZR9{VSVeU^wWn8n6}nm$tw(fOwakmp@0u*>vy^ zJXJ;V_5FDn4LTJbv40WOqJJ+@7Z>PX$$X?yTdN<1&DJQ#xUkt8jjh%>t8s7TZmD8b zIa{3D@q?kmuBSYyhY=%RN_+JfNVzn^`{Nq5T+smjor>gz4J$JmoC1p;96~PLc`n{R zX5Je>97+iU@81r3c>kJ}x#Lb)VSHOy1x{{ni4|d1A||*Z;@DLotVREDMpRO$Kw>EZ z#2L{P!pK+cU=6ilA=ioHu{QVWz}Km0kvwT78@Xa5?!WkPr20C>e71meGso^ArZ9(k zOfO#dD`dNN5^*wOmU2o0lRYdMEg-!XZ}y=_Uu-elVN*Tk&W(te+nxI4M4y=hs|+rj!*E;<0OFwqF`-%KoGV?5{q-=w1Q8Z#}pC60hFd#2H70I@1A-^4?K z-7#j@1Xk?)MMZBnjvbklFBb#b7WUnYPx4?feQ43gMc(=P)N2k9syAj}@R5TzDjDat+QXh%SAhA&4 zoLuGWB=|n-a0W;xvI7Rb$oiY1zqgvG{vL%yn0UBjJB|$e{2>i?{(K83XcCEil1IL2c_fT-1mdF7DM|Mdyig={&l2;S;$!c;RCC zc0n%#nD0?BI9E<0k<)3H!vzV4*Jw805wZgOriq$Bk~1Jz4ZxK|`TGeUR39HQUkzZz z&R+`tnCky^ume`?wz`4&it0m7HbKA#^rL~=#WgVXEt3X#u~ty72BzSo7B;tOyrZH> z`N1tWj{iaJds^G#?DjSds$8y&0yFhr=*7;yti zV7%X^<^=F&6V-wAEYBGM=~qasO^}#~Nf6JVd7nB&A+97_j7ErNYHmmMBh?YyWT^ul zp=p-~yyYr{d@~$D=(J9!i8b&#HBVeV@Bxh->jE0XcL>OB&dyS+^YrgvZRNa-Mn{FR zV*}n;)uX{kiE$^Raf@ooSL?5m+Z6I$1GxYl=krQc!bo0qB3~^iZeYf$n=~8$Y*`rv z4z5{NXiK7iFL`Szk_z#AMqK^7xoQEiOFv0~;qKO91Gu+#m`_*m(*lW2Z5s5Dn%CZc z6aSb=K>k*ww;NB`-PDP5`a?~?k1Y|SKoTj-^8zqF)1Tb2ITO}=iZ;)VfPueFA_}yXT`=i_Lkf&;||Mw zxMRaq31D}0C@~$Jtv#m|NO3KKPb$*2v2XfVZhg{BLd3fQjhE zJVexs#pxE(2=D_YY60;r9d1ZP>8CSMRxH=2Ed8SfSdtVu#K+{1QHE2-)SYsmQt5xQ zo9+_MW;ZvmoKWZC%mK+6)@>6=q88LH(31%yH>k2Kx%vDo-)0gAbeSrx4W5DTGEoCq ziRq41O#W9N@c%F7Yy^0hiU#BhabP--ZIdr0cbx?1%*%W?0sfkGI1Sutq7}e9ti-3E zqbmfTGUbvrNuz@Z%HfQ+v^)wNc(P{%HD z2eR-<%utub0mQA8&6ZTz_=_Iv)d>d$*@^=xYH_wBaeSCyQlF%gO-y5|A64%w zZ1yY%GK*w)4y0o_*8o;H>`ZW24u>$&-D%1vl*6YjT}@ymUAL#wb-P>EamF~xVFg%m zwmNl;@~5w{vw_;Nw&d+^8EXQcZK4%G(j?a?ft8GHw~YPDOq*ovP#d+TFJ*Vnk08UG znDTRC$~$tWS>DkK99>`}R{#lWuy$D|qwzV_lXEZmzOHH{#Da#i8eP1FPuXX5!*A7z6mv63wMD`?)X ziQUPTiWwj&k!#^VBFIB*X0!+*zXTb@Q!4t9+`-#Z8;2SNB9F}h={a{lZaglsVi^OH zj#9BwC4sRc3?m0_U*i}S0v@CqRbV|)@rRp|9G zQczvva=H7td634}&8`6?Gjb<)^C0!bB3%e+9I5kCjxZbLY|FohlU^>d zMQ8+gg^J`woU*&5^+Q{IM<{nPl(-f1EhYai7DOFL5ONn6NGLM%x3p>OYD-H4NOThF zmO&bS-yQ0fK?J|l4+^g|n2nm4ErSRMrqGU8EyB4|!wILiNVZv^O(4+)WkQ2`_yh^# zCTGd{xTVjMf24i=#QLlQcd_AY0xN;}`*FBp8OhIC!LK)8GeC+#&PIVZTV2c$9x2V8 zfuCSU+Z8|-K3N!na4jsOL70u|vy8+v@RcT70Sr{9`=dJTsjhf1RH=acOqL!Q>0Ab8 z)aC3pA9Z(jOXwAUPfYlGVh?}rj5qe@&Ujo9A8l%33d|EkH~s{RAP4pYSuE+2R6Om8Iib+B9GzcOQU{vURly++8PUVM@(cc=&5I@HkOt znOJUa(_p^Ii7{~h(8K0-JcSD5c({f&(!*>OssmY|#O}rp4Ne{893IGtgXpd6&|u>x zdL-#rwp2_2N!DQPk_A?gkGIU#fh=zY6&l1RT>vY|^ly(OR7Nsi{lLT`m#%?CB7JVK zuplgYz^zI*cL)BZ$(CkdXjhkOc#AtAA*9|Wc~DfA`PY-75!@e{d0$7m+ipaDzYY9y z;FE2jRsczgTyX(b^0TG`i?3+l(Tg7ulM9(AKzK(MhzUhH29lw~JyELkO%~MDXB9$M zW{daju0D82D{#~b*#VN0;_3s9>s3#-K=P_Lx%z-0Fj?W5xT_DI$rwz6qcQKuU1Vg4 zP?fVApV8@feTc1OHf-#~c7-suWC6a$rgIC()=_tdw-3g}kl5!Rq;8g)yHViHDiRty z+Oexb7<0qf8gn)d#9r)fH)jt7Js@c;^b!g2Co^<9W6s=}3MxmMN&{HwZ+&V2{GBbF zVcFk+KlG$xId0_Lv;73qRF&l5p!5n=3g2s4DHrpR;-Y-<*#+UnXSq^5-ne|Jx(qKq z6K#ZeOkAFh3pY`ptJU=<^l!e*gMXRbgqR(hX>*GNY zcvF`mt#1ZJ;2m9xw0;m2f&bB^7$?|sZlyT45}aEp&aKRIZlyT4GS9h{;_TlE;W;-_ zoEr(wjTGlbf^#Fqxsl-9NOAVBqwt)YDbCFV=Vpp?Gr_r;;@nJdZl*ZjrqI0z>t^a2L8|#5ov;ui2ST_2dkzgbw>Z@{XGOTNv*f) zGh|QU7|S06oO*XyY$xY;oWvy_^T)Jr0j^8}`@>%xySI=R{cUkYYnVtt@nNtMpi>dl@pxQtuA){r5Q7 z_HrJG_t;s?ZP;TQHMij?ofBa?$J?~Vmb`OW;Zq)F0bXjN5#ZZQGzz>%MWuUMA2mhy zL>N6DZz?U|%SV;U~viV3lgq zsJ_o^o4}A&caZ|BN0`+(5Povm2Y71G0=~&aQ$YC1r8!`g>bOyTui1_R166ln398RG zs|g_d@y6^FNK zc`LW#V-*~`VLwDCAS$W!phHgjOAl7$q`&lFMRfh8$6NK29;}Ex@sI-_onz~i-@Kx8 zs_IC3fS4t!2dmeZRTCJjT=ihZaxCcq1}j%R-paNg!OB$+Rzby8&!mz^WuWHMgAO_A zFFjb16R92`oyiU$TGVS;k(2(?(0fd=@KN~>_rI|cY60m+o}mNcD0lBKNaE|sMzsN~8v3h(7}AXl6!70H#u*?{ zbW?nvp9W;*>b`%L`CfH$e{crGUYu>K(Wp4{UwXqC%H`X6 zX)&nf8+U0DzP)i50X>!Kl`$@0+?Cvrq@Q^CVrKF6G%KPp z;6Wy808hSlSrMI_bO4(sY5?0RDjuYy@o0;B5_pk`#(*C-(ImOK?cMY_2E4*1*d&nA zkX#fDEC0jxY!*^$0x95kOf&}1CF_jm%N2e*i2C5ULreUZq11syCsX8_P9?Mg zKA`$F8?QPrR7eY0adnGx^(b|*Ft`F%?EJmM$Eg0E!46ol+u4q(534>UCs&ezFVv6Z z=CV-4(p+_T3&I#bnW@__whlwVD`&pTe0B=>1 zbg;HV?Vf8jF#&v!iV7k$z8Lg?S6JD%fY+<2=pxQ8sXW5cRtHkFg&hq-ktGI5D1{!4 zO(9g^Ehd@(61|Km@C_l*wH=~*rA0LX90^g~)S*Gha>5E+Z>A%_51FVjS32@ruH=IX z6S6=&N$8v0L}5W>R>`g=N{PN=l~iAgbw*N^$-ZKxL|?H=sxNZKcqjXcl@fi$DyhEKy7kF@ z-~F3T#Z9$XMZPNFUqFgdW!Dr}4=Rgnm9P85L%o+33sic?j_=U@NH0fs+oCcC+@>OV z_4&$-#-ohp7&PAyM7_LkHSZ%p92PINq_I8d#jc1U5EzBPj1h2;TVX{^vX=(ps0a^N z#8L=*Q6L1lQUio|@)ey|>O^9(lOysH4V0x!L~(B%Y_ZCZ?Xg+$ByD7f9Y@W?=>sbN zp~U2DIrc1~$ZriR2$eM#8>TCf6g;>dv6>|mui{D^mFq3qG2jLj6~j$~_#|<0aEV;v zeZ6@f0phSI3K|=OUYuaU{D!HtAaI~f47XNcMNGmGNF+sgxT55eC5|&fkP-(%JXzxG zqYu<$pwu&vow6+H8#=64m0g8@mB~Ts1k1e2W5ZPC$98gjh|Lm*46NQ{)W_n1K?Oo( zjmN&zm5=Q)wmVq0aiCWRFAN-j(30WuFCP5|-9Fu^e-q(t(#7$cVUYyhp`!9qAFY24 ziok#CQlxcnPz2uBrAX_yK@s@7F2y( zDvH1==ftH;r2jAcEe$FCEBP8t^t16^jXt&ihOc_)`@XdNdBd z4!vn$(?lzPZ!^&paIK2Ui9+kErnmxluZoK8D~-dir`r|45fv4BG+y`t(*q8F5YaU7 z4<=dxJUA=0#M><$8gEdcBoBDGiKc+>G0_U(-J>o7KyKiZUs#C<2qjtZf%KTT^ZyF%reWRHf#;ZL6iA)Q&+jb;Q>F59%*9~1 z$Q0V{W_WHKGbWDQ60#VBw8)qi;*Rv1-vLE4EPokO_H#uyx)4fQQS~u{+YoJ zSh4f(scftM-81yZ2s%Oik*K(;=QVWW3U5Ocx|;M z<{XnsY@kM>{#q0x}PZPG~SGa$7L7qfn$l0&~Uq-*q}Jx7qlN0y#Y7 z+f=vP%6GXnmi1R^@{3;M?B~7unIh9arSe?IQfNQoc&yp z(oc&yp>SlJyXvQm0(QiB;C+ZT+k}S8BJ9^Y&-zXxo z_YqrUC&&|9F3|!ubdfDOq!Er@69jdt1|_D%M`$9VoPHRz2#&}^D~{)-E}l{1d9NLy zM}dU6PyVZ;@-vAYy{USM{?*?*BK%KU_@jh>tCor68%P@VR;SnAAwOptc|?coJfW&9 z%g4I;+BFKKZ#Jjom1GJhAbFk5JWlv*y75TNraRXVEs_@+mp`a=q1?-kN2~kOc;F70 zb>Q$joH2qFaJScZFuOmE$92EzSkXl}T(pL>c!0Z%Dg#`=GWy|+!CtNtsLKf7aZf0p zW?P#X;A?FncIIU?E(&@S3=^?C?zndPViOlpUq*2?^<@;7)hTY~FQ=B6=JMnrg}|Am zh=BiDJv7j`&*t}X)@Sk}rZI=tIu7G-c$cj{<3MP|tIwtB)#u~3`iuji-ZlLqb99KP z=Td_Lvtc%;=sezvX`J*>RI{0*bGohV<3O@In>jjf4(8O|4}yrQdYkR>#^)NZMKzL_ zzY8NCy6#WoQH+-&JeBj6R#=J{!)~ug8MUsdfRorI%8X!_QLn`YVJFbZCPvhd~i)%dCXSvBa%?Pj59i=>U4BTd$%y!!j zA8hYLk8)?@B_awgY)=wb!lz?emnp5A^snp$fo#flt+i!~SNWf!d2npq{K8rOi| z`QIdLkul?ATjaaww6FGyY&ZQZf18cv0wE(>HH?S0`Wa{VocB~eiwFBk9prU3*e&2+ zt0+Di@Z;X)q|Z>JM!MF?NEaRI=&9A6it*hr^4iHq{U$f{bR zUF_m=b%vGV?J}g0Ice)76R1+>{e^}$v*MsAw5~bQMKPW)eNc?2(bKBq+qW&yKwGT8_NjA5fjI9xGJ!uv^x4*eS9SJ%q$! z%Chv(5`;C2+6hsDdM-7I4~DaNp?e5n#S7nY)sjneS`K{TgW)V*3?BlUcyW9fY~ls; zA+XscT0oDi<|e&>#?s*|UQr(ck9dWB2rl9k_aV55S77eK{iW^r8+Yl63nwUYss6co zT1&x;h^nPtM0Bn0P8l>2aTks~q=@1mL==#whL48Op-;9 zJRhwUE+XJE6&0p5Lgp^%WP}<{&fHB-@prTO3HcS%IKPMMrJSbzl|k@XiKZe6p0}vUuv}GNuECp3CB~#y=NM zrOfS!9(EW4h-iBW3lKdU&lr&?cqN5FGL%zybtJ|n>N zPkc!qD}u})7K+VsS^V#E%@@7p2TgD1e+eGDtyYY6Ag<&N;VS2=5cVSR2Xp35qUaCj z<)hOv(^A^Y{nh5)otopmV=5y+NXhEszhUR?8radBY-zmRe`-I3)9)gW9i$Sc9kEL4 zlP+xcI>JU#)G%F zqu`!jOl-$uE7^|aPg}_`FYYva(yQmn^ToYX2uoJ!*neW{5tA$Tez9SCM~H18>k{}s%EL_V82 zI%KCOO)_?cZQzVzU$5# zisinG)^LV0$#tJ%({8R!`J1Ko?{;IICi^?hqN!3@z{15^$Lv(zbkBP z23C=4CyrAVo((L%4MX4DU!=#_qi;N6aNG!`=fc*0l^hQB0Sm#r}CX822*x2 zK-kIFA|_Y(gWb(le>6fyCVDacSDmu5GnyCc9HL7p5B5sU0C}QT24F*n1`li}o)Uel zx?%uiCqP=rNMdtvbz~LxC$Vw(Bn|=}m9IhdQVHZHV*iPq9geXo_b9PpdgtQP(NB+9 z%@T@Naa$r(uD1i_ByfX@itPan;*-5a99&{cg!lF4eGG`hViQ4QW6+C-g)qNiDoqI7 z9FCs7tcXcC0*Ryu4_B05a!UjSLXa&H5aP)#(QlPsn7uPh)G#M!V(PEeGm)$HuPg&V zzUn7+F24_Xd2NYXf;Q^{@O@h?M}c>%sC*{_t$Twa@V+iZTE7j7z~6N#(ptD-SrL-# z&VakBsO&lp*mG{BIJYv-xs~GF$~@;*igPRToLec*{^cc}b0fvMk>K1&ah7+IXfh!x zcgumWEZ^Ed3!ipq+h zHNMdlfv1^h6nM6Yrs7h9#H)=&3kVsRyg>TMNwhLrbR&vDdMp)tb-KnnZ2+r=en4RR zTr+F}zhWty29l(lL^GpB4|0n#=5>O){hIlj!Rr4CBH)*AB=;l0V=U1#K$0Yj5->1u zOBvCSa5;Cyiqs}fXJ4+oPqwtwfjBB&r$GakvJ2VKrh)%LlLk}^%{W~n+TB>_7D!LU zOEPHClQ`Shp+QfDCXGtB{(aBna@{6Z&$GI%-R!rkQQ`cQd1QS+vfCh?OMp_w6XW+G|w*b-@l8O1K3ivEpizTR^1x z$#SJj{~m+0wX5?(m%xhM+D=UMGSzPfJ7C3by9@Ce)n`h`X*Te0^`rc4?cm4XK;9>S zjI@Nkp+lo)qc{O%fD1hu*9ARBn**4foB{W;IWq(Nwu;2tnhq1`S}T+Z;3gFnz0zQo zNf$t>NESEX6KoDn0H14fP-3O=#h?fLanQT9LxZ^}y#nht;Uo&`hw)0Boia4nhfxNS zz9Jzs{+Fe<0VIKi9*tW=f`Q*O(FBm3ON{{E9@4v}LoV0a`;%mA5SMJMqDiBp%~1|)Y>w*&@c5X9F_ZnAN(__hs01IS`9B)5Do zkP|uv6>f=#W;vvM)#g@%6_4dktUjojCQ~AMs^*0}iHL&*SrQRF6OoAG+&u9^<*dF? z|H=!LzN3&?E+QavTvD?>G2vGvCj9yi70Hw?j0uy;DxCnCs)Zg6rkCgenOvf`Au-cd z_~=)}(LYT?eV#?%^wGPQ0UxB=6J|B;t;QKZ!b%)OaYXm|7aPmx7A~$04QH{_=;N*F|r$okK<2G7#Ji2#`rt1VO0A5h3)K+bk35GIU>{Mfm3 zqS&{T!NyMPy%5I!3t`+tA&l#TKEmvnq=&&9%84WU7vLvVRO}WxFj46Fj!XxNycT*i z7){Y*o|9k^G0$!DSDQ4btKlyBoE%!ZB#ZZggSC~NjBwf7iBBM_L!lRE;LnsZ+>>dF zmBjg6eo`-|CuI-l@h90=td!`>tGH$F(^`d0v1C`VQlcxb;<_Rll9cQ!R!Vf`Ra{pe z*TG`ZTqs8(*c@PYzNXxT!hW1RbOEnN3c`3nD5r_p(yFY-0(~T3nRZ z;=1|TcU+JQ#YR7(EKgU%A^9k$MVs2Jn8cNJpYAUnu&qaf+46p9`Qu(}wYz*vv9k2W z9-@!qh!RUL3FUB{-(@0NqDtL)!j2Rd=X}s%&OewF!Et(Jz8A(cppsa;VRFAFR;KK{ zIQawm8M=vkK7Z7rL}jpcsed>VrOL@&+!ZRXv~h0w-K0BKV@2=jOC*Hx;Cy2>3u&+7 z9{xgQiN!Sr+*d`#EeRS#BIomQig5Krb2S3QTJABf%xD}G^kP@Ubdssm$#xb*y{rg8 zI0B)ayuCtiu$60)KsXk}zRD87WcjDoza;#_D!U0}ZW#Io8kM(3=k?qzR zz-Oo^KWtwW9XsW$f+Oz!I@d)6WaE%J;2(O!vSLI@rksSK&bCNI@t||kgjIez7n`NN z9)TSvF^LI;%A1v#EN8Lj#0rNMgvvE0HcYQPq~O8zh}A5iconzupz=bCb`1Cu6&16U z2JuPa;@}dw#QQSyJ_5vHF{f$NgI+EAN0{fC$|wXzt>WCbRB0OACa>*46 zXM~VH(V=W0o?M~sQKi%h#g&wNg<_k)IZ$eAfG^InQxDB)cc{n(yjuM4$PX_oM2`d| ze4V5|B`rYQOFG-OD2HnRBJ4%%dFm|na4(mw_SSO*Ss#F;Q#RP^J8?b>Vcg+y>IBPv z2Liiy8;P-Au3x4q`L|8tf4B+V%R;6b;ZO(u@$}eE4ok6F;`|}D(p4(zFTF}(OEiNu z`5_k?D=Z6csm88pDq}!$C*|RvAjgg_y86R|3_TDE@`_dO>;^5TFItVdv(s}l3vbiE zGP8i!sHnWXP3wI@5qMpfBCWp;ion0=Ql#~npa}eImttH!o^vb3xs`d&trX{0<~g@g zoLiaa+)8ow&suoSjTGlbf^#Fq`Tu6`Ou*zSs>R=#Ookx|AwYl-0)!-JWX%lfL);gyY|H1c>d+OBL&#C3sy?v*Wu(^`5xstHClCn8|v>)0m&x3cA;+U4> z=#}F*MI+=o&GG(MEbs}I?FHaZ6eUks&jH^0;fyT(`ogbN*uhFAKc$(i;K@0vtl$MX zYPEtJbJXPuekVtT6?nI*n_?ursC!C|Lfxr33UzC96zV>kqZ+AOY@u2TNO)w42b^w# zmI7j&EQ+&$PsqstG1!p-FVD#Ual(-SUz(Ev;+P`?{z^^;xK&Y(tdZmG##YAs?i_`w z%W@QEy)H*#!qqt{R7d?0T-nUONJCC=6Auz%u4OS-^jZmAPq&=*0KQOB`S+#V8|Yv$ zZi*Zjt#8jxZd^Y)SLXr&UzZdJ_T4!H;QNvcU_Y2M0DdgV0QPG+1K>?b2Czgs^f62r z7NIEd3pz-!R1T~F;i`{wBOP$H$uS);5)!vYblqgWM1rI_CL#$5qf(3ngsZucp#W!OD50s5x2cf_+Z6Z@T_x|0)rAEJ3A1`)AcZ7tr4^x+ z66}SwKBR$!#Ui08BuQ!N*U%NV5TPbWSc4LpLXwmw*b6(8P!l99Z3#^wNlFv!O*T9% z(&MjO;Ej;m1i;mX>IHn0qOv;&T2J^dc%$9)*GP^QDT0c~*Eck$5`bd$G);AO@gn&hx<8Ad%w1k$MDFc95+38;96|-lX-H1#a zH(jN2b~TH8W_&cW^bozuR8tE6gA0E}@OJg&Qsatl?qjarm@7)}HELn5Yhlb4y=Vs) ze#t4*_EPAXM+UByJ{{>BjP*soBcdK|x4cvU3ntu>GU1lcf)c`Vm-}K9qVIFTHP=#E z27IET41hIvQ8zbby~L($UDREo1FYEAKMku38`76dF&lqA3HORHUP^ zBBujfWvHcqZ!WUZ$m=q?nIv$z1N}IVb4BN*Ay-@J=vv6;aA9^S2XZ6lF@Vo5hLr}n z5jwy(73pY@8=(WlAs=gOBN0Le$hh!y40{qGbb!l>fu%trgbt8I$Qvnu1k$SrL(-~g z|3H=U7wJlb<1bdViw#uIg=}?bg<}mpZ*!k*SDp(CyrJifL3fX`8sU&ImhA&sd=eoq@W7m@9{k({X48EO#lb%rVfzR6IF06%P~GT?t1Y7j8Dvy7deR;OiJ2K<7d1_5(#qE!)Za@iGR z=UmadR3VpJ0YkRbsz^3-ZA-}|X_T72Sv??$f+CN(O;MR6n`B5x0AG^F5^%Yq{7OZG z3<)dXTa0xP;CqU!G)SAU0$y#bgMj$$wcT50wj)} zl}OI8TdgwS0ft%xNW493G##lbREq*4hU0V1@j<{N z6(wZR^7=4x9)>U!<*W+pr%nER8U^%Ot*Mk*4`Tp?W`&}hS>a%EW-*#hZ(`QNDD%*) zP?R%^(Ze`#W&wXwOD`3zhcV_uvqDkMtZ@EllTXAjfvj&5tp!?lE3G8^0Y7Z0Uciql zDmz7?wfofzyovI@L1-%Ct0t=t@I8w1Z+Or^RQhNIX-iC61+d5%pKrkUGGpumJj2Xb z(O#p0IdTGllS@8^lLLU~8>$TWMn#EJD{3^bEpbqCl4{Qt4gsP#KX?IihUpywL`6;< zX5tjpbXw8hkmw!S`A)sc3=4=^@)x|Z;})ZeelULf<}6yj)~lPU6~xrUp91et$M)81 zyL?OSyc(?o&Fy}`r`8eDdVvx40iraM7`lCuGLTrYHYCBiPjMt0Sa!l@ay)=!PFACS!b zPv!9=Y6AKbd1Or1H05oj6KlYqwYrJ2^fv9fM6VM{7pz&+S)+NWaaI5ks+3`NKCB$M z&S>KwlB9mz4b2u6Cy`!vUX4~RydSVI5QH$aYO2oUN9$GgF@Wb7stkCgq51*07^)2T z2SfD(PP%4+*DUz~e7m9g0XHa0-WJ#>Rz#{iE| zlvhJz;Oj+hT7!!9q%=NRM_LO0BckV+e|>;vd}#?_+b1)%^3(+I%jVh=K%B`INm{ea zwLZXi7-|XN)r!h^X}wZ?keLRs=vMsh5NDFI%c&(#~NG#LGf z$;+!K1^+$lG!SS}2uPSk;XSKq-2XXZ)bk3B;R*}yARy_G2cm!^!%L&_S7Tj55|&vK zdfu!c$q;FPq{2(1K{A96kW_d&8YH9Tyxg<8wY=Q3I!H^YS0}O;<`WuVVM5z#TLBiF z4*7dz?r#**W{TKF@cw}{2iB5<6~?|`K^Xm9U?TqaRKxh^Tb-{0+L0E{#0PHC>fm!Ocos-FeYmeP7Q)(_LTQExvxOI>G`Iv~XXbTY$u_rx>`V{NAuJvHMgX%INXhCu-zdVjroeS>*l z2E;efvm@o`j#9$W9i;}2o|15MM=9axMym{b^9Ile96hChqo;(9?yFTv{H3lp*>+=x zmF#j?v2(_!N_LZF02{mv8fRO{A|qbq=?RVYg5r{eypb<#WaLB^@JXLu;Hl*WY`~uy zY6y^hzEH8hzt{Hn%K*Q69kP}HGH8VnkS&lH0m#lrr~uih2o>N)9mzrkxXDmU0NGm# zRiiLaaZ)GS+Gvnx`8tlSuR|*U?p8M4dS!4irlJ*O_q-f@H<7zZsWoP64fNw0)(LhyaY5B!zHM=QW zx3RAnSEKQCQ_};ux=2T(pk{4a%_dy{uhpvyH!zE&1nsWXH1f5u=W)8=ankH~fjLB$ zw@p|I^CfQJySjh;vtHXuU9~y7{TZ-rP(R=zLzMy940+pV+@(7s`3o9%Wqvd8gsG?vm-PX_LLJ|=nvyMSY3^Db zmh4fQu5R1ElmW?}DBZc5n19vw9D=05ZeBJRfF#5-U#k)NN**adf+KV{R>M7_N3v@h z)v@i7_{0|8s|kJkn^y93KefPjxv~QOxZSZV1^lj}#+RzYH5!z)+*AM(7guY*d6h1W zt8{4`r=@LLuq*5aXer=36y@!p!QGMALCt^Mig_vEO^WiYG(M}vEUbVx+wFk3N@K3o z=TgAqYzHrNG$=BW$5eTR&d-9P>X;}673Srs8tA864$6RcTMkBctR_ae@4e4d2T!>^ zOR3BXH+(4rUvF;qUaa6%dUc5hW{Asu3imp%wzBMfmfrTc#Ysv7D0t;))KYBW`vYJ^0NQE^owoWZ4 zp#m%vTR48ig8OwyF??j&IyopK3D>47HT4NCBtfnlrmC2sJ#rtd7UpV+4FG zQYbRM0SgJ;xtfHMFk#-gnk2kW=N2!KMjBXa>8=BXn0O!xNC*pyYc)mB#DzMHjgl+$i@FBWvZK+(P!%^}S#VyXXy7 z8tfD5S;ASv+rYd*?bWk@Wvk{{LU#{SH=kkdE&*gOEn^Mv@mj$$7Xbc7Q6ha!mBy7e zB@MAT{$`#Se0jfZD~AB{NilJJALR<;pJ>Jp0De#{5VP)DO@mzRo}Gx_v%2GP*|Qo7 zio`?&bdV%3mIfJ%xN!8n>Zs?UfvXW$nAM0Nk=Xdo?>vpkWfq%gBSltUY9y756#RaU z3M(e&>iKu2{*`g{jKwwCKAP_SXfy(@Q~~w9V6V|$_a~*#7B1QtThA3*uv5A38{5c} zl|=Hjsrp52+^E;oIXE$avEC9Sv}R{>S)o@UwKti-a97}w6eD%#k&VPX--1*I#5dvF zSVhCO9j>#6~tYPjwiFw?Qi4pR=Zy~D9QF;*Rp zfyR=Z$}A6Sc>hs8cF%K}%WOtl_%Q{UCK8{-k4YAXaX=T#o$NBvm9KR6?qsgMCzc(J zKV;u$d%mHVE1F>8!h6a6_SM*g0Uj9&^T^P_Zm>D46~c$*nk{$vnMM;3~@Rmrie zt@O-VXqdx=eE7e&?J_DM$(<@1(!|z#x^mCc>x9zDXUe{WCPvLGoqY27r9J0$l=kYJ zb=0)d{C)OZ@cg+m=IvVQn!C@Oar+$g`v>-V=ggBIJF`Z9iD~Pcq4ks0By5mFyHI%I zLE#evuJNVLxsg#G8R~nUa-N}A+5d$k@!o7^`SCl;NIj2S=k4aDHTOyK3-^HYt0TC!~iRIFd%IT2)3R{~Z z+wnZNX$UR~Au;2@g_{MB5#&7%MR{bHbcErR++hIXi+2!Vc;k_wu(j7O@Tg1dhIRn( zy@u)sEZpN>Rpo~EOgpoZlRp}qKgf+dn}__YV1Q!*_Tq`#eGDt8+_4&!NSD&yVFwh* z_sCF#M~2Sdpw8c>SBYdu@;>dl)<`PfR`76(Pc&!8Ts_;AYp!wi03v9n{Bb6E)o7jk zf05h=^LLRBpEs_w&tA1s@L#B-e%Fk;KdIXKUa)`DUhWBKV_7{HSnNHdnY~y)J2L!P z8iD8JkqBHX;oAq)KFQvu>aI#ZRj;WHXUJIZ01|Mdz8CDnbg#JyEqO{B3bntYbZpdUN|^%|0=%J$+73%h8L(i2iE-&=G= zB?I59^rYGS$-S!klY1a8$d8}_(P6YG`n;aGNf-dpZM0(hnkVGb3SwEOe8NP2!*%k^+SfD_1TZoG zhDzMs3~WDCc83H0nI$*cVj+qG*-PwLKTeLYy5}z6+14@JzlTES8`S81wbCL&2yl3c z_*4`W)&X=xA$5Bd)5Jp|q>=M2_ZrQPwvIXN$v0`Pi-H_2S24+8Xg=}%ZE!m{(M|dja_&ro8L2wc4>-?ovbJuG6#mui2xfe!#CQ%DtP|-vVP_iZT8@eyp*Q zVvK(jzg6w|VQwnmkD8hqGxntzV^d?sN{TTyb-UU#{u>K?z~x~e;37kn0bggRe!z`} zDg*BFO_K$M&EdMybnlT|i3xe|Ob4ap2H= z*a9A5s4`&gW-s7^?`1@(D^Zwr;U^K>Q49- zq)a>=4JH!zivS(46v#uKQ@nx}rj#bZC02#%xWt0H4DglTwMcHP9;RWv_xnU>*()@H zEArq1e#B7A0DotwrTc4?39@xtPr%0NBN^BQw`Q&oO9|}8Dh&o-Vy-?^ zXObiH83nK~qwKI5g^9dFzH#lo=0KY)>>_yoH3twlnSbB_4AYL7x+9u{kWYm%>v&=s z%s3F!X43J*G*~borcID?1HlkB^Qjd4_e2falk(sJ7J|2Sm%z`K1OaMc)cGd7n z7bwD2{5}D=LI~Fm(;#HI7BIw3;<9#_20_co!4Nr5PJ^)JaC=BpOw4=!*Lp2`Rf{x3q3g_y!EdLpJLi=S_PW!3vIeEga z%dDKP*Nt0F{sYDQDC$Z0a@lZPXpMe;-lvg#VLto-DIEW!avD^Q{8IJK5)DenQ_`Sz zJS7bhrineBs8KwgmWppIugQQfe&+knnvR_7L_s}K8x}l zsQ~OKB?7Rc)F1$-Bm%Ib)F1$-Bm%IblnB74;RXRXr9l8r2?KDkI!?YMs#n?i^(-rX zej$z*WXzM(c%cHpp+m^-o486*E2H-h{ssNsK zOGYhu-cqIUiJxRh`C+#F7~LI5&92?8xLtml^@&)(3l-&M(AciXq6~0%u3fPZMXt{U(!4A)%){;lDTuATbr%&K$rDp~>GWT^7h3VuefPAl*? z4A)fy-e$P&8t|=#JG$2OolNTjy^2=AlMPj2`dVze@%c4iCVzQ7W2){mkI<_;nqo`A zoagLiRa<&%rrYHK_|=B%s-3IwTElhMfZt`fuG;$*zS?lzHQ+BAE`BQX3Uv_g{FuN{ zjP%G-@E@G)2;?nzw{AB0S@y?N&kcI@Kg*`^9gA{5;2lLe8VBgfO8&EKFc{HUI=jIu zUT3MmcbQ=M!8VN#m;pV2KQGeJK%E$%FRXzn7|`HVHqPE2uTHp(0~3AskSPp)>cP`y zUCchGS65NMEO?T%%c?Z;b*+cvCl2%+FEA&1^6LxsYx#23d(V#+_!^j`8xQxS7?@3l z*Lth&E`F(383TZ9JUktZ3ATgi1!QyK>1f=Y(*d%CdO8~UdfJ=Vd|Xx~#D%@YWzj1K zoY}p~HV$P#LM=)+RteCjY_S(4jo;^+0ze|>nfK8R+hg@AkpjG2gC}&`s^LD&BPZUP z<>N5DdL%w^B1zA)#Cgb@SV1^UjjH$Lm)4C4%cWmKM@^myV zusU7>c&TMaLPFydJIh`Y)}5%J>X=mpRY-{k*JwTno+NLp6B3ZAZ4H`amwJ14tLtM6FzbKpb;PuHvGRG7+?m+u(C)Thfl@JDR44lsZj zpu))*B~aegw+2rDoH)WpwF~btw`QIaNzYpf{yRnM_zPCgJ-I7PGfZVmo?hNpL%VPx)cZKSO>Tx~;^~W<%KMr4M z*&h0M;I|xT?4iT#KWw=59Ha$so?d0moTphnT(9Fx!)-Mh3`;?duTc*SLXYHz{bE}T zdI3ii<$tzKB75LI+4q zdOE5xAMHJWjCx5AAQOO$YrqR^9_+Z>VO}t6{A+41kTg?mrQen-D zt<%l{Du9Jz3&)RGz(U6Gkpek6C?g5iRaI*0Y%L@~ZWyMjs0ojJy(XLjkOTl061B1V zZA~PzvjmOY6^8A`Y8Y>?BoXm&cnjUeDr0??#@sX0U`ryJ0m+f4qmj=Jz2WS@rh+U< z_yF^SH9OuROEMV&<_TMPj81q<^eV}Ji6-Dw8`1rM`8XBNZ;Ha(`#h~!idv)*$2C^W zWx%|cd*gsbfjC18@8$NqcmR+`$Q|n^ObvvC}|27^C0JU#c6C59n2b z5BOr8wWY!V57sdysohegag7bMA$ED6H?xKa)JrTnLx5DJ=%hSY=o8}_OAA}3O{p!@ zrYb6tFV=3VQiy2q$gmXI8h_iOQNCBfxfYFb$aSNVYoHdpp>^ZptcvNa~Q^ z;kqBbA?^S(q5_BpdH+xjeI0Ql^AIj23mEDNS>t&0wyO#G)~c6y_gIR10ShU(syepx z{!qGPX>%%{nXV*e78EDn7^Jp)x<#)x$- ziv=xfB|JG1tB799rvep|Pn~q6yR*cXSXqKQteTjz?-GLzgOVv=6M~1?ZziB-|#hs#8nUO-%Lph02 zxhgWMQO$?KH!A1*^*X*JM}#y!WJZ+%e^I2P!AS2coz>tZ5(6(Z_LbvmG~S-80lc|L zN28#o!Fl5+mF)}0F28A|aeb}^upqI)Z(4t^Ow`S;@*2MWrZr4@p(~6sQqY(9J~$*O zlHZty3KG9*4Y?XmcA-gk2adI-53VMT#3a!!0g_4fsC*gV?uwF|L;2OIHu9J(i(?Wk z8%W>9q?Z8^=zrZwG$NQFm8CB+tdeFzPR=^;=)ZS06+wJk3&%&*+>{($t@8e zzIX=_hBqD=3j4C&_}{Eow?+ZqZn&-*@J_>Z*MQkvyR9ADoYe9DF}E)8>Qh_5)ATlR zz-~ka0GAr7AMhqSZIF`~8r(O^nuZl;+dgankQ)+-8erkxWG^*wZ@ngN<`PRL#-FM; zt1=8X*_|fRJrenZ!Xtf93jc6-yxC4(%HLA(uQ{r~;K2`T7hEFxs_M>wFBN(iuQJ&c zKxDD|M%*8BQa~(|-M9S4Dz*E55PCx2!iad^exybeZd9*IMLphbhvMXDCLwG~O4ycY zyB)qn;tp`hc|Ho_$B^(HtVc`oK8yrTey0iI=={tHkRuJFMbSnp^fDm2jW!G_1|U>| zy0(fyxIHrDY+~Wrp*IM`EE#uz=xAsmdJx)bXhXi%Jv&m(U#M6AJ53rd&lB=m1+gsu zo#vIA37ivu!|;Vl+?)&B7G-xh;4fNodm4$NK$5@Hr1L3iwERxfS6cKtO$cy!tlw!u zM-);wB{5Ar6ha#Lou+%P-uQQ#{}BZ_+BU}|(P2FC{hUfm|4x(M8#YlXQT5XD)t+DGrUL$|sj1r|#=aC|Y--F{NioKz?o^F? z{9-}L{U=cc+*?uUsP3Y@(~L2iK2FxIWjP68)lj{F!-k3kyvckDXtZ;5D$RE* zy3?Ttq^A?i`?Y6R?O8I~3F?8k3)HyVvRF)XTq7b**uag~zDcE#l68V?*G_reMmtBQ zj&_3GQ+o@>fZDgHQ9N*L;0`d2YK`jzc~DX#*txMrpX9nBy-L*pzRys-fHg&>3+d9^7kHWP)9Zv%&uIJTbeiv1bf-i2OV2bgpVyvA+OtqP zr-J-ysuSe*Qk@`w7AXXCN7Mu6m|tc-)R_YESnc#x;au>BdY=pbtW)7v=~bSL03yJ} zFCO|Zjv8YHu*et>{cDwhkjaq_;2*VXMrqdv=8Mw#kMp0anRwT@8SW=sf|*^Lrf%)8 zS78U7W#JHNN4!FQ4+avWn#nH2#e<=)@uk-e*U&a6Hz4FA;rP>58kydRW}C&Y50J!5 zc?14VsYFtHDCvjVce6=~7W?DX*5~PULTS(yJIz5wPxR=?i=?j;+}E^kx4h3)66jxR z_mo2S$n??n(8$$Fw??mSm++Uzoh|?G|Cdq?><6?ru_?StyKcxy0I^|WscWRuZ`$TY zi|X~t_TR=H1?l_R^)tOrkOwN=v80{a{TIEuoeA*$hU=;UPx#dW9}M?$A8?1^j*cvy zs(nZ5RjL5+sfx;GG+KYo2?77(cbQIkUKS=I7T<^X$D5!5Ky+kt0Ife6VIN?wabRqX z6*MByiLcaXonhiDfVudgQ2c78ezRUDlom_<&ZT*=qB|YBEIqTpjB3wB*R$f5lIsKa zPVL*x`W)*_(3fiW?$*6xyW^ip$9t7v?|K3e;%4D#bYHGy`__}KxJ`t4LI^&fgtOCx zB6peiaF?^86WSY;XKtEj#hpT(b3y&}=9=Jg;pl{xA{axH{D~jO1@)pGS&@3Kl`j4( zT@Y8Q^DEUp<%0UuJ2I}>LOQZj1diP4{<+7AZkNA~>wQulXm3@XT}q?V#pBV*Aolys zgE0Z>#YL{b2FY4r1k~SBu6gd~l|of-`^aX8t2ZBx-zsM-E<@Y)+srd_8N_MY(TXr= zSL<|Q!a!`$j#h+0%TP99rQp9uD{;)h@MtA6lK-KV_#2%If3H{BmH>Y4*V*`zUn4S^ zFkA%{4mLprMP6bSe7w$r9i`Q>C7NBMje@E21N`o})za)P_s?pfK;L~jtHDVdJ7Uhz zte7ISa-3Sn>?^x%)HAOM3J}e5p15H+6t!n0Y7nWv)L>4M4cwho-NAwUqjpY~?O5Im zmMOlDH8ycDVdBPN?ES4yuysPhVAXYoWmnf3dOkPuyoX$2mi$Tx5ThiBS4BaLMJGJ+ejq zN$jF0XjTukaa0C;i=irjn+;V4{Dz|B1bTBDx!Ok;H>pX$XDUkQcDB)Yo+8`jC%}8v zXjO7z*|KIDHvkQTWiCnbbgP=G|- zo#00qK1D-xhF;yB3h+w9b=82^7_PepOqkudDpAJ!)LH9KRmMGfO+*NiU3C7MQtFb8 z-}?6Ky@B|KE2Q5|=;w`1js>&o$m>+imJh}xYIC#hYQAfCQT>278>$!ZHbrGOWwiSL zu)t?FaR(NQ??XJ?+3gmsj^87z5Adai8UQ@YPJgS~kWxS}d02s;I z6Qw@AUbzv>Ef`x4{M6ME?h&8d5#Hwh+2Q_ix4GhjzqR$ZyG1|3Awdukmnc(62;QS{ zegxv|me@wQAjOG+9p+;1bj_10s@4yryvl+)hl&!N(@JeajIoD4h68+2Q}Q&r-l`-_+(zP_0ZgU$9ZI#YmSvI3Eftd?n5w&; z8EHay_SOl_0ZN+{@x|l(iDzb?C>ODw8mh-B^?|;Km3sMl;mRg;=%G7a>1WB$@`Y%5 zV-6_U25~;O-W|woaQ|H9{@LjM`GEUpll-&Hc@h@y`!!5>$S4lNMK`YbsA6C-01)8Q zevN&H1ZVbd<~BrlJV#F?n|d5N`yvhSTl6ZADFClE)BxZo6qRj~8_8N{jD3Jb#`qy5 zM!d}!D}e7d)Iz|btoXSkvY?h*azGg6>D$84_7AF_t4v=HAZ+sf*24Aei5m@giIPkn zfbgaRc7rPWohgm(LvPlum495|O_m#xa}|7BjtafGBNE($j+|@pSX^!{h3yDaRtD@+ zl)P{|u6C?826MVlJfi+&-u3`?DA%r~*S4>3oA`ii2%n~?lT2U_U^%94r0YQA?mJGw z+`{Pn+;+h6+%c7-JCl;q?;E%043A^}$#su%FBipmO%)-^dH z;GK1Zw0`$jl=K3AP6abPaVFe^hE=YYfT3z>>C4fVQ z>H{R5A`XyTX5wg(P9X#&ncid?q*LesNv5ZxK`t{bwDQbG#~oyLr;a2!yaIfw4U8c` zQsl%}EMa|?iXlMao)OX_wDJR1K;oJa(%Q>HIs{0xGeTMfTqFX%QAb8bNbAQogoXf# zPDV)UIvpD#5pc$Ye7ICYkO>8;;KUUh;jI4#~--TkDykIBlq*~8Q!F*pwYCSJlYN(!<8lpt%d%?a?E395JSO#r9Z#Zb{d&8kx-%EKN z*OApyeQ&>FKb7pgS+Ns!g0AOnR4(Mby>#}vwi+#}puMzQu(YTG`^x|zn5i>zT9k#6 zgJ9at$Z3&HdE@xHRULCgFBmeI(bFP>CLRQ7$;fGu79$72jGB?tB2T$BV3=7mdRk1a zIXxJ1YKlQHoHIc!BxudvsRRi&ognE*BuKF71W89CL4s|NAn8dYNU-SyNk<|=}9C=uniIJ>J%pVexq+CC0u= zp5L!)qrrB{Q_^4;(4 zVe*tT2$QFz!PdZ2hE9*uJFZT<^C|-toZg&rdRxlr%_*n1rJUZJa(dg!gwvbb5>9Vh znQ(e@Tf*sW(e2G{>g?{h(|`r1x1^ljk#c%V%IO^`r?;e>-jQ;8OUmgTDW|ujoZivU zX*`sp2teiuxzz_`ws7~Aj~=M#`oV;g1%(!?#6Rsog>ojH9rB9f1@a0bZH`hpN-I5{ zHo4wK@a~sC(Z(hJ+enlHdlh2=!TguAfZ;$5t7tP1Vij$sLg;C8&I3Jd_Csb82xgL9 z-9cQ3cMue3aOj}T>Vyv3EL6F7U^pv+gEo_sSp|YKBUIC7Witj4lv%DD425SbAXozn zo`Yf=Vmu%?qQVi{>`{yX1pBpu5>PC+xyfKS-$Dy*c2K4T1m|2Z&}LCHhHzmE2Ss7J zdy!511Aq(#d5efr|FBV&0e@wvrD1hH=#yj0S7r^s)9p~R40y4jmI4mny}aIl({ehQsELJz&9$R z_kqS+b7_E7y7zzvyLrz{gEID%G^k-Q8Sp;cj(a*9yX$^jp5*|t&6lT&YgW1MVSrev zv|a#>x8?!>873wG#Jlyh*9)M*_>iyxGBm_^z!&Sii}#!cBgM<3!7veKK#uf09gSP{ zZo^BY!PxSYzSau=gyU*rX|=oM=$LuT{9OYl+3v0F57nVY?FU)kY zuaGVeGIh+lXMt}^m0=32jG!1_tRjL6R}DcikSJ$CQJ$D2D2k4|SwT^9f7Fr_RA^^d zTk_>QO5^qVI6zQW@KY2_*Z-~WIw9x;!;jUcrrZ!IuCu_u)= zc-<-*6tYWve{{f@_%*>3bk*XPNX_L@0>g)-&-&q!pHW81B zOsoPUG{z|! z`ktXQyE{2&${QlxDuj*~7(wUL>j>zeSobIi!N``C9k~o*@=>S(>n_yDAs+-ATI-EYwp3LPLx^K_y7 zJ5}xTJ$~+)rQ_z#KC3i)PG@A$S9Db7bH<{0^wZ>b=3H;6*F~pR`y0 zqP)Y!jker*!j{rNV#vxr2t-RT+0C08LEe~7EbgMHxI`t z^-0E62ISQU3HgRqVaPwFC(gI)Rh;Dv_F${?G6i*4jza3e`qG3jhO$r6u2&mLv>!S| zyACyyGT=@_EyM3S?5t^7sB4{4*Nm$S$c!caZ;br!)5G;Q>Q&s}lzZ3n|; zrm>Uq87{QxeC@i_NFx7xwCjaNQU=^JcNV|%v$n|pc~<$?e?v=jg1lbliY`&Z`w z5a1qqyf|5(Cys1Arz8IJHwf5d0?L305XD>134;S4+qZUE!_SFcSIV#)VUn2)%aOp> zH7OaE*0(KH{eZu$BcydlP6*hh12mIJtHXwEKOnPfMo6F;t{Il3aSWef!*HCX!9L_~ z3w&2bHK2Q$`XK!u3detHSY%-jBX*j0t<>wTr4{^|3Dif8yFcWn>(O}o$PkV%e_|Gr{zB#Mt}mO(evp^g89;&xM^9qcxJ)O? zoNIB+745K4`&d<<9+pt;lN$*LW1=))9$1u?9a728FpLo-Z&QJj@%M2sWWdun9ea4fP=L42|K0 zlKc68dn2+VG{kO`dQM2hBK=G&i)c5H2%i}BQ1(xS@Fm>BloQO9KENf45*OA)E__Bi z8CV_i>Yh964_!_8vGJT5+?150R6uVk>dN3- z8jc_9br**|OFBT{snUlgC_1!3(V-D&D-_wLC__I|Q|{5Lyx$bcf#h#SG6Xm?uiuT4 zXextgt)oc=G)mz?5*lC1235mE?oK zQ=KzHR}~%l+F*u3TOs*7qGNz_)s%NyW~1ao@@^wp3iugC$;jUxiEh7lfsg-?UK1ya z9RZ1CEisfrswRSFslZ94bI)wJUxWz5(prxUY)fo+W0ECd#X<5E3qrIhWIr*iBx!R( zs>GuZNX}3}ljZfZm8C=UFD!yZ%@iM^yg#%Yy887pti80cZ&3^2mDq)hiEzU8A&XTM zPDtKnBvCl$>4d}*n6~g}rFxEDWzqxeRn#Q;#OcwGX>dOPKUxy*K=MRW5!niNlRX_I zi&cvJVayeJV+rb2rl1G#Ju!8d-}GF#R2i9xvgm~qQHBjunvOIQo9rba8P2U;60e>0M38#+sGdO68R*8)_>=C{azFm}oo!JoE7!GQUx%3OCQ01`2K z`WJTPo5RP9pC4ji@#f7%5Qui*>Ye(rUIbok!qx}wZ zzX$NY6tzofd8xzEqKNESN2u&O%J~BmS^?aqD5pR57j{Fs#1`)&t%bFE6%xQrhFTWd z^aoW;&?SK-9HHc1_4o-4c1U;hD9wyIm6Pzn3e~2Y)ykH zI%Q2yEoDtlEw-k`nKrb>uSj8Z#+8O(jk1ChC`JeiBiJu zk)Z~U97~7nP(lrY3grtblrE@Hb~w8x=21b#vZJ|Cs2a(Bk}Xq90gqFZJ4RY5vjAHA`)A&z|+hRb1_fKEAYibuCVrAmd~RGSuMN!^op!krhur zAmNf^thJiODUS?8=8>pTdjLSyX-S!KM0T!f^gLooZ)U07uq zHY~IBXYqDZNk{56@k-+Ut&=bdBp!i(NTuWxTK~0r-+#Sc6Yr@UuX9MnZWN*uCP?zS zizfCJ%6)-e_bzp77JI%zWM*0<+0^tDaO zjy6Q&a`&f)0nLqN7FNr!8hD3ZCBA@!epjhvS#8$0&#aLJY$|kt z!b}kqE;0q`&|0C!a7w&D-N@*-sL=#>KuZ1}o2y8$kOOfi40Q{-!g#m95(a&<2K$3H63djsH*-{I zo6Hqq83+n<$aP{y3CE+5gkw(*(n%Aw>K7u+q7*$cw8A4pJCf_zYt?a5Ea#noq)t90 zw!IX(?vZGqx;-)sZRqQi`iUo+bf(nTc`b~>q{38VW0q*tb%s%?>%?}lRM#2$KzWb} zS+v7E2rASks8CB{wf}@>;B$JFr5&(qayEI&LKwzhP{=0wiRBVO|3hVsx<}0JqbvXw zKvLj7Z#SMK5FL-CK}|?N!_0WPIDQpENl|#Zuo&?2mzpqt#HY(#yI$P`saXooHe7e@ zX$mhe+|jj8b#gazc46p(QxTf$aG_~EDq`zy6pp!my`8C5*6AvWmx;~fpbGkl39^|k z7gUY~MVk`@Jz^Gp*DUG*#7N23*72k|H?t=;Gul^T=7s9F3vNx}%MI66`+~w>G+cM> zXA1w^a7WiRsg+lmwYFivFsCB4ES1yWtB6!rAVx|~ZyX<+*%O-?L;8sEf7h2uDt9!aZ z8!I{mzr54kGlaI}J)uQ}q@z>(s z@_4<L3v~2+V5%}4Z?gIT;dZk6YP8PJ2>~xP@k4-D7^)92 zHzGPPh>fVEjHs9qRWqUjh-P_)19-U^Q31@2h@SLd#8D=@2M}Y%l?DL`hYL=R*|pp> z4gz9KMo25SV=x>^*nuYb2?StK>B5B4g(!WS+9A&zp973)XUM`Dt=y1>p&@8Ps2Bo> z%tu)MLl4?%nzR*`yd{7Xi0n}SasDCKN~~@Obqor*wFS)8J;d>W>?n|<%=kOt2Vp~Y%#3-}R3^#guXQ5kO-#n^|wln;I8!=U-l2ROPn zb}Res!@p3>|S z(TFbj5flByYu|`&WSS?0;0Y>lKSwwszk(U*l>fypQLChoJV{As$lFehXrXzs((d1g zc4Yc!J4EDhW<)$PHM7D|L&a!l=o$pi)q&B(HAr5dq|ID|=47R9>Ka50(&nx~#rSRL z8U&2qCayuksBS0|g7@nHZz2Q_*9F z07B}zg%CjcLtXjy&%<0ycjg8}+mv{W6hg-68Pk3U8AC1D4=F>bl@vnDh-xW?5HtK* zOJT!d_^-cnVMbv?DJL_4Y$)Z}5s(d~PytfjkNn;eJHD>-k~=H zJu=K|$k}ulVHPvitbEKxQ`_cYH+MqGgf+YA+?Me`%Y@aW%4Jg|Z__(P5Hn#Vs+@_} z&r=x})MWIKv8K0-4RWTfvEu+GIcX&h&@y>7O$|g$UH_;Vjw) z_YgDFwU&FRnHO8k{a6D?rVbKE01^)|Zr1_+Y`ec30A!YNf1ht0m4--pTNA05>)uOH zfZV`(*3i)LI&<%0Gk6*B@B45&x&(#XSjth`vvtwTZCHX0+$ttsz-JpYsOsFG-m^cCWRzO>zw8?V(YaM>e;GwL>zB0huv+(4tBH?s&P7L9GFvwdn->59k(81??Ir%=vLO_FiH4?jKtCg$wXV9 z%79oP=i)O%$pcEhM6Yg}3;Z&}b=81RGhBBK7^_k#yFEz1q-;l({4)?e@;sHCOPg;+ z!N`yi1W28D-q5R1Yj!>xs?D&W-V7U-V}@;{hq>p$+`xp^2ybYLps<5ELQsH2LT-@& z35v5ee*V8%Bm9&Z>^u4a&pQBAFW{RD)epEN4M#0^)5(NNbfj zx)czHMF}7-WV~Uu;lsX`zaGG+9+-vJG4|AG9j{o~Q~(w=#IIe<+CLNXRlOcm8W?St zlRS41_kB5QS|r!4a+MSO`879vLU_K$aIRy>>6Kh|A?w@cbO=Yr1@-Hd>oKLXx5@GM zn0(NalX>u+z1^LjI}00VgoC^tSQ?$&jQ)da*%rNy8Hjt8kkF6G@I1}Um3keM0m9!a z!=WzJGMVO4lOG2(a37unv|JBq#(S|y$?^pGY-MXLyHFpdTr-n3fexsSR_gsqqn*p@ zlL;v#jH!KsBtl33dmfb8P+z57Eyf7)GjxEDnMVkj2AYQlsyD0n)=~iVh;p?$z@TS7 zYJGsg!Td7T01M}~>vdjcQkF>(@Rd5W-+)NwJv18>aoL%=x=5@3^oLBcs&8y*zIyFegXnQ!9 zj%8jri;n4kIDaI`0?KqY2T)j^4#WS5|S<#(3t&C0s^Ahcuq&+ezj7|l4igxaicgmf%>=j2l={j9W z{9>0)SZ9^SPyUI1bY7{Ux?HJeoIw50=L=+Rg4NQq@W;{L2^G(JLBuw#RjR)wM;Kv>hP%^!@ zG9I8Ednth865R+q0`Wx@tIUZHCuNe~X?cF7Dv`><)Fd~MFrb@|GO$^gnvg=mYV_#Z z<>u6}>i*HStNy+#-;6!se^8=K?NC9FLkK;aS=ZZ+z+2(r6s zu+?DQ+St-kn35AMR)}KJdV(CKjcraO26h&-CaV)9$LLAczWFwfQ^stM0|7&*iA+Wo zD_;{*NFH4r-J-~%_YYefzpv+MW6u~+-l`mj<#iDahp7Hw;wG_40YJii;{TQV$g^|} z=9Is^H1V!+_9%k^oMByQ6#qL{c`wS>all+$^{(*=aSmG)<>1Ry42xW14{%|iaeV2_ z;Tk$G&dmTsRC}rVr&u8?9>&VAq?nU&Tu`BzaDBqmRRCEK<%dLotay{%&4k_Yg=*@1 zLyaF-O(47gQ{+);j<(O@K4BBjyaviYdY|+OO7et|FhX z4Alz?&1^RW1xQe&#CKL%fXS9ehAK~2mH)0+dGGD1EVwCczLr#GNV7TTY`e#l$bRkqi>fi)vxc z!Q&+31w0l|z2L(Fz9ve3AF9Z>yi@{6oMkL*PZe>;a}+0EtSA`7_O@Z{0GBz z*MNzId!q^q$h5-|0}|kHq5rA3vF_A7LYrE~p}AXW56E}n@(Z8n4G_rKR6oMczr*Zd z9UjQ4+lXWrTkFbz7@2w#?nEW;)vJpQ@FK%?)qr1Yxb7M-R;6OUuMYEva;6p5fE-|* zsp@l;^Q{CJZ!#_c85P+%R~RyEqwb{Iw^c)>9*J7yd1NDT%v_QrKqfKS8Udm-+Zu&h zg%_PnCl6<*wNb-%@WH%`)(`mab3pY1ZctQa8m;dgVuYdRi2tgpk(mVW(*;?5HCi{A z(mp`MJEidtz(1mr5&MtRKwKnMfW$>I07y_;4$(;(`HgygOi7-Y$aA`-qwOo|9tq0n zl^g>t;jJF%_nMtW|A3s&|FAP4hip^>yuUpE78Q{CmwQb%M3*S>j07=uK=Ur8-Pb)c zcR$1w1H4oSAs|BjbtOj_(9usxjN#bVRqs#rItDppht18xc2FF)`dC2t0cDs|8gcI_ z_?OEX1`y)wm8sRFLjP6e>&mT*ia`*C$8456SWnSWIi`aUGR~S-MmWI3>YdfZ;H-MX zy4^SDplmY2uwwU46rp$>*6Oqb*=T!MrF+Cx8rG+O##~pQVP!TnJ*>%wgkd%A7YBoC z4D0ZK=5-iW;Mj=?Ywxh!ma%Iuthy$Sg!R@SuVJMz(eKEY5@rZi3Yq%>$w2C@kE3*^ zeTH5q<&KU{1^I04+#~OlvXnV(w39Ase`Zb+1!}YtD(ZghL_o;^XjnZEFa#Pzl5WPp zqa(yGqyOKH5Gsvf;dbnd119&>9S(eKVaD9epg;&i`GN`u?s+9qTxjU;O_4=Cv67yIlK8X@QFtN|*jhzcAuL-X^E_o} z+3zCDDhR_IxhDlI$dAv^{!_R4TlFfH4){!~^a|jqhAIQ*F3HIl!L8b@TLth+MfsC4 z8dobaI~k+(shk_|Iz{Qp7|nbLM}JV37-1jhR0N8sfX8%1f)8W=Kot8vL=te>F94!N z(ik6M9i-7bOs{Sd1@1OnR}J_{hU=~YUgcO6ZPss1Kex4t{U(n!*$ny zu__h%?MmPI7|u2N0Uu+iA;2RPmEG{un*LaLqeUbZ--r19P5cr-1ZJlnv=-!qfV~Af z6(p!K zJpxWF#Mk=_0^br-De)z4ZcfB<401Sh{880A9Pr&{T^TUfxGan#QeSJ5x@!1I@ucp) z9|}#v;cKJ##V>|v9BVcg_XB=RQQ4$T>kBy{;Kak2$9n*~bVX(2Xbm4>ykYQW zMW*$IrpZoZTNTT`ZDNN-JMCP1BCd8f}M&l9I2R2E=?9(@{fZ_0`ZfD0b7~)x(l+%CBI>@mZ%{EL8s`lPMnn_eAYr65777KELKC6kOmVL!ddj?P-twg>H*!+q z#czp@PKID_C7j{X7Zdt#%tBL9TL1aJn092YJK2Wlol5LK-&Z7-H~6z}%zARjC=&mH z(u{oM%R=btzK#LH?;o0Zn0@IW;`Z2==A%&WK?a+lwN+KO$NGaG>A`I<2bZKl_A`o4s z#Dx;UqU=-1S)3s2)`ipiEDLI#3q=M98AtweRSl`3VyzmpI3PSz?P(=G&@!aQRMKc* zQ~t6A0Hh|eC6^Xi6hc7uFK#>XPh8oVqp)9I5(mWiY#T%i`-Kp&;L1O7Uxp`a>*XE; zkZrxZ=>dp4&bfZufl(2O`;$X;3MD%Kx`DzI7Kf(O14Jwhh3O$%!|E3V$cWOIQzC?{ z3ayM^XjvG>9KR5lp`>OoI)##Y;s@#sVqZ<7iuq7QZXbc%Ws8ozr|PoiRboo}!ho zK}%j+3LC`idz&T`B6hun)$d<%bKFvZ!ntkix*BV|$c#YL8E1(N)>%5hcj&#OJWQb| z;1(1D@|{}r4$0e8(1>2;DJ0+}CVNRNdkL~R2TXi^oc+!3RW`4E&_+T1kF4DD4D=nL zH;Zu6#>bd5G%KbE6L4h^N}1KRRxtoY9*Jzi>5-whFR8d+pRXwY z{s@g8MP~1h&>GCS0bisjeSd@|^&oo4(h56@m-ku15+s_`8!w3vQsjZB1Sm1^q1;It zy}Zi$5J_s}r4m4*Bn7o5m8JMCpM@G^>~^mwAO)Kpjv!!f>NSSXXeimUNJ2J@qk+R> z+LmZ6!Fjz(@Kd_}*eN`t%a&s(IlXCRIffRKJkRo5t{t$8knAu(!zRKSpbhJ1in(Eg-2dJ4SW)AK!TmAp;d7DwrIZnyH?t6 z8hrN|%FfZLH1}4t`wV4guk>_+d93#Asy$0aJ3$R;2nt=GUanme9oNYG(N19lcdGVH zDvivuPLPCmvUQFeGujDux`wTw5!3y^eVFsAQ6@?Ob`|7Oi-AO9qv06>H&HM)kqz(3s#3Z3SH87*|?``8(z_!7c;5$58zSYN&(}=Z?xz zfH)^f0mM&X#J#84M7Rv_I79UV;;Ar(4l@1Vja%!(dB)*FOC1g)m>L`|l>c)L+z!1? zC{^U(XD-d(E4tI6-DBQaVE&>#le9;6ZKE?mKA@db@=p1o>F9Lu`{>}`qpn|eW24jP zIzUOLg(QoGYAU@?RjU0HRPwMd`)imG5yRk5PZr*v;MIlEPGWql)<;tX5G_~Yna&#b z_lG4Y;ZMovqaJ`#RZ*HYA7aQ%ZiIk-G=Mv3Tg14_fgyte#eQ!s=>;C#*j9 zG-SrF)Px=fJTkP?v5-9Z+U9=+8BZBW4TRFo5Zb4~K%<|rv#*Vr zU_?T|pqXxj;oyRTs%-90SSZb@$qi9&P8}A`1v)WcS)=8EvP4U2Ip_{#H6DARwB*=B z9HZ;u3^Wunlv*2Au>}u(2*noI3QkNDEBG5Vq*g$sih(+_>{;6Nf%$UJ(=q>X^Tw6- z*=tt!jA%hm(O?kTQ#4_Z`@F`WneP`e^K`0A5?Lt5T!&IkT$*h^g6aaH!$ek>?NHT4Z6O+=~NZUuFa?OcX-E z=GJ~j=u(CY-}?dWuxrz^TsgD^S`%o*A#2cFbW=j|$%sT9hc!4$qjjuaWfnvL11zzK zkvK|Sb`XGA>&9^3n~X|irN5^9dgUeN@_G|sE^dWJ{jG%>yLI~L0+HYtaeyi68BaiNJ8jXfa0R$wd$qoXN zOH^jV9RyJr%yZU)zo{L>87g6=UUdgS^8!WZI|wkt+EcfK0Qq|D40jOV-=zJC9Ryu( zQIhlyg5Hfv)gVc9Z!%%2wUOS>D^+p_K_}%APWRxciv||aO;!9u%ZAZ^qt;Uco9L#5 zT3bZsYqXxHSJ^HgfB}}+L6A5~T{Z@QSexx2XdyGPgFqG}sT~9~R6>Ig&`l{cAfuaN zXh23cHPL{KZfYScDewd+xr3lHnllI>*==A1-2}0<OUz|` z2h7E}jRE35U<=M5V6GvxgZQE{K$X}*gtGk(0_h*k6^1J>6@aCQ9RwuT+Q1k9q%aaY zh^v))i*ZFei0x|Dq<>?j?gzZiPy>KpR8+R92>m!*d$$;4^f9ol+O^9urVj8)hUx`; zmZCCsv@TizZ$t6zCaVt+?rbKYHSS5s>H|c4Mi}~t1LI6X1@I|`8UjR(lU1qFT5W_C zK$IpP0N+bAb1pP?8ER<#lQTz7eF_dmPm&STZd?`|1Z6&lveJ=29Ql)1R{#+w@9c!K z60h;#4R-0nPMQey4?ErS%voS2sU!Yjr}W5APe*5h++92U!_KVJ-N8=>znAv=hn?2% zo`+7OYlf2ehn+DcfV@PgBSP zQ>UycAM}fr@3Ezkb&h9DMi?mZqe8}!%bc8z?w=31e>TZKMMuN29Ogs@A2CNqVmyXi zEu;mhq^AWINJ-A4nS3yrbh|c!PAXG7xiFWlO;{n9e=Hi0Wegyd4an#w)BD=+3&t!2 z47~k}FdA==FxF-?8Jo~ByyRO9jmINI1a(eGoEn4SxZ64QT6J4Ml1g4klUQ0pO(`)$ zIuE@}uZbIs6LpGbwH{Zpg_|VHL_tiNtmO#Gxdvjc=ocz6S8vP}?S=(b9HvPEeqamUcD)J>=4s)jF)~!~%Yc&bkZ;`Hf}F75&}3zbe;#dX*&|Fc-8m z78Jex$C<$MbRu_On{x1!vQyWr$)z(( zc`sO2sXEr!#6<}c@2O%h(|eteaB8pX45#Y4&M=6_MxOVO3CxmnIY5k(AYL>)wy4Fi zsKp6U@dp!M5Lh*#bhZ>)4RL?f+EFx3L{BQcwsc-;_W7mR!UGc1_i*{|kngc{%$Co* zcQ}&Jk%Khl&(y1oBfyP@ssMh)P-VcME6V?9lw94T>$B7(;32v)`yY+cI9!q0k49-7 ztytj(JXTTqqfweH;X)ZYMpzlY{N6Jhwcv?2V-kEQ_oYF)L!HdH_00naz8v5n|U8PR7(l+1`ez|&0QGC+*U4rFP4!3g^RzkC9wEd#t) zQ5i3-U7l@(fCnooBcydgP6&8v9U-keD$B->N+lJQibdrv%V7nOP{Yi1 zEh-hjYb+{5fW#r=r9~)22_WuegtT&Jqto`d4trA0_L#HxnX^5BzqP7Z1UOOMk~w2j zl?FaL<9aLtZ?_2a0Dk$IqN(3pXH+JNp*aXyLmM0^)c^NDJSE5D?GBo*h*h z#W2Jln~!T|FcpTuL>LAWVc47u!(bu|gQ+kKCc-e73d3M64#Qv~3_Fuyh(8~^K}Q0< zOHcusw6o~b!gnDA{QsaD;xPQ5u7+!EejEU#D%wgT&k1Y6;g}cSqcfqyg;UR$<6?u2 zK_O)Y_@2Tlnw%!$k??RPgxI(x7>6ZtM;y;qOE81^Ch={eSav_M+2P2R2rySS5UU%A z)kTNcKhhlhLa!6uTimO&(`9hK(!M>dPoAo*c89*B)9L%O()f3{JpagMIUZFidUw-> zZmvnnshmuA#w(9kR%a)}Q17l>vzv3tyZ)n{&`xVjll(Dy#|8EA%JrDi>TP05;77qi z<%9lR$~P~`CtHHtH|2zb3O;lU&XJAousO39o^y}SR`f&N*J7miv+f@k_5a7-nZU_a zRr$Yqq0=D&nh-+3HiT>_f+hh05j2VmAWBztRomU^PEZGBTtgrbFraIv!=Kqn8rhfO_v)^-< z_v-bl&EIhUIIHr;DEX%(i_Cc{D_`>_WD|l2%-5;5Xfi>OFI)|avh@T}yRXxe%#I`o z8MgNwNf0ux(Bo2Tac(-k7>c{}b4M)x`h-GuFTHSS&0@Gb^C)D73%aN{c5<>%T1le_H~ARoSnu<#q0FOF*R}FIo&&03orkvH%DvTuT<{cE(yF z-vGpl`eQ}Wm)*yxqG#x*+^Gb5-K%2Vt76^JmjLX&8$6RW*%Em6+e{ z-_CadP}#WM?RM*oz3s!>l53aS6(YOh1SR3-mpc{iY7->FJib24V+}XRl-%V9mOP8! z)fcN;8LL{EP!+$N@jZc8dOII=XX#F=WSXenP44o`Esrj}&kq^r+i84GiN$?> zT3_J*c_; zoqo1gmSF6U?%^&DCv>w!|`4t>P_PO@1lxM_aOA#8A{(nKl2M_ z>z6;i9u=-0dS@wDueaD!wcbcY-HRt^!f+UOqXk!fE=uNjyQE8l2<_X&I*c`DnK<_(#seI%h0U-0HkS zjzRg`7#5=@W$+gpLy98x3@qaXPh#Os#M--JU!SR;62;Uk!_8Ru4!7yoxmW95K)$dQMj)3I?z^CBmil5U z{sC#c_ja3nxsF}FK9`sN0LymkW|uF@oj3K27T=YNIFJKm!PqP@+=`ch#1-rcDGW9- z+{f$qJm1Et4|tl9Mq}NhvF_;kx*=6S7P~4NmJZeabq$B5$W28Sh>--VLdyUAw!k zHI=S~o7(e!=ymPiOP+r1TKG`g1SUH1Zf?Zngf0?|Z&o$uzt$e17BdI z0U)N^#WY6VV{Cmu>?Xz})BJta=MgyCM{4_lrx>XZc#V^>e!X z!j*ezxQq#;7i-tId`rm1Pa!lyNyXIr^ZdHT-B~QXJIapbb)=wD|;3^{x05R>x z1S9`sY<)oNCdTAs&EL4?u>pLSk@|sGDyf*jFIXF`;Dtj;Solv1HJdrQnXnujQ~a7qnQhRZ?{q zx`Pz1%k&1~PAUq-;7PcApeErU{oJJ@7s}&{Xg*QN-Hz;T?aha>Tzh6{kNX49UWkWl z=kC7Ky(n=m^rN(Y&k6nRrr>P4o}nzWLl(IzAMc^}c;)I%aLKJeeE|YopHuGr(%kY^ zgz;{8{!@7mkm~4KC|BijJ5LfIVM-+8skk8qauHa+t0IppiKwt}rzDYJfDUR!a`VIT zxXXAqGOSTc+>`urXYGgXpL^XuKa+n7PxD{CRk^UXNDg`ut352_=5P5cC6VNiztk-f zcx=q#8yB#U*QSM&(i*UowR-_1zDZOd+7bVa^*8O;F9$$U&i3MMR}#PE^6b| zEn>6K2{&A~EBCyBTYm737rx}$;d@B==QroqA9RJ2la~Ai_r%+t;#2U(6_pGNvZgkP zFrQm-7U(9leF7IP0Wjd#J>NDF`DF#w=~1 zpL3#S-u~LW9RcEB!AQ%y7TqFwirQaeehdJS-c=s0spy~JLh~ArYxgY(~*`Lzas{Qxtr=Cb? zZc}o8B5}us`B1Lbp8P~2J?`SI7vgs9%uggmr#rFCg-)pX#hvxbb~`?suFt57{LoTn zaZi`@(EG2-wU0Xr2VCwCHPx|4TIC)RUL?n@|V`Yc_sDrjMkIPk6|E|i&3FVTdM-506PFF3XwTq2G z4T$=JF&vpowOGjgW8_D1J6GlR>|{fg4O2D<+0B+_Zl8eZS(<}6o_W06?Ig-a|8dGU zx3bP%#TRNPaKdqda_(oG>n;_E3B2%}sJy)e@47b$W6TW~#XT?PUU!8vp+3XM@O8`H zTtp3pLTnZ_Rk=b3z8N3=&r-g<<*Cs`YUqBBvdt>k#%GoiP2X|Kvs-0+cBvaOmF8oD za)k;`I_xk}WC!?R_cwLLth`T$Y=&hv5R}>8UPZag!UF5@Y+4FoY0a{Nad)JOA`FXa zw(*q9BrK;nae?Hi3=2t%xmB59As9jPy5x1l3_8DKP~~XE~TCjxmh*bs-Nx`!R2&M^EM^BUj%nkXM8r4 z?`zMDQsMMK{E>F+mf0aoUMlo1(Q0p!WV)9s+x*Jf zN=N06XXW`^z+zCI7jPb{5>JRFL`An(Q5Ht1Uxg13ghjH`OE%YoJagIov_jbqbS80= z=;>xA$8&%l=JEfnInFk5kS?u%Y`T$W&6ra;DofP{{j3~uKIviU_`kHKyj3UMHTo&X zW7ZTFki?fMOhis-a*_wKNdJ+CHH;@7p~NUGsV{=yVF_q5N9bnWH({fj8Qz4A?h_|p zmgr`lH_=8n(|xLCAna%UGrOZ|q=OB5&tiw0Z0~lzE>zx-lN+}?XO)?n?Ann(;e>9I zmmQt29L}0<(X)%Vg(}<6xTfBqf2EU%kuH8qhWJ=W(!VhC2bG4{0)r7W1~sH_>wXzc0T0HUh1vf+PYgUl72ZL6#dJrz>u zCQ?r|b5$U-N`5^Wi0+wghqXO5G&8X`wD=UYbc|Vyp5M7tyHM;F8U~NDNG(3du@p|Z zDq=tw(a)uIxm0^bX%8>gDv&;-*tF@HyIfiGE7LfQPO!x%e%>=R|US z$lW%Xy&k>$viwEd32Ojf`C>@@z_X3i0G^|yqDpApZjAlFA5CDS^{tnfk+Ar2?U{Ci zs1C%8Gg{AR{j(XZ15sZvHnVVqS*QYEW4cCwL@c*mfmn9xtLDe+OtuRAzL7?NWT23x zg~N$8=wKZp#%DiwK_|xsxtc6?Tlp49zHxF7B6h>`Dz%$mTo%0Jy%0~=&fSXJ2y{bU zr`>y4cX6KSrGqT&8Qq5Hy+vj3NZE&Ymv-(d=K+jEyIcF@sF8o+GyjQNC>!;2vgqL< znTvd;ieD~pKg^In*C(%n7W5|hXS_SCj$Z{+BR@nHewOn+S*m*`Pj$~!sqUFN)jgA? zx@VG9_cTlOY-*7-HVK?=vi#V|vmBa7== z8idXe!PpAF@F?J3WW9*7rfXMXF&hJAb{^ex(--+HnB{MHLyUsRSR38ec!mCc== z%ZiHmOz(dym%DIqQB0WhbW*ea3Z_s3onmd`5#5ySgv3CkXeTZkJEfKxWokMM13G0r z!9R%9b!l1oOV>Juz1^yKX>@tllZh;lrK9Cd4L_I*O$w84CPNc8x|t45*yv_fG-0EgSx#? zGBGXRr!3p_Q~o4+ETgeqiE_3FqAKSLcS@?7=;Z1#_A>fH7^&}gYe4Eh@dVjz%6+>@ zMc4A2Ls>6&Wa%KQtQSp}4vu@QVTobDk8?_kD*9Q6+@Oc5QiXsnpW;V#8H5{L`^d#I z5YOBS*@%2gTvg**&92Z!!-x{E3O-)B*P2ux@KPluua=P`sp+d_xKLTTB!+~QNiT`F zsw>lV2or(asptLk|BegA2Jo6wAoT-pFj52f86_17r}fI0nk?{4B^8X}B*c$S^P>)Y zeo3&N(Q255IEq$z@lGC9DdD8H)B zFLy1c`PFWb)nV298vSgRRr*g;J~=kHr#U93p3c*ismGYcy9;yltWyqo>LY(BWM#cy~Cwa@VF9-f;L>;|}nvC`HR6Da#%Z zpHmn8C#i-}{oExV;kh(lqU3I^kMY@1UZy=WeC3SyKzxOE?(RF?NqQ#qwc5YugnmiU zcsE^C=U@bD4VMLTGH>+PkHbdNR4$@mTjoyKOP8$jZ5A+D$hykpEa>;d9McWim~yU z!j<8jYX0-)Mj!A8O3DwcH1On!_~!opr|@#pzkouH2o5Vn;8a*RlWJ+^Jt;^Zo5L7O z$L28B3CHFz=EL$od@6!%Eo}G4OJU?kmV1I~O|ab|aH;eS^s`%NZR{h`8LRZK3oLn$1%&ZPQ)C~B{qHupig zO;cZ+LXJU#Vn=u(BIp2%GB!XsQ-wm9{{4TxP+kP)kl9Q zhWAf3T~*-6j5Gp7p;KSYXnFO~bK=ia9oL$!QKXifO3b5AROl5&e-`=zRqzQ@vH@9Hq&-&F;_HbphyJe8J#+B)`0ohY~KXHIOU z?5C=%yK?$Xl`i+o08r2^r_g4?_n0TqJ-67|1`2ogQHP3A2&>%R1C~O(JQd>Q(U?9* z?VO;W)8xUP)0cMjW)SM;NG){aV(q%hScZVl_G#TVMvm_<6Pqb}vnuP7-?cqMULZ;P z9cE-0h_P<>$^sdd-VDlh%DijdBYzPi0W~v-j0_qU{b4MspvR>6#nnzOrdtvoEpWc z&v_E?-;C7wcZK&DX>^ZQ6a(Tj7afVNQf{xP2K<3`Nu6yUYtyk^q@Ou4EMk;huabVx zgo5tGo{8{3oBz>C0T!<%s^iZ_@l@d^9{DOOj3)8z)^AkN@AcC*M+m!Wa^;VH!aU5^ zt_}Jrc{s8w-&yZNNYm(hNbI3ePpJJxs^b)NRff+~c(IY{KrH41hH?rbIf|QA$z`Uj z4~R0!SEwqH!Y4;w4VnUtQnrFH@Wk$Llq&KOuLKgN9Li(eD`VYJO`+Q>dX!F5Tg7gH z2V^|GhdfsUtoxLQqaF`&^k(*$CE7z`mPb)2v9f476842Q-VNY0US+!l@C8aLwiQ~l zUt^5GB}yt7X??^qhS|f9e=)~l1 z>{)>;0#+gplmE!8i{hma!~r15=#qB?ySJdz9c`#)-I1m~;z_N91l}ea$=%j(PLl1M2 z(EE%gdUJY0Z%%KfmmzJoEQI#uTLD61*%cCy!YWqa&=+BB=7t}FWr-nJmKcI%(GU;{ zy5#JEtBb7&R>w}Q4xK{kR-MiMIChEZ*{YwuC)9DR%6`%oiN3EYJW=OIK5fxN9VF-S z*HrR*UKy~YZ0lG;S$rvehAJUCIa!5aB@SB~REoId9wpE#s>X_PR@@79{AIM_1fy)DN=lLFJwhnxPF(LG ze2#e%We^sxr2+haNkv7CLRe)0fu#^HPlXt7F1MI+;jlGI4o^RCRn7@V- zY8`mEC;hj=pL-JU0VSoLT0r?LOson-qg#Bd87-70zPv)$x71;|W(6YUbk(Aj_Cz(_ zr=PpXi}vMJ6*TKg?sjDPOPBH4P+q7#GqlJ3u|p5UmD;(x@04FG9G?k&wf65hp%2^YQZP%ig4w2QT~%tYt= zmC5}%Lb)IMXD0WE?tf9XIo|O#GfO_vf9K@t=%z&6pN$lMkW}_9oDAR4D!JXNr2(V@ z_pD5RWa(*@rH_AHTV>8(yFS>qUq%~X0HWEA-Id#;J-_hd)%a=TR@f5ip}a|{gk5#@RYShmzyp1 zjMl4^DvZFj6Bq*@Dp>fiS*QZhS?pZ2P?yLsy7n<$H6T*yxqYt+9HXu_nJsibP1#Bd z?pd;@*(MOiI8D{#jMOo3a=#EHBN8)^bhMltbe(C*siT9DN(@@+u<<6eV*QbRjww3O!t>fS!x@k_YUPs?(=j~ zUg7Erow{n`5#3Xj9sMUyUd41X`mOErvo+9O{ghP*NJ%C(Xu7bK+UMzGDkRi|6YBZu z#LLW?!B9(LpQiv|O|J!T;d;|K7dnL?1=M6T>8663u+dEoHDRNhQfk6RH>DI-B6Lc~ z?ei0sG&(4rCU)p1#jVxWwK|NXM%G6lj!(81>0zf;M#U`tqw#=?zhBmE89-o3ZhIv66SdQIdXeK@D(*US5zXC{Jj0Ru z@;`$b?}fOhcJAgnH*Xo!Zpd@Bd#`f0=x`U+v+3+rroGEd`A=WxY;-SFw*6CVqEr5Q zVOb`6yJ4JlMiI{eDS`aG?Ir^2eOi16hCfN;e~EjdAOh6sfs`OZxwdRGoO-uuNq@-JmsP;)jZ_8x!bq!t z3{HDxa|QeyTUu5DaoDke>&8|E5|6yrteK~*i1HlJh{Kav(&Msy`3l#z@ zbD-tXLV##?ph6H~CbZxttjvLy-9&(y(1I;knFB4`LVzOLfs`OZk+p0yEJaFynIhE+ zWTwa=1;|VhE+B*RC!aKU%1rr7O`5}W(hTUQ`)-@6IY^g*)c4(?oE$s0Z43o-VmL;j zWD7Nkb4=Gl}J+1K2pteZ6S-WGOR7nG6as$dYB^@M0xr&>6}{VXC+uzoFvi-M+j!t0Qs#@_fvXl zp;EHpXU$kNnVh9Yd% z`YF$E0M9maBeA&=%rOdyw|Tr>b$&-byDDeOgCGks+8F4mY^W?>w4t(KnfvEV;lSSB zdeTFZHett-p<6LWI3^!d?|gPwV;(c|MJIZXv+S;pvm1TI`&rffHT{(A0x>6x*4B*} zBtGE-;zQ1d7gS+R3kE4-jx3M zMWH{B{IACPs|o#gru5$#UA_IIUgdmNKPB0~=UB;Cfv+@DACL`QzBk|APX4;|3PU!0 z;Gs$qzWdv09Hqq0%9`8u%4j{qO9PKn(k_)XcaFDbG^q!%LjiJV&ix5Ti)>PFU6q*+ zKfAq>ky7+QrvxZ5=%L;}M59zkB|{_kw1H$v3hJiRDBTo|($SiC{BExtnxO@s9gQU5 zucL*&Zq-y0n|Kl541no_%A8*{%bPdl8}am>4OmyK5OIN3SIS}g#!!7u42vr#HdFVm3P|4Mn++gIbD0^;Xg$p`vkFMk3PxHumncs{!k@Ov z5D+P+YbaXg&eB-meiVX#HOU~Rw`goqJn>+{^ajrqR|-rQc&4}v*Jvlx>91&J-K?(N zs-H9Dr0yOPS}j*Ey%4{tox2s6Cv-!;UAy-xcXM}{&O4Q9?=q9WyG-{z%C>)st$cSG z#;=~uZQJ~ktMCoI6K@j3x>~2=;=)FLyIkUJc5fN>m#7%qtGmk{iyofNf zTZj%;W_63v!OqNX!4j;@>J}}*&YW&Jj|ecMcOZ|5Jjz9YV*5)^z9;IAcZ{7V2dEUr zQEtgC)>ZaM{S*?Ae8?NEkK{YanemNUN~ z3v&@04L7bAYf4!4WS1glmI@&O30IE84{anujON5p|2eAvBK?$+22!3Pw{0Uv&#>X| z1D1?#Gout&E*u)gf)A?3fDwt@iUgMQUo@5o>Y}l!^$*AThZFkmHUCRVyEnQ@+!hT= zHP&BE=--s`KYnm!KX)^S3*CiYn5{G}>MI?I+HFQQBmC!@I*GB_VMi@mg#$`ts7c)Ig5{GMjZ6|CPNZ&RKCJ8=8j~AJ@UBK%mVx&)+VTvHSE*s!uKeH7 zPq|e9JnDjCT6fByfnO6XS_s~0f^}d?aBFkHyG^hPED3IFF8E^;tN}lGUg4^|rX6^- zlH5(ET1E>CiTAk^5ISX_0)A{G>idBocpb@cH_mANhf-zyfmkeVoP~w6L^Y#U1cB%* z1Zkmmhb@qoN0TDF?bqDyA~!U)k7YDh+bA@EM6aA-_`@3hD*c>Mkw2)DuPRo#yFn07 z(azm{r`(tsp9}p}+P|mo-~6*{2=u?u&ibLgU+TNqGSDdd)ui~v>qx5}S5Rc&qgOIAJwvD2#c zi(0MEu)U<8RfITY>$^oKIWt#C;heln72Ttsa^3l~5mT40Yq(Qo?r7_EJb>YnAHwZBqD9JoYDdI3$-Pr~Q|nl#}d7sTX>0fQbE1@y2i zYEk;@s3hld0S&~8l-Tw-{a?@=|EGSsD=hGLjJ7NT-)*$z8JPIo6;>ES>Zri0`PIDDR0deInG!FR=kL}<|cySaNsM7nw<7wTW*!v z2gDT_qHXcJ&Bs;XO8s<01m146Wf}O>Mq8eN@hvqRE3~kWx0-JNPf(J(3Ts5qD#U{I zlwkamTCrd?B^bX8qEG!f&$|lT_z%T8BG0e_ws#i>C=iPUV>6>~HVaiCfyhmNAZDDdsu{&X;@bgYLZ@sPz>Ce*eqdRBEulU- zXbh;=XPfsUK+MP=Mno@(YfvkMQRuC6!@Wh2NH=Kk(D`mpy6 zPYPqc4F#SB$FKLKP^w+K7<(B(Ad^Gx5nsN6;C|&*5huare4+j$mH(w)j-))tmW-&7 zU#8q=dnv5VZ7$WPT_<12TGRlJ8)*P|g_5Maww>FxtT%(O(l(Mc`HO8I6OCxMnP)fqv)P zV6o|=J4PkZHPOVtyNPkusH@e2KvB?0&>C3RHRv}Jc|<6-@D zCn)gVHeve0fpu)*D)DkvaglzCNowdSPXc~eN#ey=M&mP{FVxPEKg*nI0H3WSR}Jw| zEEcS%1mmMvELcqm#z(POu$B^xUvn1=4yFX-itp15ywN8I_(dc21HU{WU$lDO#ERJu zoTH>7v9w;~8N)%sk5kNO9r$lX8USM1sjp|W_B8baKr9!Gw7ehDeHi>W-;7p)nURKo zHyLRJh%?S&)m+=lTpI!6Ou>V%hmo%V>E& zqL)T~P>o$@M*D!@ej}AI44h^b`(oYE`;@=0I)9~~ZUF}G`i4Szdu4eBo@cZ}Gw_i{ zJ1he?jP~RVyw+%kXW&bWc0>l=X0#(S@V^@EDH-@XMms74-)FR^X5il$?dWJFB>w-g z_?Kp2VwTkrNXT+W1xUog1tehM0urxWxBv-PxPU|}TtI>qE+DZA7m!ee3;3W$Eq(z> z#0qPsk7nZGQbxc%UMFyok*dI_8L1Dr(nwX{dLz|SC?8jzSJBS6BJjD$hcHS1zZerX0?qBPM2JV;3$mDAdy8Jya1(S_h1^3n-; z+u*KC+j{e7=`uQwRn0IggCX?zEMsp5{x8DklZFDOxR4*ZJe+nCY# zp63Jp$n%|>(Rj%70W0dQ*xiuPm}MbUf%|#B*Jd;h_I$v@Jl{DPjc0m3;0d1Z?2N{s z=L4SX`POGNUhVmSXL`P~G8z|nKH!@?-QE`q^38wXH3qb+t`^8t{50iHWxCxtA2hcD8=X-B;jimE>kL_*2GKTcI2@6FAV%*nUBUz zp12{SafcTM-dX0OahE5)Hly*77Y6>W%tzx7o_J11V-K4*b>J*p8qPi^qp`Oqo}JNH z;)Q{aFZ0nj*b~=hG>-AYz!haa8pnF#Ss9JG7Y4qt%tvF`6VJ?Oyuu3uPxE~1G8$(p z(QQ&-rl2d~t=iA@lk+nR__K9MDW*<1!Di}wd4f%)DsUGY%+;)zOk{ai2kKd962DV$ ztmT*NIV!l0k8DljdnskCLZ7MiXh z;FFXj6TB^>ahT@=KDEq8^2K^8>nM=KcT~=7%VZe<|zy~Q!u3%48*zZZe7bz*f zYfNK}=erbWLonK=a2NgTsGQqoXBx7Qc)}rcY^Gc~E?*?CR@&0Jur0$~e3jlY-y7EK z7IllIRf?U8@E%0>58tm+AJk8I(QaEtc2phmLwy;O{SJ{{D zHmT^_zW1m!UdYu5)6g$F!{M4JT%m_3V%3#xHdeGcRvKNlw302 zuUUG*rCc`-A=hWuO3^Prd_<-0(oeV0fICgos%Z-6deSn5TkW!O73<%t^^#e>VFaG5 zq>jqQv5dwWjIR#-WSNh~^_~yt2Yyu;r3-wk5z~zt@jES~0U#mGkiT=i^xWd_ zT7!!*CpB>1P;+w16k@Lwu2T&XfmPe9oa#0TV>S`AjWP`LZwo@6$4o z%s;HZ%r%;0Y;DK|^QS8XG#r^E%~RXTG!bg)A2|0?|VuC9t*jTZd4>_)z+-cPr&9Kk8fCSD{O zD;oH2q-X#|Yz?xU0g2Lojj}xdre#H2FJ0^kA_c^wgj6_t?^k!2L%YdSe#Q4Shq|x( zFoMRQyP38JRO>9++)CQEbS_(z(f5$@%#1u*w1{kLByxAj8gc2Z)wT%&u?1$NSE0N<=6cSRaM?Gg*tQ-blO zY%Ew!3C2&m#DcYyVEnYpRt@B4A0zP4x3g>y0FSx6Ajt!Y*NxGrDzV4~t&b^HY8`k> zS(?T@o-Z6ze7oIzs{@~IzKsBJr?^Q#>zSs01c-+PBQ5VobTt$EQN@o#RDFBptO^*@ zlBDg)BvrG}s1pzWi>=nMI>VwL1@2*_29StcoXrOwo$@UV@QIeQejpAzU4xPPi9hh# zrelUqxnBe%Pk(}(VfEjwVg`U%m#4jt3sW76@#2?IRvvIYPJWR*ezJm;Ki`i6@JC9L zZ)7?%8kC4^?qrtH5>hkwzN%3zku&j;i{&})h25oh^PEvBH4g56WWNl4LVsLE9`QbM zv3t&Y3dzs|62a0iOIFwMZu!5w)Vwj?E&m(urjNCv%rh|FE&m(urthtqlv$P0@ox8X zu=4ubG3BA}-;`$$$1~pT8R+=3GR!P7jQ2qOS~GDvzTJ$04hAdbBz^a(l3DVi#4r&0 zcGu~Y(nH^ZNqFcxQhD-(_@s^Z&_PwnuYZ+%8SkOQ{@#rE(PL!jdxi4!$j|e1%L_op zJK>;bQ&SFl&Q}ii^FuOnD1dySb=S+?kl$tW80ff38D=>RelT(+^s&XI2A)2qeQLHen}y70Ik^Mb1m$20 zq#9b@=E6xXD^WP7Wiw|Aqdq?hP+<(gquBI6t&1G`yDAU-{>rvW1rsC;dYB}_XZ9ub zCRjeNleb%*yeOX4no3zSqXVkxrKa(W_Ch0+joKr}Od#fGV}AJ}`6J|oDj2$giN9Ax z)8&-BrF~a|bAAZu zjcjO=6olVtN!dVBAidA@)`94qE*DT!qykonb3?jBaDi>ByK;^?3uUW$FbMQ&VQw}S zAF;^J!BZ5D-su(N4i!^tUES2H8BXJoXvqbkJ z@tZS5!vBy3|J%~jI%IF={~4NI&OY6h^*a5bc%P#Cu?VkMCDSYGndFdOYa?0*qOimr zjwdXfG7|ZK;@dxw7dY(Tlf;u!?u!9GPrMA7y zQf6jGftZ;SnrX5HT%#(!Y^tJF2Nn)|@-;e;Ov#_jUpLkjjt_!UotHwn@ci0r9PU@2 z_k0Jpa2mh`N^-}-MnoZ6LY%9X0QveI0s0x>St0K~Z51p%IJ`4|O~D(6}?qlLqX zXEfr()l%Np%-a*q+Zu2e^P?Y#*UpbxMhnMfg$Dk;mj&W=AxmqUX9N;f!8r9e45lI* zw8%)l+{OapxLgwfaa_27I4)ek*P6G3Kmu{z4km^sx@XAHOw)|-QaQb2`QnV`3?+A0 z&RXBK(DgteO!<@i6-5tol(>9P7?v8DI+>&BJjx1U6o?tQQ8u_iyRR^|mB3q-l;7o~ z@n6Q*2fW)zRp2bmZ!SuMSmaw;AVVaTPyVT%T$Bc-B%;kiC+}Ty>&(6Bi8fiECUhBz z1<^3L4jG6AR%oQjIv|iY8J@weTqKMGHvq(ryO9}M5Xq^Ob8j=W z0396A%#UJmAGvzjC4eOKg(k1i?v6`iFYgaja~`|#=YmE ztsf~j8IiL;-1wUKv$&rMrr^LFwy{Uk&;{mx@k`PAY+YQ5Q7^(GdHThvj6c{ZV29Pf;Hgp zjnoGuj$*r`MJSSeVA*K=tqn$hZ59TBSd#r2h#A@2fS3_3;DMH$5#TZmjfPE@)UZt5|gzU+B{wkaw~7? z6Cc-BnX^|mQ-dOD`5A*rM}WDu=^7=WB~K&apvu)xs(2s8^tdG`URjz+T$kmeA4p!L zoQpFrmFnxelBjWrFRX_jdk0G2czS8^wZv8L5%I>I!q29`9P8fuD1 zVrOaUs*K5*XMwFh#08x;UQQE!Ils%R>NEn}mbKy3uDoaGIe(_s^EvwIoO0)K+V50) z;@*Eet@`kk*&$_AVRA4J94-7WR8_25p*CF{x^00%rFDdQ&(aKj#A=}rcmCZ1>thZ3 zrYD7p-lqz_rJpjH`=1$`72LDEB{vrbkRqQV%sAux*dqQu0Pw2Dn zX~FYK!t+Y#xvktdKGRm75g@TBZ;wsJm)dGM0(^y%j2FT>&kH;^ki+x&+rcaut)xeUb=t3=A=;egAlQeNbhC1ScP8%=nLJb0L%UapwLhX-t{wke0 z&$Gi+A63l8oA@q)+0HsD6b;;_sbs8hq5n8g2(42LI zHjmL2X}zsreT4L4TfO^O{9ofqp`y!G!B_QDCVqTT2NP?Y%t2N*7BwNUOc9-sSZn;~ zMCUqm<3Qrl?|$`d58E62fOwRzSNL>=4;P;pf}9Vp80efBramWz*?No~zK*a%dUTNp z3(A`gVaS-c5Q|x&tFD7m+s{sp8XJ9TOH#9Gj7HrCB}xTy43ecH^v^oNk+F_&bgUyB zA#8HlCRqjIMsXR14_vkhBai|v81abBbf83JIg=5>lw> zAA*sWFFTRKxx^pqgi>xn3&m24TtcoVAr~F#k-J^Jb$|GX+%{dt9qkK>@3N&mx?{Sd zs=jQhs=)6kDSwR+4R11fRpc%ISX|5d!D_%gNsdD)m(NfvmE%y(iaZpyj<6=s@H_R} zWftLkGAKB)a5-X5ywALkH{gVv zU4oY6qS{1m=9R@LyUbvZleD9qrMA(Pm{vQg>P@Orp11?PLrHmU((opu+FYaAz04Bd z0De$OZUt&&wD!M>dsPi!--jTr1`aBzkfn9IXAEmD_M@Kiqi%k*nICl^#$^=+V!2pF zX?Z`Q>WTfRru?XyA3f$r6^L>9G8u?v=SMZ8<^71hOpg7irTnOwAG?_!HK11?otI+u zgDLfcrhb8`9|WS_#T4HMi!BVLEDV{2UChD|uxuf|4>lf@u1xf9s&sEv=4jwvrRn__ z`7#*_qvY?*gd^m8!op2>QAv1F3B4%3n8AaaE%*^2PU$k;V^wpy&7u+DdLjE+k~8AgF&ghvV(NCp4cbT2 z@^d$XNPxMvk7|^JmUB)W*3O=ou zXcTK2K-1?;(*UrfDY{1yYg&z_drZ@6U`bPSk0P92a)jMUnz$E%I^u#(8!zo11q9q~ zevjghcYcXh=|TGGoN`|#(|)ea)#$~i@wDp0Q)Y)d5-~ZL2aXp07pf}OtWcYsBSV$e z5$ZiqXX~hb%Iy|pUt@vwtyg%iCxwdoRl$Y&DU&(A?*J3g%T!}UAb257r8;p(VVhe0 z4<7~)TlwmOwQW9-mR6yER{pP>Ty(zvk(!=huO;Q7qw0`@~#u$1Z z`%zE%Q8zzsHb3gXr+x(W13)Z0Kk6B+zH5yU_zNW!jI>sm!vnx;l~gd&y2CR93ASLI zI^Jq3-m1m>b&IzOB-v7gKq7QTs}?V=B&s#edoATh&HT8<{HOuF`e=!W)eolB51RU~ zn)*Q?>Rn9nAv(4&l(H~n7S@}EAz;};e2DH+0o2MiFrV3QzLt%nQ8W~QqusA{@eBQQ zw6GKs`sXI}&rRr`o6sM9nVitylhEIj(BG5LA6?xi^mn8GV$Gez-3@*R`4n35#3~f3 z*k4Le&*a(ngx>m|&}ZM%Lgp(8&nuzlwsPb6)mO8ii~xy6d3$U!c74ovfisk3yb$*G zyuinXybxA+Uf{DsUI>i482eLw(CBXA3`WLGMZ~6WwSM9Igtj*OPQdL#O5YQ z>MzUMXpr8NHu^|?t}RS^PMkXBJ9`PfKRQG7GpI{!tonhBnJft_LSJJ|tI#xTnpOcz znogct(*T;5o2CI^NmF#87W=dsO)oS}tAQm=(S=&LP|AhcPSV5$8S01&I&HkP3pEJ1 zEo)_yFO&ax=a1J~ztKB}0FH}PdMvz>KR%B8{tWlA|3GYXZC7S0Pt3;zrC>yp)N z>oi8JBeY~4p*iaaZQh~vGS{|?K0<1@)w_?ye-BRz6&?;qkO%>r=xwi_{0$8e0aq`=fp7eIWf%E zb$a;vh8@zQi$qvZ-gF2<#>9nK%o1I7-K;s8kcgceH8$?|HL_`p#)CE}Q7VwbH%mq6 zpLK*IV;$k>SVuTQ*yOTJvI@kF;xY;!xNH+fAO&17;u9BW1(g=VEaE_ly;#;p!$}MQ2SZa|=$n_-Tq9Z+W8`N9(ytO9}Ct~g7LegW5H@lF#f}K1ZSD6HDKA*_>a@6?>P!2M&K*k z3!3~M0`Lt=D#E7q4bO0Q{noZVPh6kTSmTKsG8!LNqFl&)N#PGYi2-1|WkiAH zG_^GJ3pL6+EZPwub(bH4X<)nC)s~F^0xN*mS>i{46v8yeEH9SMAmFCg0>zIN6{wC? zDu+d`4kTQ?c5Ie9a+EnV0L1%tdCMVqz_YDXxXF`ZzoH!rzkZ}6i_f;V1-)V$3m_Hak#UxUj7K5)+wG5h_?kJEi&f@e{i8_fItEgf0m0k=#c9` zAkwO?C!sF-+tXOxa1(WN6Y8R$1dCUO{>yaCpJG{wu5GtzhFRNpY?)VK`=H`~r4+fo z2L8rKjlI>arM76)o~>}yNd2!+c!7~>zzdBOy*3^XSpTH7fvkSALjzgKpihOdy6(c;)!h;jgNX_dsIaK zshwZfPq)y5J8Y&mx)idA$-){&FqtMV7F}In?4FVX;C;-2=!f+V2^?~MW2J6lTpgOP zAHel?BUsO;EvFWuTIrSuj{jnB@LNif?RhMt@k7rS zdi4(N%8aE3e6Nujz>g~F36+!9kIkC$XLIEW;iTTEZbFk?lCQOuMOkRq(bow*H2y@P zw{Ct>WdD2(;*I(#Kd3sE(b#N3*MJ0_^M(1lU->(CDSD*3fx9WmEfw+onpm)&5{&QH z#Ddk7V0^zO7ObTNQBqn}Ad|bK4mwG!FpWiq z=p^XbZhRvydKu)TJvvEJT2+{^4H_Y9l{@i;Q!4cvBFGWtRF#DOqAo)V>1xFWwNfe@ z)cHyXRr;Q(LY9Rf6i%!SIKzS{cc_3k} zbEX~nnBs?K+}~MDmAtNeSC?DJnoI_!ORf{$Zzs~mi)liap;{1)m34#`FAJuLlsj?Y zw2(rni&cv3C$3jw#Y3^;=;mcC)rh5{Ya`rQWP?^`yx;0cz{`v@z-a%v%BF z1~&R@7(5^$?Ulna@b_#c4F6c+er9hNh{GL~b9$^=w|hR|4MwT~KdYpU%G!+?jqiFs z;Jro~2L8c|wr4b`rnbs3q5E2hZV<cmvZzM>fj3XYBzb+xlGR$CoX!t`_<3TouE538gz%VW<-sg<5K`+bmakI zAhzZCnm7d`aSEb%uhZCWwn69z-lC+A%IRYn4RRoN*RhRZA%(FDDU9#as+FL`^H|XU zp8JuCzFcCcxuz&~(eSV!>k>x;E+FzV9t{i?(%~t&poSrYmi=anb&hPM(;J37b7;)> zG&A^ED|owZC`Z2o*A1cV#FigMS?uIQ$OVra1b{?XFoqTiYO}})!(G&2f@?t8H-Lfn zTAJu66H=%uaZJHcTuFTS$~ZHq^5rX1-quiTD>}Kvis~qO!17f8l|rw(9_x-y7wCSn zDQckmu{P`tD(20eL~;Dulc)plK_m7cI$_~~Pt!0SkX(1<0e{IH9sY*Gz08GSAP(n8 z9vWA9KHv>TssTT%q~ge_?&pWILxCQW8SNI`jm}i z4M^&QD@>P=Vsld-c4YjoR5MSp93)C28vivK__!sY0sOs@27sNr7USbK??y&zSEY*U zz{gBr3luM%qw(#1jK0x~R)Oy|(g+YUF4C$+I@64f z0Lw<>uLdwmPUQdy>@Z!cfMuidZILqU=NW-zqw&`YV;TsNY1R_@vz_R){ zrciyC{GR7pxvzi?0_>7scXo8z?^fH6zESA3?>3)mKoXKa4(E6w;Iw}n4hp`boi-^m zRS*c$s|w5X_3GHe=3u`;9g}wXb-J)lJ6xUO)^r=ZXzHP!pxyVDV8rEg1;k&+7JB*v z^$_v2s_1Ff`Rd#W=41o-JS7z~oz`u}*bn^P1V&o-c}CzbCoqN)#IDp+uGBMJX_$pN z@D3x50`aG~+MxBIX9SkLi`S0WyK2h2s(H8CEL4H#7-;~AKgA+L>weD&EPEGE$k@AD z%Db9*_cF6k17blgk&c^sYz9;M2NU`SL;Wt)_+0`=t3#-DCv_z;-mMBY2|bJ{w%BAu zrxt8ds8b6znQWopux5fnlI0voA<51Qq>3J8O<|O2E(Z)?*+L_u#WWX2AT{M&iNB}E z02CXl9M&PXz7-{{)GXn3Qz!>Hy_O6b1hnucZApSV->KUzP8)F@K zi;+fwxLC;2y3aEL%MQmoPV8_MhnJhK0U$0qqg8YGBx9@sUv8uUATAcNw0`Otfn|r| z`4&4|!{M@bHS_L;W}ybeg52P_X>~Y+bCiFPe##HK-?BQRg)P}RfM_ciLv{P_s5&2o zv)fD^#^s<7^n&iMfTk8~eW0h-)q0grt*iC<9O^njb4?Y>QV*mGWnBkSf{(HwG0I9J z8xyc>p^?%0iFzrFKwm)`QBo&Ob2)U)QuchLxS-huM zyfxt4jMNV#Xy-@G2L5?A@HOBiM(PKWheDPXsgOk)NFEACS_CbOKp%hf#Cts6gQ@rj zE&huw*g;@9*g*@Hxh)41Ah9@y2Qykfvh`pP*j^fM(WtE%Hr|84{U?v1?H=$9aFWi&yMNHTEOgOJhI&6x&JYn=7e$hI{M*1(gL3nzAzBS7V1x3P<>C;n#!d5 zXfq;{9DZbPPn>wz;5dH#F((m%5ZN#_IKgpKW<8MOraLEUl5o{+1hvHZY{yTY-nFLr z;bK6R2VmKcYQm45bzR`Si?(nQ$F*3#kZ>C0sw=s=Aj##fyB{3nS5}v}6jsKiu#&34 zup5|0M$5Y&?Swc%Cf(m|geDygo2W|tjR(5kZd0iW{GyTifcGm&9?R>@Xgp*@c>xZM z-y2b0fJ5U6dX*^R>oXcPPu!5v7`N-mzN-~}+mpgzrMkF4I^E9lRUjF4`K>1M8+tynM43J-MW6wtzB_i<5P#8b)BU}PvR|gWV-Y~XSFT?Oi zZr8=G48#VbM|3cFqAGN2D%dOU3za)I0wtW(a*TSf>MyXa?xW+r-mYDTf%sLt>L!d* z{z`x#ncPAGQit|J90(*tc*F7K7Sg!6QU{h@>51mQJl}~c>ugHYfw<(t=*ei|kG*0C zf_E@_9&@qg3UB10(KstZxPm3&0%Ajr&P`0v=mFDsh3Y>`C+3@Mm5829d4syj>}21y;<;HP-9u|Vc^o`WJ6Utm z>ra%b9=$z0Pd_Cuzyp<3Y`@_s$Aa~gVEk2QELcqm#$S-cg0+-j{GHp$8pzo`M&Rov zjS)dUW$gf-r-J*+-`LG6{@(uSb7mLdkjJ%E=IoWhf{p!To2@tD4YP|G{5)9xxPRHA zWzB3Y>s_>9VOwE+`JxOdJb02jKrP$>1qkB$Qd7`0gusO`pi4DhTl^RRFPn4>wrcOU zyddxoEeXEQ1na=EU|gVB{D28ofn~wCKyOlkyG*bF{9hyW0UuP-ee(C4>)X5MpCqYW z);smfKzi4=7X!TmB89-gd0non6sj=*UJjGnC1947nY+N)(!+Tnv+S6((kJSwSKVk{50==`tK%90N zt7f!tQm%o3xL7dK+Rais4D>NXwH0T(mWrWfF?`Hor~z?YE?R(i?Odx_3@@-@7y{yA z!AQ$HT$?)SgDHmx&0$=U`-8x;cZ25L6Rjc!frCntnLn1%Akog{!Hm{-yfpB8Woa6n zIy{9mEgx`n6Z1+s8d{&cpi+*;=U%9QR^k%fN z{CJ_Cbao4`VC%JVp@Mfnw3&DR>b*PQH1Uph$-4tiT3>15Oqd};i62zyK0r>CgRRIS zL-k8!t(rxIIg*R6Z7H9KFp+k|V`P|dJK!-Q%*!3{7!fA#j(Cg=6J-ZHo~nB0%Wz5c z=#^z&=fRaIbw%|b+wuTUq7-FR-b;5}X=^ht2!?899{#!PHX8=w|G&OIU!{vtN9F9X zjK-&wD5q&4@#cKtVC2LwQ~T)gZK-}vm*-cDvwH?Wh;U4=&uL?#Ul`-=ok+y_%?5XcB)0 z9x9iMICOhOBPjZqMI3!y^-(qd9rLIkc)yWW0UuJ5`(AC?qIH!8;lO@8P&d!LvS{g| z?xjnXFWOvL(0y|6md;`ZO38H1i^h$dv%SmO24vg*lOK(#+{BR*_wdENkV9tb;&hi04Z8^Ox*E-N!O~Hg{)cG1Pt?k! z)ZIh^f6}V94|t=J{pW9qZ!L+`0=j3 zOZypbrw+`(i`55}ja6J05g-Xk+<2iZNI^Lr-iLn10=x1|38o*hi2B0;`Am>6`SO}H zdEF!Eev2s@K{o*;zV*iONmd$$`cvwLEbl-@QeLkF#IwZWi-$yC+A|^HacR%Qd9q*X z3KR9#V2H)jQ1NcE>m8$8qTE3Zl?UCS0cr?0lf}{3OzQJCGAqBP&?|~onj2%5(A&iN zgcP=BR$gldq{Y!Jh8s6#R?aSQhvp{shEaY={r!r5y7w%+RxVL#V?&BCRGg^GU#J#X z6Ki_R65X>}s<%*HG`H|pmPO$CN-CB`T0irQVSMNsG?qFLX{W26(ek?LQ|a2JB45-k z%0hiz?xz{*iz1E6K9$yKxmKz2P;YmkS9(HUO{AG_(wBd$O#?0NHgZFL^nx}S$eEjI z#JSmpw~LEY#=^Gc?wDe(V`^`8Jz%37%X4Bjacx25S=wmm>X4Sv8}!qy?%;PAZCM7s z%4o|o@byMJGy{LhXoqFsZyW8&8TiLWJ3Isb#%M=m;I4fN&)X|UX5bk{drEe<;{Vl8 z*~owo8>t5Dv_c;S?q;MKaBm|G1Lqm323%sKVc?UDR0AGiq+#H(Mydf%Fw*d_!k6f$ z8yN6;#?}WsUrBNjZMRamtRxH0jCN!OzRGCJGVsTZwmbuW#%PCT;D0yTVHx;VqdhqT zf7fV-XW+Yyc0>kdV!3$3VkIX8n|M9l(tQh|4f-j|BJgec>lt$Ow|$JpHA)md;CTxR z6**t%0{Sm7{dM3a`xVS`VgO>LSUqWd(rng&pYy(5K1SnKB^J`OX3aPEE+3Da+ywE{!(4H zDG0aCggMA*x=>;{wH=|0IUomlAcfj=iib9ZF-%Tkj>Z;kxqPi(E`iK4d1>{=j0SCEpyvSPNp1;G8(Mh1Pf5(5h^CafYKGQ6R${8k(IS zLha)I_OVcZPMj(x{%DHSF;MafY`Goj{$dLXNu8eexWzIWJuejcQWDxxkN!#Lj8_ij zv08NUbAE-da@_^ZPk}1%N@bT@+G81wJ3O%?qp|yA3MYlLJ)`k#Pi)I*{8))ne2mG4 z`Gs)VD`;N}`(z40V^wpiv^r1L&)j1gU+~0@84V_j2p87PXoOhnsMa;(tBBRr|4om3P+B4xUXK6G^=I$7c1ZFu;%lbLGNM8bwJ2< zh8Cifzq}vF4HB$Js7O#0xmx;|^ifgdT&x4f>XeVT^3tJqn{p+l8(kkyvPjL2chmI= zWtn`s!SyNSo@%3+@OS#DaY({64mzf5%_fx7Maiaoq+l>dEODB+Qb*H5g>YT7VWxvKN5V(C{4mEpGR##Q=9q^*jx7mObopV9)%XD} z5T_Z=t8x<#$T_d+9WZtVLTl|zjz!K$JO50M1%|w<^1#!7-d3rE{u*7=PTru0x%z}w z{N4KLexA(THKEP2*DZe?v8B!RKw-_27vHrNJAs$SVKbYOrzG86(0d@NZE-mcb z(kVBO9Bb$aA-_y@cgf$@xdNe$O7~{Obb+7&&P0KPEK(rHr3xhEOAb9dF+saxcSBDb zKc?k$8Tv5fkTH?`Q#2c=`{VeiAX>F4B0hl`V~6cF6GpCNxZ0yuzt;=Kdhm?4a2vmwEk9Hf0U*8TO<&ju(5Y0q5w za>5GRpi4CzrQss;Blrsqs%aZU3mVLs{OwI~aV%5EAeb+~1 z$qz?{Lo9}xZvk+kve_(3@FfR`n(~%6$a1wo(jNKjXST1Woyq0UK;xZBfuE{G|ii;d`+paia!F#zlo#6c> z7pGX+svc37v{C*4*gNw8yNW9Rzh2VmECdjcT_AuPGVHrb1O!~b_g=r(bSK@3in1vh zLVzIIlo3Gz!DdE8^zsvR*kppZpbm-xg5tuAI)ei!iXb{JaY0d0f2VGp&#Bv0^}0I{ z73cTMA9>$8b?WR@w{E@rZcAOR^fyvaP$qTR;amgRAs|Uz#w=X6YSF)#FI#aO*y+**PHtNv+@u{+ao)+jb}B1QSm?S{S)}4%35ObV5*n$vh-rMgL&c5TV@k!1 zYv*o0a4ZCPxEB5UgxdXm{gg8fAPHQDqm206y3-{Dz72Qq=FD(M9_4q81th*QlI=DS zH!9qbH#G!5TLsJd1Z1@Qp%nT0sMEk_*uWtm1Er?ctcu5RgiEkG@6h2yT&E>A!#cCN^<1HUrM+E3Jwr(1BF zz~>ohk#{Hj&nS#r(s9+Aw^O>FZR78d(++W!U^;0Hh>@zG4V_6Wh z@I3^^N&xk`(TNk2I?Wd(R`{|3jKkDlBY?4HIFXnM#!~e#uq45Nh40`pJMQ?qw@%9? ze0&pJ*>OYSOWz4^FOhSz<);O_MM=4P8iv7>xxHlX{BO3aALI!~9#fS=X&_&=_$O9b zIG72-NLYCaFms@dMxNZPJQEy@7{CY~o0MnXfK?-!MQX%AY=pBmeajwppj56B=dMC6 zI!a=!!gFfe@w6ny1)0%tX->zbt1>!-Gu4qk|DDmXCZ}UfPKR)&I?{_}ti#=n#i4Sv z*o%@cQjoioL;loi;t$*UI7>|csg3eEN*W}Qha+R_WPI#Btep~xf&Mu=_veG5P)6Yi zJOYF{p$7^ERv!F|Z485AQ!9hkr?o%Ug&-(~a=LJrKJJ8iU(W6%B&;5#6IyEru2rtp<^ONGr9pipGxiQxo}( ze;nu!Yp_Uur^l!$g|!-{PJHw~rhJ=}4^oAPcghEI=%n_eN2f|2`VQ4dC65aEhoZE1 zFAZJhp(wfzRhABdwD;xunKOyLvnP~9?}eSX%IDbe%b&AbPpH zP2~x($8|RBtkr9e+(6h@JCY}>==g7C;5nGe6Jm5dN7IyhvWmXTCg8C?O=LzDOTFNB#pQ9DA)G%x&Ea7QUS5_hRmIZXpT+ej@S(xeE|r^Q&iVdk`-_AV;VpS`xmdLgi;Z%iok>{VFk zVzJpIu;^izFwsc>=O#=VVmme;>xGwKKataZ93v{3LxjE6=R+)(O(0fJuPybb{SYv1 z%gBw&5k4$fzYeGUXj!5bQJ1t4Ep@4wE3>%{WK_2t z<;DLP22bYQE9@h1z4O^UoCxc*U}pHr>qunsA9slk^_-wb0Kn8=zdHJ)NHn zg`<+5b-tQn2YzMJ^IID`5gBh7eSkTfylD>> z&P+33w=K}o7wadJj&|jKHG5;n_zGXv*O)K6(T0h$lFUketlBx5z7@*1T0bkXg~Z9O z#1?o-${p`T5d3as6&r!qDJl2OC&REilZvT^Jj0g_dCZp$IZT{FX4d8Rt6P}fshqD? zzJJxvO2{F7Ub`}zb4K53EXg;Cn#%XURPThw24H zJFc%afxpV}Lqw@O8fqu>MO=$V>T%y0`pNITXy2gpaQFj-Q$Shl{q?M)Y&bKAKd6U= zwIeo@bIoJa9Qoc2xX?&VAkwns%1P+zipTE0gZAZ0m(vcp4U(toAmDiQWXFCgNas&= zboQr$bYkBVxv|DfWMr;mZ&w>G)laz{0%C#vR1gB<*5Qu(BP#xP`bp`~Mr7yR021KL z6iCb>1(Lmp#C+M*^Hs)A&0_ykLE6~S@uz|@Vk%I{Ms{3L%bMb*L7xB3y6L4o_rvdIJ;VA*J{(a2#7JBk=8*wW5*)kqe}9Ov`#XE z7Xk6XGj?+0f6R>raGF^+2&}r%FgMVUSviQ(=S_SQNND6?8DLd>)5QP4oF4$8REkIhVuxp(+>~KVc%CU81|F&;Cu_K% zb+l4teZA?_)RN?o_v0FUe3k_RINfyq4dzhBE@Htv6;pvs5t;6(&o*Vxw`v&y-mj#7 zd3E>2YfhCH1wna0d-~)jbaJ9!(EgDL8Kt_;x0y8pB+)@`ob1>^_p|s_q9>02bg@hp zhPeglAPnwLy|>FAC`1EMw26*;Ke*@ zA1uVHq7M?DPatKoSZrvqaDS*oN;Zzjb{!+40|*^KNF>sM@6YHI9Tf;40hrVR7zs;a z0Ug9^Qi+pC?c)AEifGTBw#h_I7&?`b8<0jVl9)zK7&?@d6PQMA5)0C(MHZw{n?zz9 zwLzUJED>^AkIXCWa!_Y_Kg|}t`W6az4y0K2RVrX3TS14OtK2WLQA3?ZaebFP0T46F zOFg0}7~$(A>R$6?7&u#PDleOe!m5(kNf<_S*Y+$Jhz*VwkGoZsKW;`efYZ&0B`XUW zIQ#6{QLi|4)l~jeAy#0%+=~M-Z0f=~ng58HKMee}k?O#Z>iNL^%}LpyVCw#s|2pvb zMw$=AiwEUzc1-oJFD~htDyRPm&=UO%@&4~CpQ#s}smgeA+Pnq&7r=4$B#py_MfkB0 zdLVTplVepLz7BuyGEU4o42`WLn%@rr1~Rj^miKEc z>3Wk+hVfIRfQbYg|5uuuwn7rV8>g&zs+BR-Y40cQRHb@uzrfhcq}={$(TmGqV_lbb z(oebaW7BJFDNeqAdYN(`A4UPWFv`NU=2TX(bAqV&R;lERt-4s{eksUj+%-W|JnGBJ z{csqyM!E0oJnC-k+Ui+ep4?vopQj{uuiY+aEeVXk8;vvq{Pei7v~GJ2N=9PqQ#)EY zJ6dMPai+8dM7f+Z0nzMyXce^PKN~w10XJ0BJR_~+%-}^peDI8&-1vn>yaC+N3?2j$ zQzxroo}nRgd_KN%t-c>n1Ld>LISHghfL=` zn(sqE#5>=Ik_}_3VK}E@*fb2AhGAe;!*HTuwhnleeom7a`_5Gb&2vp&9XO_eQ%;)K zSUlUnyT^0Ve9$a4 z>A?JFIEfP+qQOT(C$%8<@jNXj_DQnw{j`#9)=wd^6?l$Ktom*W_YS03_7_yZZaPh6 zNk2fj53^Cpkr$)blgK^>h#BSf1W~x2Ir2i(qvpvla2K^nE=h9XLKbFuYA1g%!yRxj z5-W0tUiYi|tt}1>;4_sZH{B}>8n`?4b1JOBemV34V%XG$#cpH%;~|Q`klFdbW3qug z*z!Feh=JX;eF`vp>xtw4H8Z3RBxL2$4$;8%?0FAfVfWOElTqk<62*zA2i2SM+=a%5 zRw{KM)lu@%z>ak|pCLNLT&6dWfUSeHPQ`*Foc7Qr-0N^;6C1n^Lu2cHrm?1e<$(<# zVV?RC#NsKRxcIC6Vm(39vkw%K%Bd@Ua_6kJ%#ez|%afzHD5f60Cz-amq;}VtdT5xj zOs7}cqL*y2Pf~7f6lD?uxw^<2D7#%&#N|QOTNE-|J!5RwwW@&|OwqtV?j(g2kNSa* zss(i)R>)mtX4Kuv@_>zM1HY#vcRSQhw#MneR&HQx3IiLtfsH8)Y~}{0FE1ie&r=}h zV?eerau*0(q+H%@TL0SR$(aKk#P4h3Tfk$CGyp`P6W=Om1@Xy24dV6jznW|ifQ<4t ztT4u(`r`u`H~>7->>9y>*O`MOz-x?@6dO9Qq)+~m=qYk*MSCx0m@ag=nVwrfc#(F< zJ=c-)rM2`RiqAt71RLwv_>bC^spA9H))D;_79dfQKhA;NsS5izOS@L+XYNmZ(sy6*N8&_&%cW2L0@l z=bz+evkVk-G-1FbNzT0kb4XQcI*&A=A$9X4f__7^nHRpb0< zTQ?M>_6_Ik8!oW#I-7yRK*HcO3>UOkSx|<7Ra4UQ{n!-uSf9JfpqX$$ClD+)&@tdN zH2`|-`e!wOU$sCEqU!!YLJPKMwgk5-_g8Gx01(;J-L6XhaE3W^U*H9z)Ob_X$#wzN znLwOg(6}oQ z`wJTP2I7o@#{UH3%!0-r0&!MBg9TblnqAOfH5TFq1r3&7A;V%Md z`ZPbTp`Q{o;Cv;?9${HQW2q9|m8o6OdYe+qN7Ktklc9V|d%A09)WDA^txsruJN6eB z_jwyG(@!bNu7bvuN|g8luMd3P1&vPzKHwJuUr#~fuD}QU&%oDP(D-iP1O7PhO)F^p zD)0gS6!`iI8q=qHC&dro)08CsPA_O|9{7NB17CkZ<5__ZxM$#-QP9{g@Bt4Ad@~Cg zM+81#Gw{tSXdDyxfUgdGvkMx>2R`7Ez_&p`|T%&$`TtBC|ztKFl1{?UWhNRcFGHFOOGHj z^U9IZIphM##G@p}D-CnRBW4PDm4z2bMqITw3R=&wYHt8vs3iA@;wl<%3w%H(LCHts zD;ii)3?wimAB~31h6a%JpyZ?RwZI1?5+&c{0chp|(6j(hi}K58AQj{y)huYe$Rf}L z;*4jcHP1|L0?Djw<7;5t@PjA!V<_jxkoiGYY|Z+HWoQV<+9eUaY84IK z@uO)SYR(M-amd}HfVl+t!L&$fsfWgY1c|_EXwpXn(;6Sng=RPtn&Dh%hBKiVCN%i% z92qWX9d0%c1Mysuv@OVjh_n>agUb!M3dA zq`{V~#9%*#(U=0&#L+G7@X78d_|Ks{>C@I4ZL;B_-S$*q!& zg4RcsDvZG2jbo&>=`527JYaTJBCSJ}DiVRO-k{3parpneYU?NTQ~p}ZM)IdR#`+}Tvv!k*P+Ml&=R&B66SSmXq_NyneL#O;@4X9UUd!c?vdpD2=HjpfXArrqjW%pvS zRy0fh`!!6T(@%2-!t*pd;|!jm0oqMJ$90BCdDN#a|42XOx{5X5vLr1)52A$&>7*lb zLEc}-!qmn0!cF|E|W>K3_-!wsODv1!u#RVgOV>Q!GjZBs~)cwrVEkfN- zEdYy<9qdk4uzgj)8WS`CyirMWAnRh}v0nB6*+AG2T62=aJuMV_2 z_}V}_5=>e|-aIg?p6uCx6sj{1yrpHb31q=@Y~WW|Avb|!#IeOw`FFZpAFna$uZ?vt z>mKVG>w>~0mvy9jf$NCv4ntz|Z3sThQy;<*D6AVcggV48>wtiFK^7G1)rQ2SX&rGi zW@;4h-N`<4Tr`+OIdL=R%$fxS@sI-0;uoG zhUC*am4yVnSxN5Cwd6E`*N0fX61*m`0l!~i zi^GR>hPzuu@Mmn?9!br9OZ$FoJZJEe2vc#C?wRZQDHn1ebF;g)bR*+e5Z91YO7b{- z%#+Dy%#zL0?K<`D)la# z0p8eXX$rT~KGxpMO~*4dbI%GH1Y&b8gO6%7XW5!R3fw|T{k79;OV_&BI>zyaBV2Bh zc*iVq-@bh!haJgIs&Phb;h0?3m-UQwL3*)vJpZj*Y|Ib?`>*YV|kV;xmNC8NufcXC;YavzILMirU)~R$4A6)$RrOw@Q;sBai^ej0RFT zP8yi_3ER3DLG?OOVARM{c0lUVv4KBs)i(^J3LV?JSiwYbElP)VMV)=vT>kl1;~ zIDAB9$V_SdEtIxVinW;&=+x9>81;Q~`xqb*k(b1*Syj*q;(4Qp1q4|HShhbvMnB0| z-=;BsrditnVpn&KA6S}QwSU)P59%=8w_+4SJ8WmaucLopN(bYSne8m2X4$A=Ao@CO zXLr;g1aJMv5e=letMFX^87=giF-Qm?N_1tVQ{Aizy5@4FY^ z%`ERt;FcA(trb6`LIYDLq5((*a(f)6V9Vm$0)`?zE!pE7s_d__VND>zBuEqNaUM$c zBz8^SMwcVMl(M8Br&OtX;K@e%f9{%`hR?Qi)$p}#OLL;dstpX``}^C{{c)=0Go*e~ zGk*`|6R)PJvmKI+vt(FyNWiCTvSVl8pb0zG3O4yB^Fr;qv@^@~+Qp77bMm~qE}r}9 zrvwZ5aw81_2W_dnxDLKLu&q&eYardHuy1qkqHLiyQuu$iG*aw7R-r@c6?m1&9zfkU z0||9O_CPFqU*$%iyB>f)VvAdCW^ zZ={955hINP-)N+Tz_W}r3Ve@|76J)P?mCF=FEB#~UaSy<`f8__zi*>`yRl|+VS&HI0-rn)Gl3aF z_Ih*C5_pW+JpjDkNDbf(=JhG%X$$ROStAX8gWz9mUMHtcptsYB;9t(!vNiMOGy)%%oxQEr;p%iUQ5Y||`kYdG|%{abL zZE^KoWhoo|R1{Hp!Xwq0R3C!zati#v>tJxJFAo38{*7F={9 zTu5<j#B}dJyqED46w(^Yzo$`AW=85 zcc{y_64XuQa=%=?yWGMwu#IMVzmReEHS98EB0XQHHq$OMw^G@IsqDd2_FyV|FfM|B zNQ_MO{(sW)3`Qp`M<-4sEa?{4a*Z>R8HA$daxj?lnP*-XAEe#(PMRfjupj^<>K(z@J^X>n& zsrSvac3z}dWx-`v0F#0?Z=o8$wlW4+Qmwg z*(P47EG#$~7N$jJ{a~g%$wdndh?9wjJ*;*@iUW|2Iwl=;Ogw6Hy(Xw%hM`rd?j5^jl z8UkX5do8uQ<%y+1j}19PMg8THhXyR1S}CTZrc#QGBF{4SfT({GDM$L*8Z-B$Gx#>6 z?Q60Bm{yeI`=;XitjbzIN+t8eEgCVNX9Lpg<{!#qJVBT#_tShi4Gr1jznn$`Q=vK% z8itP_*{Q$~dk0YY{8q4YK1sHqe91LM0&AS(1%*!0v zA^m}h^qi=;JSP7B9I5?QbGsMdU~?TspRKTgZ?)2_1MjG?eMRxtD>N|GE%OaX1ajXS zPQMxl|5Lc^l)9g9K(h~G{_qoDOZi$Vj4-=2}yt`>y`5WhVmEnJr(2I9A8q;;l6 zp#iK$A^m%AE7d9TD$^H;J5EE(oFlKo2*e%FNQ=A*BM^5yBP~1EqNxb%CY`*VEe+Uqsy$f1qlJZwk%{D`0SyNvGj z(*9?od%d*r9{F*Je@6YrU-<(NZ_$$bu$A7(bd^3^KQn(O;$Kw|0=sK_6yS%nPp*2$ ztI)UUr~6~S;4xEJe?%b)yK8$DvFYztsUOwPnesK&o;&YV{_X@Cg(^?d*W7c6z85NX zZQ#l`bjLPU1!L=F7&uOFY<6{IY<|ouB^$FGrNwlTe!3`tFHo91z(7mIQZ#b+c;HT_)ehfzuKrDVI-AP0S^3MPgpF=|-=xZh zEINsX-196^u|z{G1riO(Y8r(kThG#LQ73Na3BJtgV`#SO!xYDc4J))Sb1Sx+N;p<}Deo;peb_*jA^PlAEkG>6cHdp5>G_FkJmk_9X z>&R#c;ljmIQWea)`pJw^Xtp?vTVCQwZmSZ|CbQy|N6&?ct^)$f<-Pjz3CH2X(i|zq6+J_ixo!L#*WFD2eL*4nFxm5?^sr>e{hO3*HX?5UMO6sjGUfosD z+D57UQX}OOTtL|+h#)X82qYk}2;-qpp|TnSSuLEUV3x!lw>4%lyhat3M51$?2CO8; zOI=B}!mdtu<`bu_HRi!cZ)5U*jSeWOKT2*}kR)>`xr+c8uY9&E%Q zAz=&BGRqT!$f^9rjf+;&z;HM1!34BqjI6354YR@=#JVRlZ)53C)qPoN=OO|^802r5 zY;RTd@7w!0%KNr}WUl0+5z@E_7_uSq;%pqRbv0)7RR>aflErgZ#^bh8{>mz_-X24kY}UW1#!C_u3!RPr1@CohsL(O;l#( zdK4RtAl#LDz2X;I%SgofYF{Tg!Jr}FpVdYx4zXXeDLn+lU|4A%V!;^#5?WY08H9Ez zOcXOTb18|#(OuiWU>IR=v`*RxqmeN+Jzhc7+K>L&+GcU|D@!!&%+Ap$X5b zeVyb4Q%~P0@lgtVqr^!pV2VgJ0}&?Mn-2zQn3I_ao-wxV>sk_ju+pdlDFum5y4tVR z`gp^RUZPx8f$ufaBH*`_L_-G95ZhXKCl;joDdbd(bEi`TnqrI5$KjAycV$}jNN<^(* z9!7Y?xLG!?0bF6EK_H@>t?Bn~$m+F$O(1!bbr49>oUEp$aIK9U0aleZlZr+u#Ud9t zAev;609KU_nNo^H7=cuZvu>!MMX?AYkV^54v^KW783j@*o{<*SA`*erif5!XV7VU! zQWl<(7U2?!ak9HKxMW2R?|~PDxC4oqTb|u153&2HvO=Fsi#v9gw@VbLB0X-rqtmYH zbAN`AHhSIPg=|VTf+bXlzktjwXp=g~uR#*eq z>0105Tc!tqpHq@MYFsWVXy0b6bs!mo_5He*f5F!D0U()@sqiITpns*GrOPykwHy=@ z-34K~0SRyBYJ9Pp@74!17Ca6JgAR+jN(c6-FmJole5y)hsKF zIrS^u24$J(oTam^lScZPompYb&q9S|ndoF9bkZ0n`t_Ox0`0zv;Ji>!{8uZkrvkB0FeM?6+Sy2^hE6-zil%ji@qrzY@iO6r!u@vz0wFcimQ zJ1c&>O1RAw&j(Vk-L+LU%98SPGzxD+-Lg0um}2az&8H5-a4G??RrWZy)xJ&+|DENSJYjbEmmkOmDbD^mh)-%}8Tofo)Vt|Q zaF~9|^$d8pk($8yMoNB77z=V2c_h7Dhq))C;zFVCl_t6g3@cKy>Tj;A5alXYlvwaL zG$C_rD>Wa;l#|72Ba+KhbF_Gh<(I^bP|o@%Z1EIv(mEOTBrAov?hB1raZUDAT3nW& z$Q7DJ!b`+w=7OhA!Ns*+AuAWm3F4Cu%oupwtv-4sl^b+I4H&2bd*ip}7_6lb*zyp*tt#+n+r6J818D|4{n30mT9{G%RQ35kO z7p1RhB|c==i25NKREkqh%qi+{=2(v%Y(bQ3L(7VB`dSc9X*uEp}b6|sgV@5d%;|MvPR zf8KM8p|O6NxToaU%5Q2E?s|^zk;fV^@pX38JG3^6i*UtZ5fV|k(DstadioPp_j^+} zfVvNux`9;PK&oy4bre|U`_ezFx<#h04qT%q-%3hGlUXaRO&`CCq%ZdtI=NN3Jc5Z9&Yyzn>ndUbw$9Te8 z5^4O(V%!80z>+U^vn0ljmc-cA%#@R3SByc4OgW*(T0AvYjdhwh$OdY5O>H}IO){zU8mz4`tDJwSJWX0B7N9+LWRV7ez z;*<%LP-ATp5~0SbJT=y{HWR2*#s!Kw>rAb?KtW&<&y)_gl)1iKth_j4#X@J|VvCN} z+QxJFS{JvGE+oYL!RGV`@J~we-%HLdXz!t$1X+h(qL6|Ufdui=zy)M?AzTD?rCn+2 z)I0?#T(L`At4l<#<4V|RAfc8V-jbd>A|9^n&DTqws0~lofb3|2X#y+ZTc!bdi*34= ztSV@n7or6uYME$Vpq5^*pP6&cm?a;Vm@6BXTUE<1O*VX?mUA(XO{m z;2uiquf56rdAq*!F*G=s8F@Q0T$!%`U!dCN>F12vBKHT~PwN@$g0xV(HuYU{?dsce zr;8t1**AAb;iC6sGrJTpJk|wevGzP-9@q$1cN_ zIchppWv$T98MU^H-BO7Pq>Hp`Q{N>QITyQgC95uW^nO^m`aPG#4hm+<=DA16abU0j zwoL3|mb}9;S2lK6s+nlZi^_0iz9YfGkBpDqDH0V(pO0d!*c4HEL; zlwT{yag15=p2b|*Af2gK9j;Mc6w4eJhrEIh>)6lR#cT+8tC5<(uNo=2%ZUZK+e0LM zNr&amLmtpRjL$p>_#s`x@g(yzp_)oLOh09-oIaTd)0A(#k(LkL@APo!L_42P?w^(@ z%R7}NYZgqLlzDBmp3J!!YV-DEH(XicVB*OoFNziqFIApnLRwxY=UFH1?&pK667ZQu z+oJ%VYqUKJFz0`{r`>K*{_pCid>I0KOiAvE{&qoY&K}+nVFVt#52QuFNB8q23DN46 zGz;iK_sC3*vC&3ZwFmBv;|6p_G}R{3cB!wbzs;6f#p zuF&|mzz5v+;EG-v`_40Yz}G0Llt<&Czz00=MSf_>N8=nNW@dDn$yS_65p~lF61mh8 z*#f>_NhKeRIeX$l3-}Htm3%aA4tzioU-Ho)^)dy3n}!hIw1Nil7d{}-FXho7{=x^m zKcwQO6*P#y@BxWtDbITu{!ioCh~w!BuVFKz&jQ*24(&zk76J)_A4`i!$fL4AW`k#> z#e@(>AOZ7?w1}B70*RPsq(#t#5lF~9BQ2sPjKG7n96ci~!YYg*oF1h?ZRUd7w4g4w zpf-UNrX1w~|0Qq%DNEr3-ewir1b*L013*&Zd}0cmgESo{?6Fd-6w{(zp+?4y?4;4FL(O{8R`_){Vk{bg#8^(2)4&dGs|x|4lz~7(DFf3*c$D?_6}H}v0*8z= z1Pp;6jSHTtI?q;ufz;;#eCFDeIi>zea-kl|&72{dIV4jY1d`14k=KVT)uTZ2>#A}n zGjoP=GiRvCq>`+XQZJDf(?vW3l2y-0tCC|NbHy{#BDWG4Ah{JKzzNcvW@mu;HQVOb z0FZpR_g#^bBQ&0l7EdQfBe=xS(TLn!y?#1c93)4JEM;rcmKeyPZNz@RbBX3ICW;oZbgv1@kuf;h&;CRXKQDogBmxv|yNk#R0!URe`j zmIu|bKkBCo0(c)SOgR~&mNuT}1-Q2qgSpPRw;5AFO)a3zp&HUDf64P&-7_IkNPDOv zgbRtf7x%UFHvWhGqml)v!QscrE2Mg1=a7qzjdQI(hKe{a6KfJi| znqoS8lxeftsqzOv6E?c1o2&{OuPLUJSZy9;$r}iJ!@H5w2-edG)=3yK5|`50)Z;+h zqSZq6xFaI)ZdNr-s)i&wHWV(flqEHJr%Hc7Ki!`uc55%zxkM7APzM9CAOu{iu2$zDG;a6C&t9n9s*GP1Sov8Us&ZN7kJnYJ+6RsKK3->R z{ioFG?^>cp`gO%bvDE(SRL@fVw3iW|61|KV29$4ETf)nTW7gP`tJDH4a_c#m2uVT4 zVR7p@;XrF{@s4>i%inL*g32u;H%1fR3~rm==1p)?!<+eJFf@>~!fS#H^$6;eZ+&nMWW2TCyGGXw{Q#(0_P35UqY*I;#Eh>qzK_xM^ zrzFPal*CTfaAGVcn!t*It5;?WT)h$lISCdGt5?$CAlOBrUC<(rB`b}P)QOLU*#(G^ znJ}zgNf;=^QV9)mC4-3!)mie<;NVyIfE+`Vd^E_i@O296348e-SwVs4D9OF4WU0Ij zg!bunKGTY$0&9Bw(n@`6rEauRi(8#6X2v<4?j{}Rdg6&Hn`!Bh$vQWgt(7k7kGiOI#r09?qDm-;N*9GeZYy1#B1Yn*Hf~x$gQ+96 z5l6 z+}n$2FIT!;V1cAk?Ac7SK?^zK$ zxPv_>8DKe8fY_9Hp7A4$yKM{Rq4FdK-W zVz#_xh*rhywK=ob=FDE3Gka~$?6olx$Pb-q4oyI*^qfRu+O^y?_Z>2hP-OFs!tx6bVOMpBThKDfgHt z2&2~DWes&?!+*IS&tgxd&e^|sYP^JcYCM%aHJ;0!8qW$(jc2o`#^L&o=8fdYxBW58 zPnG2{{ge$oaE4~LyLRfTc-W)SFy3MkkCk`ucX)lu!&(r(u)=5pDTJKjpGCt^9FOg+ z_!2e!TK#l)&|nhoZim1m-`x(ysduzE@s1X!-O=KtJ6fD_M~f5gXmPq7El$bPG+i&# zPkD$F$d+7w5L*K?g`{#;uk7QZyF%~594DB|hzR7Gd+t21T}VK#yl!NCF?F=~ALbHX zU%_#JJv9!or^Z3{RKm=<;Hhz-Jv9!sr^dnd)G6364}E6r7iz|Sp=RtCYQ}z{X6zSg z#(tq@>=$ZmKMVEFRxSg;Bb3xD8^-MmS`^a^`P0|>ni@#dRP}{0(5SQ7Wu5VYo4DSp zBdNRhXf1j(Xi<$7GvccB1LDH0s)!4;+Se&if3Jl~qOrexmNKr>k{Fk1NsOzsB*sNr z65|>ziE)XR#JECBVjTAuY87I)pN6sFo*G;3sj=yv8r$xvvGJZ7Tkol{`JNivf4P?L z4f-ipR^W|BssryZQgbpv?t+D+PwBASBc}8*V>0`x#q+!}=1OLRFkjPfiGIqG4`hnV z^1cz6>E2gP_Z>HiCwk?Iah)m`60mX+*(jc)PD>mbjC(572+P*^8ewhnRN~A{SiW*H zUdeOanaUSxrt*cFseGYkDqpCX$`>k?U-3$)RDQ)Pp=P`iYMiUrXaSvMg){&>UrB!9 zrbU@mR(eQOo2=U#aY=zj1lCAc23djGh z(k01wRj%xD<(9;_a7$ubwI z*d0%e{qfY;Ay17x^3>QRPmO)@)YvIcjlFul)_+4k<+=nMFj5_OypfXYQY^?_myk59 z!*bUp`j~Q=>(V9jywyy$%ysD^P0p5f@;x8ON+4H^elV+oUYGije2;39chCTt*=~3| zqcg9n!H}mym9WN5R3&&$>F28?p;;ajtLET>tM%(T~-uE?f47fjvfQ0yi*H@=MQHkXyZxG((5wR_{Hu@9!n7 z59k8Vq|PsQA5^bCWdUmgKVzg8@HQj0cR)kf)$O&isb^Wfc zB&M@WIGH=s^}1VfrgKL)e;dW4!@RbPn4x)H%&YJXl zl8iOsd!>pskL0Wg-z!zD3EwMKta&77P5Nb1#+rNV+n9nUg)9r%F5fv>lq@#??_JUQ@9D`=b^_<(N?e0>Ftw+BAp zKLx(&1&zxBAMitgufL#iP2dB5Jn+pZXnZ>G0dEg{GYcAD4t&6G1-@AYjsFgOz=s0g z?1ILR10V3wz_&p`-npAN(s z1&uEU;>?1^e+S~Mg2s;nadtuD_kp-UL4(CijGt4`m}4v3;@rwsr!vTMB}WwULGygh z!zWZr)Z@DfYeJ*ZRxV)aDj1~+UmTBxf&L1Tb+ZU zi!d}SSc_ttQUgv&4Tzb<@L0~P)HfWF$N~RRyL#lLAWE`eRpnzeVSQ+d4PxXrBL7pNfy_8<& z2lcSb(5iHIjo>h>5i8qL*PbJ64N6!J)~>^hC5a*|NRh2+Z{pYSI-V`9tj8Cv>{}vh zX>1gnMQ4q$AWXI+oumYoI`ADPxB+~pl6q<~EocM-lBggmu(V?XU_piK&4G+6wZ-FD zn^=%~YNw`E0@FeB{P4^Xc*>H*2$q)xDZp1KshmkPFe#%ALjp_7TNeI@1r;I%$f#0V z7~v6}VZlo(Fl9-6{rx^(8s{T&LwQT~J5ix>b!YqJd0C5jwANGOlSohH+OzG!UaO}P zZ#?l-<}RzZr^Zp(dS6fJ*K;0ue;9CIBQ4;pbuaz#kZ=J{HqwHd70%SPMz~@(}EI zQ#-4+=T1GVA3dxL@>~=JC+4S4%uk(|A7`zh#{bfs7yuHpP2~yj*?*RQTZsi%{8}x* zXe`Nd(Hra>i0#Y|Oidr?Wcs?re@GoY(kdc(lZ{-yVk3o=TC^~>C^LoSxilh03}pHU zH8X{Unwi2vjjiz#GE-Q1GgDZonJFyP*#16^%{lt%Uf;W{d~vH$G-~cmtMU%Pv95TE zyN(5}1FxCgHUa{rZ6m0aTXh-)?NErI$%+GGvf^afh}dOTt;(t^->P9e3MJq|&7@4I znUo1Nw#-Y&q)d1-DHCcYWkQYZ-!79f|Mq55#x!;z7_4T7cv}Yq!X4h;425{ukl0P@ z$R(iE9y`DaT}gmET9ZkDP%{Y-Y9;|fjV+S|2sM)cp=J^w)Jy_|8r#2LCIS9U&Ln{K z$;C!qDLK}ai?bg>oVkA3kXY+QsukJt8_~}yi9ia8D56kEX!x34Yes=2SFU3O`XIZ} z8;zqZvf>a6iTIL!ApvLVPD8}U@fNN)*g}eZer4iwa&e7WB6`Z1D&$r0W2fout!@&0#De)&3p>o;}vJ7O2z?B@RZDf21js8?quvjn{R_a%-F=mDZC;(|96^k8k>E(F!^s& z9I;L15)hkI664Y+i3FC5M@fteqa?<4Q4-@wuhIxVl_Hd}N%nVfh$NRnjbr4gagaPU zj*_RwVe-^CPG@VJ-lm^3YV&!6JEZq%*QUNJdJ8$d*D6=P=W362K_Lax%AU-`avHmF zEljMEoKCEg9J_(7=*&D^2@B>(JZ{M@UP8@8U#OT!iFyf{=nHQq`aj?hv0-ROO6Hpi zSWqAyN?m5+Db!3ng_?<{P&4rqY9^jS?PNAKmBksGR1#y0N@8qKNsR3&iLp5)F}8-Q z=P`OG?@@GbtBuqIDy054aPJbZM; zs=k>QwYDiBtu^iu;A2Kw6vr|3t(EhwRp8rynQtv14#|Bc5O17st%4Q~2_q2GWT$@B zs>zLOWalbL+8bB0U-`TD{Lp@pO{pgE1|tmuzi6c9eF}eVq(R{0Mrs1j zwAnQX{IZdnz#l8AtR)&uP;m@M0=sI1ac!kR8zpGpuyPv(5=VK-?@$~_NBhf4|FY)d zB8$W*kcCQSR+|Lrb5J<7~)K>Ns zG%g8zRL*||5|#5vAW=Dw1rm@7DisWCEJAn|0tu6t3;d$Rbs>-_2^a8Jfh$h;{aP_0 zI#Zo)og_VEHP`|^Vx&6oCq^0qVnVk(N*aX4%KlRUo+1T&~>woY690P$xZ$EAHyby9xV9&R|>;GT!;{Ty@m*jXkK17O4j`6H6OvI!dN9T@9{cBSdjzOi0BeZacR1V}86 zSSJD@5tMD!aC?ubyw|yVrhI8WSN;pTiOXe6>urdl4kQGt5fE3?m znT0}@MY`u`A&M{{UUnEsflv)oTM*(5%czW?xPH=6gXySb8DZ3e8rJEuG3awo=0hMx z;xyu@aUlV5(1uTL-6-N1oB1w%GmSW5mdzrNOS@dsV_l4Trsb;vWVGykdpZrgbX#cDE0)~kl~9i`NV^_^)nP>gbv1FS}AN_B}W!n~em z)nZNP2d$c0!0VLcr%*f{8F-NmYycUUStHgc_cv@*8~DA7x^_}{47|ezwt&@vt&Rh~ zYy%rWG|QI)z^bf9M_F*o?gGd_Iao{nR@7QmJ@arML`flV8zYSXw^fo~7-@|fV;gv^ zkw$>$7^w|>!x1PQ0lra5UIVSS7-JiFxsgVIR~V@cyx&M8z=w_02L9YgBfvixsU25W z8nPB4BO<;mX%Q63C6J)V>;qnI@o53q8fg$XU+wZ5XuaO79Rw0O&q#~ti7CK$gt0&Z z>c`R|rZN_Ibr=gI%zi8_A}wQqL|Q@s3_)+jL0`96F1Nt7rsnz7eEbV~GttZm-n3J1 zY(kJc+0Lcj*n}WW!j*dS1SdqVMw?k6YX~MY(}e{5spVz__(vu6)s~kp_NPsG%k~|} zIwAH0iHWPYc0m~4ZKXMY&-VmUTwWO$@Mhy+?*v|9%%paa8)3vQ5+S6??P_7f>b43Zv{0|us4pQUl=Cvf~8JO1Co6| zmKKIj*#|!yHwe3@#-9A$7|6V;OtHUEF}_7B?mVl!Ch&YC)qxiosSUheN!_*Oy>Yx_ z;nU=M(#-cJV=4ks?iSPi^{-gmyOr^pWoXY>mBQ5 zd0|>uM_g6b5nC45bYJb4OOEQU)ndueW5Lin$z|rttN=3U<3#}Pgf-SKq*$?#I_YG= zju-G3*-9~RbQDop)6tn(sy`P7#yWeo3+MhCexj3rkpn!S8lcCagWk!PN}Le6Zb8{p zgR+NyPLp?7$hjL$jsDkOqNw z8>tN(dWCU$5dKf|*dmXl$?tw>z0Z=?0v=+)83hs~KbBUoBl*%SwWE=I-Zf%(=9L!AffP#w9W{OKnln+ z(pqFy(Ew6Io{`q2fe}dbJ!2>VkJ1D*a|vi>63`?8k7`cj80knbX>eMbmXzmMteZg6 z;2CLMYq4$uiLPg)MS$g03rKW5BdygI>n4!sdPZ6&S_qp!qU#xHeIPIb&(%rj8EG*e z#YSK?Dd~}Tnv|hjQid`~86qjw1PoaMmG{MavZg;t>B5%6R#L}LZwWroxbLp83n1@Q)h*eaR)1vjHQ^RWbDEMB~NC8Y#U*`E~OqP6Q#(KFgm|8L3iYM^iDb} z87=n!@kk-XqxRHMYxQ$lZJ9g5h}}{$0w2;@tmIua!Tac^l4D(mB^C}Nkja*LOAe!6 zucPH}001{uGx}<$l?N)cvEIGs(4Un+Avlac|IZ>xw7b=#APEWP)fX1yk4vbD=>!k2^6@}m+-L_Ixhm#K8+OBUyz@3!j*uWH$Y|MZxn2xQJ zVN3$Y0}h7Sw*8MbF?KHun_*3IH?v~chKiqFp@DHkZVZ3~F=y@-3M|Q2`j6;nZfi43-88*Z;)cjAnp^?7U2wEG;_*01N<1u+rx~hk zW4k|V0XH>L9k_*&hJYhV%Kbzo7T)0}DqE|F*XU=buUfX#ej+1tGW9U|Mb}n(pUCZ? zeEE5m-cxX!=DrWrl4d(6Bub8ra^_ppN7W*9Ou)ivg5&_mD0eHE-n1hdZdnF_)q&{) zIt+v>^B^NE2$W4eFpNs>Tewf=HaO!gEZjefw?Ibay=5TBQQ|GIIxzK?fpBHKeUqk} zw9Dh#@#KN!{dNX30^|-~mcj?FO73162$whvL|xut26C(>4g;$LQ->J{SH@vVeJk}w zz7=}od*z!y>YZYT9!Db{@BVahJT~{|lo`i?m;?ieDLxpnuzZNl)tSp!2~j*yNU`y+ z)T_uv`YE*y+*bFM@|7TP2PHWR+6Apg0%IrfTNz6W$Y>|4WwPEH7=c$9X%L7;KX!7Z zjhxa(Mri}3SDMlhU{z^aXuInG1W2rbSS>365Si|q*;C|EY})9^EEcJP77G5uj2#5t zXQU<&+45H=?@XuW3w6M$HZZA`lqI>>PFa#)1)%`r<>lDD1r6dOKP2fbXkd7stCStn zH%FhRCSPQx*MSVpRQ{)xn~2HA21vwYMgkF(58hgpb(F?TW)W~34S-}6xTBK%ETXly zF}8s>YJ6oZ@E#+zfzQxOPo`$D>qN7w1#IgkR2D=amiSpk3u9zg2Yi8~;zo1uaaMLI7gAT;PD%?Z?u>bQue*dXcUL zc(JjCy$QrhsWu=cI;BlB_#lh%AP{>zBP~o4iNLCL=~ij6TD5@%v<}?Zln%rpk*h!~ zUP$rM+ov)SC_8rGla@`q5ah!U+v%yXsh%2J8%vZK6pIs5EGsk5VnOcyF_z?Lv8ddA zW-Lo4Xe>%dv82qt20;&7o|8J4(nrvpdiN!ydlgbfnH?aq$%T|cYUET3<~3^E2hGUV zM-&FTTgX1l>a%sT!Y>BWw-nwNNO<>CTQ*z2QP^t^w{ZA}ffScYYG*sOvyGicm<4U% z%al~ERTzJ3-~--O#Pa1@5{nm7EL}*ka3OWF{ry%5qp9rCRQ6~pJ2^wb?)RGPg{ka?sqBTR z?Bpi*G?lqRKV{-Rwla3^W!kya45$M`;V+6u9iyeaNI$cp_t4Hkn+J8^8;q2QzFNC( z(9f*sv$gXolT!!YVx&a$f9mAU6Ihf->A<^=@%Bgo1HY*xw|8h4v}Wl$j5C#v_}xr=3;1;tKMF*k z6W_{+Pl_G!m@L~-AVxV^>2@?7*vt(~cZ%u2q1?dqd5d)5Fa!VHBC!xyjl^&!5=k|z zPgyTcYC_99)aU*Qiv|9x=FJFRkc-SVF%>k5D_<}{0|>fyn35d$tim8$i+v&^*K2KK zC)F$aO(6D8xscD)cwQFf$!U5Jy<9)#hViWm|0$5(p<#HBMMjt15QbECJC)r|Ww%q= zx)KK2A5nYH)m+IFceVGcbqsg3FROXK)=x)!TrnARwD?~vU-sRxcp=5og%k_VY}k=K z-Rf*U@MtB;rQ)JhD7?<1J|FmVC6#>r>dy}PSrTdNrbKy;0l1{fhYQ~ed_ZDW%1ix9 zULcQOr<-3RxIju|BBBuKLP}*Pzw$!%sU~|Ml|7Kk9!O;;)AewbIWG8dLCjI6XPN+}ybPvi0xi@s5r?yx-9fqybmBKjmv*tl%QqW4v%ms)n~z&9Hy5j~tP3e2;x zHnv!69yIq`BWmt(W_Bx`@l#w)SZB$pEJLw(sh#cA&Ng;FY<9MRKUR`d_!X-#{#Y}< z4Sb!FO1{*iDXw9uN6E65der*7CW9qHCNgkb!KR+C*9y$_-5S(a%W&{e%Zj(bQDq6oa#!no6c*=>(D~ zDK;Q!nsU;%Od_#fOK#6Zd~z2sL1aE)RXKoTR8r8FrJ6`mYOdubS6WRXcO(y8orD!ZM^Zl|)7 z%jNsj-eWaa?mX>uwT|JAwnWW)mwr0hWs1pQ?kp~rFPr&TypUq)LW+fF?gEf~wAI-_ zDtjQ6J&?*yE-lD@naLhWWsjt?M^f1u&9G$es4{oaPr1}yx+-*g^gB(=Tr;2!JkUtV zzVkX&_A&j;iatj>uhmbvJ_BzxQX+bD&GvTsnHBvD9shJ?64AhaFj6A=@O1J}k6{gI zv4(ui+;8ow>h?3UpZKg}JtC)>-EwMYJGHZoou4#2+rZB#Nn&@!DvS^7!xZOmYH2d9 zQ%hSnXplqoEKH4GuR%^H%tDrw%4(N-(;@}$vXYs~a`1Ky?wG~BPKa49Gdt0LQ|_;Z zQSns#qIPX>Ri+>?!h*ugK5!Eq6j+j-=o$~#?w?^TlDiPE}IXO}Pn z->D?eNb4$NYy&@MBn81iOxwZiXaO%Z(n27{_ze-Q^G*ChAjWt`TG%5~5Ex8TZ4;T+ zz_gXBQRds_t7xFb>1-6VP$c_jAUey@L7zuo(HwnKKd043#`?!LqIsW^`yF{uditUK zOnYW%&*)e`q+e>+Y&lLF>%UkAUA)m)Kh$@r=aqqwF43-;PNo|tLZSV$_RTV((g~5& zjMoVD91UMZBcyY+tEv&&+qG|;Mu^0Fyhf;(rWzr=zp7Cp>OuqU%5fUC6C&vxuMsLq zuJ{)6Vf~a<1bDHL27n(mQXNR;$@V!`o7qvnP6Z%PSb!u(Ecm^0ZKtkJP4;HW z9Y)nti;{H-?us9AjP6l$ejL1yS#D9?Up9-|z;j>endKxCc%G8{qDAYBR~e%R;eW)> z)`7Bk0M9YfAP|9m;h`18C->Q3Q>(wBpYrF7`*$zg#D>D)zS`JEt|yr05fZR!JsWuY zUpBm(Xbg|CSk{4WHPQfqyv#^-AmPbW{37LEZKDS6Q}~BKdX?sW0zr=^qw49Xq`c8F z--;`_9r&~g{)(BCNLj6;-)>Ucz(ZbLEB{G(0}oY_FK=4k4UB$7_#g3KG4UzaI2JlZt8U}7kSrl)WLanil= zCHfaBNekH#+P)_iv}`L9SIHX<41O~0X=f5 zLpTC5S3fLgRA22)9X$8+$+Y^Pp~s5zKTb#Y=dFr%FTmI%wTv)o!X0FTgJ;Pl6X_J8 z8wL*2TN6p=`XFlwkdkxK;#zaGwAhj_3^0r-;D&*NVNDEMrgiVwz{~|Hc;Lzk+iJyU zS7=}yk)c3>=rqLneV^v$Q#OSgzx5H22->j>M0r`+aYSGAcHq_VDKvs4o0V3Wr2H75^8k#u`&5VX- zPDA=EK&oLVr(r0gVJN3zD4Bk#hG8_EYN;9ok{lP0;Y=kY_q}%{h4vQ_|KF;{Er6JR zmYKf@v&pm^-UEq}8wn=s!WO3-e{u>DDd*X%SJ6ObPJYSQNds=Ux1(s?BP+B2`9c!s zW;!Z)O5qd9D4QZ@TJ#pC(Oa0xPNq!e3uV-?7RV9~4Bn0cPcjQeQVT{>3v5j|%NC}Q z3l)ZpCSNUK6l&M=Qb@tzXvJsNKlw5Xh@RCYU+-A-jEcT`8Hag3G;`35zP(L%aRjr+VAx#&v@L*A2_9Lp{z zV=TQS#_~&ItRXv>(eXhG@_gWrlqBQMU4`1^7Ww(Wo0U}ZeOnv9s1j-XISdBQ(=e2S zapMLH(|jP2EBRv2e z%=PbT(8mQosP^bX%JdyeS{+ymaldF)JnlyA9G9_}X@xR<)V!+$Nv1^E7?&Nh87l{ zXvVi#y-Bucj9pGmZ>OfWQ`6g+exsS*27X3K600j#rT*mBGW;1Zf7-xflvGMfoouI0 zCd**zWGi*@iRWj57CK9lOnBh9iNY#exjx``C67#27BGV{=bqybrzO0DV2k*s+0bv_(xSI{~YM1{y*TU?#zDR-DMgK%?=%sf+_ zjv7cuC10VXEXig(Wl5x9>6JDul0_62{Lh?8{j-h^EJztSFNCG(j zKcA1j->F-tPSvmKoO|!Nw{H|)5Jlc6Z(XieQ`(vBeF!N06DAh()3jrgW#wo0y zXb`x{L`~o~O*A;LG<0~j{$@fuSBKfhdNqL<%!Ib5YNINKfm`Txin{Szb7u(n%u_8i z;0Y=U^MuwnPBTT|cU2S=Y5gH8h92?%@nf_3(E%QBek=kqa=6u@^&K<62*}7mkyi90 z+4xEQXyM1R&5uzaBbVk|1+8zHAEQ7<4vMs*AIU=bI^|(+8*~`+?}(zvUqHiph+*Za zJCHG_>;bl*QtmW?bzAZWqNGS8434$BxxsS@?j@i`@pmFOF)!m)6w4r$;ql&4~f;#d)fJy>%M` zK1T2MWC8)7prSxBt&!IUtC<&+G5-lO-vRDp=0|}TEX{WcTG4#6+Fs}cKsjjGL{8oiBGH~}-1-)R9)(ccBiog#} zP^5M1zgjH7172MlnZ^bcib3EVuc_(Lc=v0~9`M5|3Lerr>TFX4Ua6ugWEwNhF+Ctz ztMtOr`2W{vzBb0Z0c-s>j>jGWeO2|pYyFx)k}TtG8uz)agyiwdn#W4$lU50Rf-0d; zMkVyjS3*y7C4|!5B7d94AF2(=tu#HUm9QHV&?XmFzz3fhl%?OsaT?GgH!hHKnaz+v z;FTuoT%~YdO@ed;o};2_02)_EJzygey>T4&*V`-_9MXxhT7L&hhx^Zx=cP#AW>aYp z_!SjZgOjL7+e{h+GNURza`)4yN9NA9IWxFkA#*2lllNOKJZP135)gJ|!0X3p9B!-( z0c!<*y&jFp#Xjj0V|V1L`3~drZIR?V#xvD?$2cU_7`bB{Mp#d(f#3uJDq*IY?--|= z`(icUk*nrA#xvENek6*p;WAI_wOgW{~F7?0^X{iV1gjMQ+sm4+d z%rq`0^KAue0>5jbF`W2>ts_m~)h1dD9I(^&*ipK)G;OsR z1I7ht@e|zXAoDo!-RewqYB9E-ZMGK!8Kb{3F7I~DE@*dDwO{5!Dfa~3K`<7?~%)X|F^JQ2}tDtglELVzL~ zRQU2Jx>OR&$SrK@!D#BiXzIae>cMF0L2@yiaxmyU=xrQQco%vb2NvFk@?}Px z7$Ww@s;hcvL~aJXXT6PslX^;B8^X2US&0qd+4**PI^-Rb69axl7lv@*e=K7|K0TS? zyvj!PBzn~LD%T|{9k25Hr0kMgZQ`U1%M5rkUMEv=sY+53sMHws1D&O0N0lV%S0*Y+ z)NfN2Ldxt8W>V>WnXUPYlQHpSuS6yBWeU~sQ?(KcCK>3aL?s#MZ>kbkg(TI!@T(P# z^TK^#m0aw7kyi`+dMzTqT8oHV7uISK`ITBkKEs|5xx~hUCAJU@;=!)A5DenM3C#sZ5`Jlnk99M2 zs2LiWh+c_(T7T=ukX_5wTB8Z1{OamH|Mg?O3hKw;{3vC!e#}=y{g|iRhbe2}#&P1q z;Y#Q|t%TmiO6Yy7gx;x2=)HNKUi-gbc{m~)C|9ovaJJpFw}9Nw4>V5My?^H12d0f7s-j8*j&SUN}GY?BQ>FwX6R}#b`Z{_KF=uA^t?6*TQCuVvLYOjID zs_Ps5r^#L%k_Bcjxe%iBv8qzuZk{f;m9(E@UB`e}lRdvX#`{ZUBrj5BSxV%H?AypH z*eyo-NaBEWmr0-ACNg^DW-R3t;@K)?;lkwm*}rB zAgleUqKV{68>bE2lqh#Ii)Q9L!^pKbx>>ne$D7_*b*N-#PJ!T?=36pV@HMKmflNn9 z%9_+6o|3y9ZvGJXGNfkeMrkde>3e|%OlP>GckXClA@D8xnFJ)t~_CqP- z;#s;z@p7`L0C{2{FIWP(G7vAxl^MLn6MBWb(rCtnrFGS1xAfj#_z{-7;pjxsvj(duP+KuC$q)M35)bun} zdY-6C*o~^IG-+~S28`^Dr9+N+L+?@p7n`9b5JTd?wc|8u!Eeq5zd0BD=5g;yWiu20 z=J8DUn^|VfdA!D}p2kApfp&2{>~kYaug{E#ylZ01J0@D*t;{n1*?WBg;la1nlk%|^ zywnnRb1rc==Mr~wlDN{Or$>@`b1rW;Cwap`-lXW_;HI2|n{p0rnyQ1Fat?0l>L9Gi z5|q|Q^1eG^(W{!WX}{!dk0qC>o1ev4C4-`sR`%kwvKOZwjQP}wt+cWi+dNZ%pR5dW zG6UjNHKDy*mC!p@3B7BT&^uQNy?d3gn}bkwpIQ%as`8vhjkkMo9sPqY6o;<~jLPMh z&!LFOU_)OC6VWDxH$@TId#df7b}m-<=_q=rZi>&f&GF7L3ZD~2N9c=z&$1T;ldXf) z^`X@DAzWW#eh%UK&!Y(V8x={WHmpwL89F|RXNY+AG3SSXk5f@)Ee&!g4Klf=PlN0{ zS5x)hma?%l?roYfW+0);(*)p_vpAvz5U6wn))EWk;xzNH;{FA}zHube=EKQ_?EvrYeaQJiQ>EbFLkyajBh~TELBUy%=~v z7=x+I-O1;*e^g0Ge7vUW2uoR_!oa&Lko|MuDJGf^)J@ffXaQf^6V!Sd^MR+TsIRfS zr=YPm3Re|3XyY@z)>#Kje(*-V_XNZnxt0gwjc5U*?d0;FK_0C@z9<5Gyo#nZ&TKrQ zpv5rdJMkk0tvJF67WutE!KL~ue-Ht@!bF3>Z<}a%S7l_jF|!ExSQ90+=Z7nA!x#_a z4Gd4F zC7GXGCAl&wDxoUDnqTa@GW{bJ$EoT|P_G{IUJY!Y{kgT^+n`T-8RdkH0bSR=YqmpoT zQti3cYZMp_4W6%6d4bK9QQ*ZUN>+DoyPOu@csYHrj+?pZbBOvEooatvM~>ansq!q< z88k1cC8O{8dI&Evm6NI@m6NI@R4+{iN*ME2vQ>heWoj>036gadZBiwiFSamGQk5jy zR3(Y_gPO3%UZ*0NC;AxClF^{Rx8R>w$&$qIAJwr28cTI5A-TdFOG@I?dT0wy zq$&wXsY*i9H*}z%nJ){0e^rr?G#=>*=+epk^D0@A82;O8FOr1h=jK>K5*Hg=ASqQz zNJ>=_lK!MVOtY$92xM#_Nq1->RX>@WB(IVsiQ#WgNJ4@ane*^f>hqkw;Fw(D0S`6N zFz_f9m1parg4SiGI0W2z8s>(9Cz&X@y~c@IW~c)^+(e^5hAo}wm=mL>*a0%Kyoe86 zXijv1(XY{%n}G1^QD(aZtUJ*%CoYPLz%Qvt{xRcY!FW?f&?;y}M_Q9R(#|>3Hb)pq zG7j8mz6}Byx%8u*@gtd1DIp{Haip0a1J*ej$+$LxYxp6%3cyH4a*e(5zEH9_F-yZP zzu#KED@B`0RLWP1oHjG5zp<>6)26tmmouyEfB+H)$^%mHbBsvJoS|tm2N{|+)G{<} zCIWV7lLGAcqJ#n}ps#V}w1NgT(AOBBR?r}peu-sduW1$ubx>KLfqRt&8u(OcdWU|d zh0&)uxqst~#=4%hpIE)FXOaAVQhqt~WL25oSl27Z*7e}p@!k7!2w$u(g57m-$&AY( z{I|Z{V!*-v#;WSO*7%13%)dF6lqM%4C4w_uNFep*TBnL8#?CStq7El`)Z6{af z*rvB!z5?kj%mAaI`Z5w6ghi=B}~s*}#|Q@f)! zo04n&R3)K4RY`b8<*OD*!XGO5lBMbr?SHxc4m6f4RY=4O540lct8bTvOFBA z{hykIhUA$x!hGOyiL&P*y-LU}svFf#Jh>-v_URSTUUp@)=wk|FuSI@%dd*hV@)S+C z&(&mLOyy47`>Otd`kOf?zef*!&{P%z8MU{OK0r7^)lai-$s(GnB*l`dBqXBpLK`S4 z7F6&hTP!W@Kdis;Y>#AtjW8cLmMHsdZB&g%4}>FfB?vsyM8m+Rs;In(9V%#j!4!vp zkDGzHVc;4QC0CI+@gOtQ0Um9lQ6R&XPISzP7nouP$jDMPz!m012N?ZIuJ`ck@n*XP ztUJ*%CoYYOz>O*@H!{I^6Yd3QMMsiTe(Fd&=SbTeVI;X}0RF&y8w4_P=|{Vu75zvy zGE+ZB@Z%UWzX(|8Xe8s>2(B@*+ynrl6Un^%xt1&=ON_w!h})fv)Q@|qI&kEVnu4$= zLTjg3@<7N9ihfDhMa?h;h#* zu(Mi4B~Krd<1<$4L%5a7?81FxLc7cGngIQXSH5OXHV^ma`rSbfz2ZXgsj|QyS@CaZd3?B&$D5ayF zqoZ9w$8@8k4Xlrl{Mr;HKl^Y!3M|KL{N=7dkrvZ~?U?T-B_>kVfkHzY` zWSqvV2L_X&H?7C+RvU=M4G*fT(rP}Wrb^?zDhyU>y-F*&a-T+>we+8&dySU$XRN&c zF{kNU)&C3hcR-$e%B<{9|LsdODURx49u2CGXN0sVkV=^b1tL@7lJS0dqKJSR!3u2( zrLsbULNO}{s1>Zxrd+H80xE_Mv?&_vfPlK818qvjIv}8y=nyJ1{{P36edv^T9bxba zo6m!uj?5d87bk{>G4ukRqjEmR5R8e)lO!T&`1pf^qas?ndw`$OUlHA^5N<>S9lzGe zCZgx+41*96#pnaNEOC7l*Z*p+kEX7Vrml~EO_zxq^jB`dh$i|uil5P`Rsm4;?$|dsZ)zn4;H5$B!5KVJ&=FZzQ3SO@TIJ))?WR#L~|mnlxZ;i!DPyzyv)3C(&a8IEC!-;nn>RN{#QiLmd2_Q=cmtk{Sv3QBI+7tg zA$|x?NjiilB^hGsv>dR7KZ3fiw-dvEpoTZ z`H+Any8-6Hv%DP`vdd6>mEJ&{lXKNY<21N@3cYD@=9k_;E;NH8t=;vuMHKy|K~UsU zg{uTn1aip|6enY5;-vK?O*{&9tP8QeCssoL&S6wHr4c{-e&MiOc>!nWZiC$U0jV0% z0>b!2uaH8$_X;^la9$xMAu=A2Y`7kXxssF~KrSj>51&JKI8A19{pos4met5XSRH*{ z{c)e!bY&vvBr#c;I52M6T?{FK=SB)Lrxx3-#9`l6W=&MVr5hy$xQli zfYeGKAX!fO_&g^HX{M9w{G`v+a~+n`?z!ID6`hnumvp)1z(*|C!@z&BTz7yZSZ?fq zHCD|)qX}4_bM94mx+_D{*Zq;L&lHgvgGEIY9w8}IM?#n7o8mv^6 zCJjzF<^%$6W^lsSir<5ip{8{qXIhS&X+?&d@h;@--;JDU$qjDhM~0m7B16u!B16u2 zktau-$dfZ|(#1+H2L0`Ac}@HOpEgXWDxT!9mmxd9ls*~BYzEVda*_h*PipHQdN+^= zceE!!Yp)tFXvG_>4v@Jk&t64w0>{#LqVBaHZhP)c;4@V;J?t^1f01#B>&v5jZ(wH( z*OP5Ie0?)6!Rrf1YDt_t|YK#(4oJvr24ZI+8CIG1j@Yj z$bvHGawT)cCn8Sv_s}B?{A4Z)$D86BxmPSok{YHWOnU<*)0KgI*?3xWW#e_EY%;-I ztuF(7Lw_Y0;4f5EmQ6O8ue!bj;|)v_%r(kM?XlnQ)G~LLvU`vgL~`#hT0U3Ut`)?F z)dkha&5ckaQU;wL%S5EKUKq;+SSu4z^h5XRo#D>*G_whOxQa?<(`5A5u}*VAi_61mg`%63kPSlUgd}LL-w( zkrFW1D-x7QX>rLSpZf$-T|tqPTrI)U-E;C_eF*;;dl=sa?rWk2!2MKIp18|@1cUYg zrrPc{04B=xf$3hpTn}!Vsirs8PRZMpB~PS~$9Yva5h*QBtwJ8KMxH1_9*adt+h&@LMJt1b$aVUdKxOX**eU4}#o{hXhK6Oq#56eaWnYxIjE^siNKLiY^)qRVR}HTp%$ z_F9d8k@}GM3f)N-^#b5On`jVtvWlRe_DfAQp`QUTQPK~ldr3c-faMJ4eO;B(zadM% zNTHu~U-%O#EtYre&@Z~XX`Mese}isITeYN!ZYC)r^@09Om)hUk zGQ9wJtBD4Ix2uSx(7w}DlcX>JCd#CM>0TxU3=Nt0qS*yMNjG67dYR>!^<1PRrKxWn z1>GguVzW6&;rh9{wgl-aZB1N%Jbz!`HArsJUwPLUxY>6SIb^pmM9U{B(?;H^`Ye%n{@;z$*ax57LFRehA~ThI%i8)bkglp1>gW3nW#~KyjS3c;AtG_mTSukozDq>6LS@_d=2*Qtv~MdJlrsj~}FdbjEqT5jC}QFa2O? zd!+aI>eGvo^debmUQao_1d8LN#Z!yilZ)Kbi`)~ONw1uHy%&-kk$NA3)O!%5e*7Ty zqchGGM%2{Kz4U{n?UCNMs84T8(u?H%=Jk}*OQ1MTT0FJLJ-Ns|y~sVmne@uJ*Lxw! z5vlhfNWBL^>c?vk z(~H~_oJp^od%YKu9Fck-g4BBuq<;J$^`kS+OndEfYUf`1!P53f@9mnrJCpPxnQmTB zIlTmm6Npr z_adA*{R$GK{`3%}{^Ss({?x!YUooPlcJ8GgENzeUenx$|EJ-httIg{vFU|yt( z8Rz9j zRM*Fz6E-F3w>{H)y!zC%^zuntBqy5JE#O$9JXyJhv(~?8KLQHUZGSd=tB!Mpxt>%e zDxXl5@{u+8GLy|Kz_=`*8TGM#N#C6Ovi_Fetyt1{=qZhdCQsa7q$;&{YUCmz^X`jC z{V>bZzH+Xi?|b7rHX`+X-;nl|g^a%IV_%W_zVA)@$}I6(SE-HL^;dWTKI~yZSsr-< z4^dJ0BXn9njEcafzB*d{^*oJR9~lgW-n7o2W5WR7yjKuay=naLarNE-$Nwjen=OtG zkd9(4{rDXHh)`Y#0uo;7ZmXbmnFZSd5_I+THyT6AK=@-a`S=_H;tpnMJ;PXN0moES znWeEJ>H%S`(xU-uHar5@2*XDL14`*==jdn~9nUm6+Q9k<>Br~N5k_(&jMxYeDt|Wy z{N2NYx02CI#%aK08Q4fc3qGVb@EftZE*YnBz}~@R=uPW{N7V+R(evoKDy=gfTT`X+ zH5CS{v_7PjT)9u9&RY8Mxi9`NnWooi9KX=t?jN5+lj5kp1V@9($n^(p3Zzn|L4n9r z2p^wAK#gF9Hic4Ip+TXT6$I1@R%lZ$)&T((LkHRvjdegk-OzzHrDGisP)l?Ol^Os4 zcgj9=%3l(~;NNUM4|+N>e{cB5#LzH?F3>qD59%=lVVHsl};Ni&GC4ryeZ!w0IAyzaYgOd@1Xy{8St5%hc^k znFiBO{@fJEDNq)^4~)|w8jRBBG*}s>!DPyzzazjf8qBePcYfSdrU!D-ZIO77!x zo({Q-l0R)JKUL;qk4HQKeEtmA*N1C_?l05 zTJ#e+-DwF1y3@j!;kwhJLv*Jl4AGqyhUiX<4lx-ma$D>J6_Jlk&Ndf~6Al-i{L!d2v=#rNq=Dx zr0E6Ee1EwS6#W%PkT%?Z{wzCj)_ayG9`%IB2{LiPfn~QQ!1&iJ38TQ3mV}i(1&t>d zwJUoI8Xt(lzJkVYqcBav#29=&d<;G{E(V_?ci1P(9rlTHhkXX!VVB8L@nzrXnZ6J`wAMn=>}F`V_BL~ABj(D z8i`M95{XZ2qTo}TDEQDw!qgKSR^ILj-rB`KO|Y)a93av1=SaZ0nqc{pd*Jsh z=c{`P8n3hTuI?>pEVT5l?ki|;rjbE_WHx4eYfn0?v5{>JjRRy}Mg_8cko4hz6+@$A zP;W?+KHcj=9pS!}`VdEe+`lG9eW`GQ+f^rG`I)<=@|ipG_ZF3xd-&keo8zsHT4n(DV znclo{9<4rbM}sXF*)R-W*+W1HV45~JrIl$K+>2JGX>bEtY0_Y!tTbtG)-xv%aMy+t zzE=EZqYO2z3pvwrPiA@1|9gy9&HRvLIP!8dp1LvbGafd%meO4?=kKOMD&uj~=gM$Lb&-n>rW!dLD~^qKZD{gq&VXRD~|SN2YIeF?@Jm?W52Dkrt4 zXPmrZac$S9XLwL68+k7VCteWCMx?Zs#Iga_$|mz#&1W^tFY2!_4g8*px_+4MKV4s# z_6AC(xA2ucB7)Xj*?1i(n@lh_>01wf&|e7#I9;DPug^7W&o}R{ zLvv3jH}fn&+_#Cy-(_UJvWM+jL2Ou^+&&65B4yCku}nlt>y5EYfVDCaML!{r(XdXi z$F@!2A{BN0Fx|6VUzqg0Z(gCSbYl_ohCc(!VCV42cx_IX?*}BBjMCO6WHj=@&)l=gpbW zJww0f^1e)sevz`fSEFB~KBTjhxU(aHz;jJB2s}?k&`*24sV4L@047TM!E`U_2NSTI z!Mv|mru1*f(l1izXWbY6L`sXLU+6a&=@&)lKUM<^-81xy?vYyOFV^3ndzQ9pNfF&l zQbg(louQ*XIMNE7ZK6ToBUD6EXzyjJNm3X96J=7sbT5+vhK9_XU=Nu6U5S3uCFxYX zE(p5owZ&d{kizxrb!`dK3$-}qHca90y81U}S6izYKv@`R-aX}6Lm zFn3&;Z_2$)cbIEDt%=2Zje0CmiZsKBNHdIx^wN|Ok@`T+Qu}Y$U-{A;@LeWa2)w{V z2_xQ6$q1G5Nt5;<& z97O8J3c7xzAob&1sX5)Kzp}srH>;?(vE27+c=xJ5lk;3uLhoE9^uAR>KM1Z}t$qwN zmiC0NS|hp6oKDXCILPdUXI?S5r}Ky(k_B09y%$01`3q7{V32wSgVa-alM=MUBf=3m z9RMGxqF%Y*_5XSDs`OA6Fjon^%azc3TnW9ymCz52zyD<^U!o~T(r1w*m+jtbQqC&~ zIF8)=j@-MC+(&@i2jTsY)U1)07BnQh2vYAuka`b-)Q=yee)I<_K?fO8ZD7k%zRdTs z2~C|=>ABPSTnW9)mC$=!3BALW(2tD2yQ_2ZKCPx4$pIG0lvB@Yb z)v;pAku1_9nU6$!s!0j&>ftzYPd9Q;IC4)pa!;T4LsBEBV(&$edLM$+dl004{2=wC zzg!7=oe|XrZcve&wtIXx?{t;j5$ynT+au-pyHv-DDM#`ei)6~PoHqk;oK$(bk$b|C zd&-ed#-5}`&gI^VAoV^3srMjA{rExZNB?&v=+j128~AM%$+>)LDPN-;vkNbDo;0ZPdIW|$JV^2~e=W_2wka{12)O!%5e*7TyqaUjTooqz4 zfv;APoXe+{^8K{~%x#a9bQ$CO?<73(ZHn&F^ z5W@-`E0zJ0f7c^hG4o&N94CD~1;~98kozyg1zIlr%VKIhNx*Q$-}nR5KSQOAlYNAhPq zvK6!Zx6W}=bq}Uo&hOp}Nx4Y94?*gSE=YaJ1!=?m*ROefl>SP}fyY?NS57VEn;mo8 zV`0bNnd)3jIg+^+$&_a~f3rue-U}(0Gr9Ld61H8=?*}PCi;So?aJh=)Tt2mw&(;nw zw>?shzlZ5qG37``ERrdw-2ds`y^wM_mwPWHN2H(sg)z21mw z1AnTb-o|OGd3oYh`bfN#@sb3uN0c&e-k-IqD)6ywkJ2EnQ#6QJ8c1Ga5l>ngLcW|; zS84|;I!H4q5#3BmM4Cy7NIhqt)Ny{;-9c2ot}5kA0pROQHm?BVa^^dR=kji9d=*K2 z%Dgg^@w7<$Zi;=s8edHksgLp#Y2P*RRi)3xz9RK~znu2Xl=ddIvFqbQe+eFVxQPaV z$EhfMV}RC|_oeS3@F7nKqQ1siD+?NrRAK%L1$gpb>XE#l0NlqsnGdYn>LjsatKT}d zfEeg)oY4T+jkOYEPf#QKo3SSFSQCvx#t9~B0WqKXSL^puZ3ymboI0(bfyr`l9g-k$s&mbbPQ`CF}_yhDG>_iex*H`)FL_-d0KP=K#9*?|T4 zcP2Zi0Pmm`*V{O_0M9nrAqDsVlO0-spJB4Y3h;cB9bSN6YO*I4;MFF3ashs`$&M(% z|6#Ht3-E_bc2rVgaJSB4-LC+{mt2Dbp-Zkdfv_c7K**Ab1cWQ;2!txp0>YGN0U=7X zfbb+*Kxh&zAS{U%5Rya-2uGp?yinHrTiQ5$%yi8{cACTas;ZlVtGl_qKf*PEyV zge;*F2yYU$UjjJ9yf4p@;90s{_cRvzZa>g*|E#~|JNE;3J+z=rSNVQ{{FCc*X71NB zJG{`v0hTBf=AkgzNHZJrs-fo2dsH&ou8nyUhbxu*h^EAlrIH`@7L5xxGtB)=Rer6% zJ2d96>si-Blb-#hcw}9VA7MWwcgRQ{0=~;ci+~$c6waKqu8E4>%pYPZ9bjRiF(3v@ z^PPfLG(XnO{4>l}3-}HbjRG-Pnr~&yk9IRZ-)yykZ!^&_ux`GcF+c3hzgTUY8*O3Z zgHZ&$%0z7-%=F7&9-b71-VT-boca4E{TSFNV zlUHuBwchAC5s0lF1EhCYH)t)>5#;<1{HBQ(0dF=@2l!9s#3EqJL>*vsHL1ze z)mF|`xy0AO@lV(W;$N7cv<@>rT0mUvZLB>O9DPg9w5f0HoNsOO?Nvrl8;CchZ|#Cs z95i`3IvsQ*H|R)a&=Egq`BB%^>6ppCwsMQs2?ISqKF_y$UC+?Ep0yXt@7hnw|H1zg zHO8C{$NoumrZ*}h>v|E)-$G}OE%%)f5Z!=SJyXux>v{?8$6GMVF(@~m{G7brks#j+ zfe&IYlekTZpP`Nunv4k~wyEYYHTCz`Ahwub_rP3Z$X%2XInM#37x(nyfc{p*n+Nmpu8CJ-5Q4MabC2#pmmWIT?YsmLD92upmsRQIyQmFt7y7>6MJc6 z&OH6NLA3B>*Ox#2_6BywK)4Ny1_EM9t&upHN_nR^nNB|5{q$UVh@DfLK+dfeFM?;O zu6)q@*#0aN2Rf@T;Y~Xx`~dR7K^-8^)F+g1>O3FfDeBByEyN+pM{q_`?Z`wEhwmf&KGBYUC6EoUWoUh|e4juQ6L4;FTsC1L9(7t78sd7ZriH7!+y! zPgDfrrEEU~>kg+oyf}QO*=hklZ=%J(y2CAV_yNAet%6ps(YY9imvWH{tUH|U zBI7W1Ap0}G%gv8Pz`Db2bNG8v5r~VW!)lT%)40MurMn?~01u%s<#vI>6N?8Utc5 zOmbS$e6o9l_m47LE#L|hjRG+krX;OsKG{9O{9$IR4P0)bVIT&>1fmtqC%Z?_RvRyi zw&-=PiYRQ_Z#C66kO5HjDPR;>kS!!2qfMC*Y>_b8LIT!pbuy)qKu z**X!3U%G`v6V@_YNEnN08Ku|jEG=ORi8i^AOGY3AblYngQd9HI6t6-*v@rKPbz80kz7Z)33mT|%#kBc z2~JwqBOB>!pC|u&e!81}mFpnaJyRPg?Ks8Q(vDMPGwnFV{G}bISTP%woZI!c{PzbH zK5YNMysR$3IVuX(N9*ZP5jbR`5#aEIzO-Iqile}#6BJ$c8#w$+bGQS1xY=3+#KqF# zPC@I0s0hSMi64lIp)akM#lFD0!|Bw*;UAmBE#M=}))){M%RyV_@N=Rf5HIB#6NrnU zFRfR^zQDS}>6FG{>Od|=fsZs>i-C2A+vad9DgyDR^sZgdS`rn3b??%3BlT_s?{+m? z!+5n{6aisN!orDviCVzA6X_b1I$;z=Cl=$xk!rqQ{`|UpNz<@^ z{7tK>*FtClcQ?@>o*WZJjCgVs0U0s#Ehk81p3B+*jL~$N@_r^`D|T#mQ-QH9Fxagt z^F^*}fN{VsQ>U9P77N)j1V&q3)(vlKjq?35<9!IY(*Z$QTm|l-qH@v<6|@!~YKp+4 z4htgL#65p0jmJMZ2t#jLgDRC_!r}P;IJbj&(g8AdX{%$N92FIT&oI$^AR~spakzle zL0h>&TQ=x1HbM)?Nac%L;QFZPD)(YFvf7Ld0oR&nKJYCn>T8@nUeI`d)B}Dv>a8kh zTpsm+pO1Pg3mV^xdcYq8L;9TNZUeMT8h4RlQ0UsRo z&Mau`6ZL?Pi+X1iG@caofJa5WWd)5Bq8_jn^-eEnERK4>=SRKM3L49z-f>#v)PS7U zec65Updk6a{>o|xyjevunbx01gXtr}O-sD7KBbz%6=Ogqj`aJ&5*pN~=mBxH()0G) zYJb>L+5~>hM2mq$FQ(33N(1|u@3i7Qt4?2Id|E*Rmw8c;HY-o1OoNq2w*1duT5kCx zfIY0x#;eMTkNQeoBXQ;4TNo*=4O(K=x6%|}VzT`U@b^u2Kmq=($qp>QyJ)5LHV!Jl zb4_+|0e*(b4k^GRCOfnMpJlSc3h*0Ec6b54&}2_4z+W)glZ%6N{v4^ly^SXoU<#__ zz-KJwX}P#{`B7!WNW1c(-p_~lL?2o=(i zLWL0#0U<;>0wXu+1CUp1&=*=A>{oz4sxq+-yuw5s;Ezny1m0$%4)B3Wh}Z%?!9*S4 zu_kH)pJ$>DaHWZwz<)PU2Y8W*TEH5EDP0|3Hmt?0wdM08@T`M_wuIxm-J#w~%jb|0KcCrhuA>bob6clM49TkBmOi-k?$P@>GR9dj; z9o$bHe7ic>*EnT$LF3&joGBkS-cwF9>!-~rkg+t&KODDV>H29LM$D-4FXEN`)}ALi z=}T*j-c&Ej1c6_nGGP_C-b5YXr%ki~c%6wlz`vMi0q~(3o6K0?p(a`Y95PV{c&dpO z0AFvS4)8n^wSaF`kx;Vc)b-O&*>P^}UE-9Ic?l)Gjr|MHNpItT!jsb5II!@nl&1tw zOZiH>=cTuCNa2a;Z5W_lZk6`q>j#^HtMrd;YF2PS$OPfnKl!!=b4^tUVt@N!FI z3;1>ub%38U(E{L)P1FJ2WugVZ*_N{o@Ng3?0G?=~4)AmnEdY{qDK8*VXO@YNs)5g& zp(b!foY6q$m~1rwnQn4v_KV~ya_g*fX(iVEV!>!XHlI3wQW$$Bt1!JzRn_tY|1QO! z)L(hVezPnS^JW)FzN$wgqMMh{`1(Zky?P2H-;3&Nmg3Zd^zJoxNrA)XJSjvekT!p!6#Zx*Tv>LF+D+%IOYR z_iWiz#{5E3S&WG(F2g+;$LZ*U(F-75C8o`Zefrgu((=AkjS^SxEs?&%gG=i)vHh}R zR{rt8W!nC;llugt?*p5zb|0m`LNBlp3+CD--kwBzZ4&MM)d)T1mdf|b+-2>dz20Os zy8!qB70EgJ+9mz_&s{5j7I8}dY=Zt`>;b$f-vco;$sc^-**+Skh=63v4ePnvUmbsP z3_)Ngr# z@Kek7c@7R7$a8biM)nq$e5ovO_5=3p)2ZYub)tAxALUGm>U2M*k$ ztg4H8(~CQXrcO*j8}qboi;BJ&F#jS`=>WfJqA?%_!+txhXg=BZ$NVz0)dGIiM590qmgZXpt!RF< zoA>9Ktv2w>CK?7}ur%M!m`_%ox2uiIqb+RwAc}ySP1FYBdbjs&7yts~DiX**Q|1C& zaUzx zfP1T`thV&+pR{8q*D-xXA??`8bxa?kr5)S3j_G6TwBtywWBTNMM`hqa`WxQ4=VqPhxdax!hnehUuXNNsT1{A5&~RdqEXnCaj|r`Q_y;AR0QH;P^5KvR0QIs5CyC|oE9JqA8NK* zz*m}RF%TC^KU(JS+oB>67lR_L|BQ-2yp$_6VBO)g@Nt+rkUbdSDl@+bSa-N>4!@Ar=ymit{e5xpG{>5c*K!G zSV?Yd+<~telW8KW(WaeAICz<(CVBLHxV}7)o z`M;R?Ht;YrKMbs!Z)eOW*N!vQ#=g-OHcn6xg-!cJQ*8rbA5|X&qrie(Kmr-98(*w2ad0nU%RxH@t(|SqML@ii_<^_>`qJ7r_661*PNxgi#coSd0^YQzv-cn0EA}g(q+z zd*;BpCn+6QsO|r>K%2npO*Duny%y*oBR)EcfQ*=VkOGO!^R1ckekNlp(va-Pq5@-E zV6dBY=8Ig{0ONpNrcO6oEEaNI1B|x1tQ+3e(aQG%<9!JDR}&2b`;HF%%Skg-&^lSA z@&z5>D<&w?I{TQ=J@cs@{P?*I+5!H}M591PF8%13AI~vAI=~m3XcUMup)akEM@8T- zOwp7}0tiR>=e!#F4I@3DL;%osSs`^-v9{yld1Hx3P28JrR=mkDqgAc>f zit#6RyTt!rRz+=KJ^r@EPk^!@1BtF2v|Z4`B~b+8Pf(;4y-QY<)VmS9yVD#V1lAoM zF^6BI*^w)E;45mG9VuunLaRLN>Emcth7csy(|3k)qbafb4qhG_RU&AS@$%dY1wcL^`znDcO;87`GDRSCe zI8B$CJX9V)fa?S00lAo(WA4K^rf3ruLlYa^Bd1Nw*ztiQCwi`YutS?!i+;EKu%1<) z-=bkFTKSQ2YiQw=uM3v{&OK#Egw;pJ36C%;p^u>w5&(`?Lht1%y6`n@>{ZnpfPU=$ zMv(jAk$Vr2dnb^0^8u%Q{9Pnkvh@F z39f0o-btmdwNuxUD<)ifj`Ac62@r4VceHhPTN!s-xXUf8*aLEDzg2gyX;iH=s^r2d zjcV$<9cd0a1jXisEJ;9ulB+0S%#`h8#0nZs1Nu8REr^lrq^>(lX+XmS6#W`b{Tfb* zPOi0ZBGyGY3H^v+^t-1u{X*1q7F>ZuhnJ0%sNHylLm0A7HYzHZ5)2 zY`RlDC|A++!2})A%AKpcPCkWLpeQo?qo_>RQsOF@HgzRy5)f;s`mABHS4~#et28ho zOEM69+3Q-2p(G~{Ki)n%4k~#r&Qq<=>aRQj174=0G95z&tuI7H;1?$-dOy+;I=K-# znGrg<5jx3Jb9gpte$Be@aMr&jOK zEPX_OlQ$JP@KIGrUv}gOo+T!Hkt7PfI0_x>efldKWZs5|rWRmotDV}KVmeY=t<=^O zizl_!No`HB)KgnSsjZnl+a_?-!gw0;^DfwxRhoZROQJ|AYb27x$UhS4c# zT^1FAgd?>JB$m*Z7T#qpWl}m@xW>q(t(N&QW`4AQ?=evah(D4*AkGAHw63wfEg&=p zMOxpFia^Mh_<=+k`c9sQb}rbq1&i-;?Fhv2a)fq4i-HkFAdUw`S~x0-Kzx=v?vuU8 z^EGNl@9Idmj?{{z9vE$PDI0I=rRu~=OFb7>D z*?sM%>q<4zZSAg1__do(MEA^z)aI6Pu`_{w)0uEmz##lQtD-C4v38D{ifK&3@v|z* zgK15<`IMUW8WX(6Bx(DnkTzE?S?k?&CfN6N_FBzZVRhC7zE4H{@@L4W$k~eahodSG zLYmz;K46an=Jx$Ip73)lS2{OGojJ>8P%Ohsvm4c2Mwmc$b_6WL_=Q_wJQ*a*c`{5j z^SZ0ndFifIXXb#t=Oh8&n!sS*&uHS$UR8hG$4;8dlMn&M8Of(F9vx$a)zcSG< z@b@Mf0d7`N7?jqnrZ@zA`qM*E%H|311QU$_|4Bu`9IX*k90oquL_@%36Z+Cx85M!& zOi=W5C1tpiW4L1sA7R0EfWJ1;C=kLVVL;d_gY6Wwo@ow`0%1q`0-+}Kr3E{(B@7%h za~&Ymh!${p)B?hiXiZL03yOBJ$OnN`V(DtjSUcPlTfm!4Gzf$$xf%k(Q!q#CX%@gB z5UQjv5S~I`T2LjI6TpRLt_6fD(E_fCTENI!vcZ-XRvXsp6t#^aW};jl0l%jdm7kgg zLxEgs9SvsINmt+j>TKB+3LCT ztXQ)90E8U5E(gLx`M?JZ5uyepzETZLfHDA(=t@e!gePjidZg)o+Oe8Cyx!&kcxoAt zxg>=PWG>xP)5^PcoO1C3oAFH`^z70&bHc?pax%P&+*{W#&XE(MY$YDsUsvD&v6}*y z?;N0BWIr=lkYy`ezH@}~@jf7D1+pQygWv$`ktyH<2bheh4;-Of%>-j~u_q-f`94fcBhlsXuY!agQ!46O3b&>Md`*q?)xWQ+-Rm zCpr`CWA|1q)NM#QCYfQ|lyr34jQzh>)o-!B9oBeKcyqnJxXL;iEHaA$_?Bg#JUCIT>^t|tv3!XCP@B8I~ zr>r9XDCL5`ta|tT+VD&1mvp`SsV$`)AahvmBz-TgYgm}J=EkO*Ew*W7w*weG=(1ze z%@#|%Y}WwUL6hAtU^L%lGtHa7Qc3!Oon)5ETj+BN+Re6ZjhwE%F1B z9X!4}&SMK_%ZH;=be^u;BHh>`$9E^)d+Vmi5wWm1Y47<+Zuu<{AQ z7W;$}4KUj3vSrxK7AG7jULgCmTeEfR%|Bd8Iz)fTKkevmQoMI+h2CYA+5|q!s85QQ z>}(6?v4yi+;XGZ*@!d%`w#f0_N%wim+y{-pA>gM}RBqS}6}0wzMo6jD1#o{84FeB0 z(Gc(rCK?8=Gtmfey^6x1wBBioL%>^1Gz|Q+iAI38O)y7m;F-jbyiJ_)(aG`AF+P52 z#C3q@njfP;!V}hj#8yVyDQNvZ_5~7)^aT<~=u0bxmF(1~VYLYBV-~<5kX)DMTNddJ zrq}|$-n<{CF?=chUb)`g zxb6ULWFKa(9LdwVh2T zsM}`S*%+k0x8Zg+CJ)xkZ^NBcOj0u2bpH^e6y7%7S;Z*TyRCOtvCCSp4N1o&Gi;lZ zj#1|6HYFXSO!jR}I(At^wkb1yLA}3Dr$^a>+5|pYMQmr&X1d7!B#WCxG$j`P^US#|_zZ^3!GZi{qdiyYsbbbno$ z`=v410kWcGikCH>Wm1Y47<+Zu)$tX9EjGX;8ep{5WiPCoElxO6yue#5z^U&bdh=gY z8$Z=w@=rVZn-uSRwafd>jVAC9CQ6Ez>}(6?v4yi+;XGZ*@!d%`w#f0_N%t(>*?o-d z@D2eFR8hH;I8@MjXH*1!%tXV$i%m2Hywya*z&_nClPxgdG!=zGY3*c+L%?U7Xc#!( zL?gfj6U@;%*%XI?OH4EbeC>q3wB8sMfp43j=qo;DxRYbJV+=1cqB_9aOf(9FFqxk~ z*ebc}6ts>vhev_1BYlBT6Z+DE9jP$j%gtN|2sNSwJSS=aVM(+mr>F%*Pq4@bfmCAY zYRgz#V2Ul^f0<|y2vxG-352I$j@Ho@z#tH+q%RPjLSI@?C37735;NBVLX~I%&x=~X z$Xc>-pB7dd*6I|sjUr~EoQ;4#SBlDAbubi^yYJwg%(FJ|Fcp2IYRk(ACR*G*%0jUb6~#C2~)ra4lo&0A2>p}nhD0} z#`SvRx&wp~-Mgm=5&8sxCeiZ*fHpy?6`F*iR%qfsR%jDXm*)sn%rcGQ?|6bBW)1PoH&+weI8CYc)Ba7!DLl*~5WxWy=iw@tUSF-rAr>n&~UvKDMZ(lN;l z+oq&rlzF;MNyjLYeOr@`T^5mT%8Xx7&($U9J+`1WffuNV=Loc!F0wHUq{#34n}2x9 z%(?Gx4C5(t@gHSl7=M}QQZ7KI=Y79i@RT`!-!B(DWfl2HDHr@@)w}Q4hF?m5s_W$~ zwr-v&FH6rZX#dINr&MyZHZ6;eLbu53ZAd)&@?A2v2$Mb=$Xkog^#9$U z>Oa-r9U2SP^{(rsd83MFmSXvL(wFwF>p^y_9@|xqo!;o*f9~l6{d4vbB@X^gm1d~Y z{B=Fvg~q2s+8(LDvUvlXqaxX$It9E}l${LTJIYQ1?-OMs;KxK+6a2U+I~M%-C_5T_ zgv!J{;KiQ~o|o=T9#xB|uo!l0X?4tg3yA%oIQa-|Mu01E>|`)piR>gWT#0N13|As+ zg5gSJ$AaNXWJiOErLS?uC97#X)dFh+2~7H)Jb73ngoVYDkCB4bdb2+Qtl2-)+uu#q zAECcFAH2?V@w5L-`BSZ_7&}=F<4idK_+|emRE#W5z|&RK*I3?L(0I63mgvnJkU-(|HV!Pn8;zJY95J9gp8zs;Ut{U= zf)OL_$q9s-XR?(iJwL@t8D6~Kych;U7G_qOYE=Jcv zAlZ`Z6tWaUO#Vh_sv-Jh9H4Q%!s1J24fXb+=r%A$vJfAiA0tUKv(OKJnGSn{4L=Or z&xTu^vNQ}!``U1ey=@ApVTMM5vsE-*zJxGu?&*y=2#9s(#tQv@Z*}-~)n`<>GzF3- z*W+aM3l7|2@9T9MKMu+OKwO)uuiMtKC(k^(^7ZBVD;Lth^Gq}he7lPJ8q0bL8lQ>6 zRmJ~k<9#~@yHeeri*(2iLtbFL27w<_QF-D_KSZ6{YGLdBW@{8!x0QZ4Keg4y)<@0O zVqo1?`icD1)(E!f*w-5+bJxrAbj!LduaNy%=aiDQZ&-xxYa~4Q0H`kJTngf)e z=9Y-^N*SV3m4v9LxuJw8Pl~tF*T}?+onzHrj28*PWbWUYn_HyaD`nzMRg!qWp{(C% z?6-mcts*ImRi1;->ai_)m?~w6!hvtA#YhyAP3Brs7(bkf8(t|xRH~8?^#>hl2TiCH z2Jqo35~9XEQ9quFhpAGAC>;2+T8u;?=`+_7qFCU!m@luCAu3f#hu zEGt*()z6RY>Z}F)jf(O&{&f7Zb!-E_W}*>bv~_0V5y^?*8ts4qDGwk9On=|vbluAhn@W(qHU1?`dq-n1Xy?$c7hJn9QQT{S0wbjDb zO=fE`ux=~83`%Xav9-x;jRNbo(#xRK)(Ez4G+T>+bzAAnTB$8#`A2F?HWM0Pm>1cZ zV8|5NDPRH-*~wrW64^=MU&Jv+zz`~WO)x=->{wqYZ%`ZGjj>OzlDz7fT1hUkQL z@haIx;=}6b$MrW(3nZViP?AOQb9(5iq%)Flb?>}cC*oe8Vb`exJV-@hD$v^P5>o`e zKt=gElaA2JjnFF65n8zs(wUTw&}M}FEvzvh!<62(ZO{`KqU2dY_=M_3eDFgr9ydAko;jWgKM; zuW`!ig2qf6BPl{ZM#;$P?u^7h=UJe5Q$98U#~dV&a-OChqFV9;KHUOp1D9JsOM8k{ z+8{7Fr2#Lf>%sROW^WkySJRVQbYCCz`-&=+zJW2&(@1u4W+}gvec27XhjHHmK2b$| zjU}rK8j-Q&{E4YK+OM~IH;sD8>4r5S zpJ&N!0SR6192#I!T)NUS8VIx01QTS?q_srjD>cCc6f|k!eyItrhmv025(>VS_um>- z-R-uyy;%RLPEiEny*wrW;(F;`+j6(Ta@PiuF6j#-n9!HjQp;Q$NQBZCSPwS6lTCvi z$pt%N!9K-;9Rb!O8?nf~Zjp@ual8y;q@ZY+h zQ5nf%ogShLq?D#NmX++!DG(4_w1Omtwg^aKqOuQqg$DZ;{jJuJDJeZ!oo1b`deUH$ zm(%zV{kL`J>#v-}Ht4$i(YRE18}nRsOwJ6OSM;CJ=uclEd9Kz6M9T?|V)(9=PI6j2 zM-7upc@|1KqoGk!1pT4~+;mkqsYu}Gul%Jp-thUFkjC#+DF2eYoZXg5O${cTEP$S~ z^-yK@-P$0>m5>Hm7N39+TIsz)&wBr=zj634sf7Bd)O`jDM^JxbWn;DY?pZ<(RsXDo zf!w=`+>iGM9q(`YTRw@dt_2aWTFj7%sh5d}R_QV~L#B%!LPRoWHrC#}LXY9|wtbD>I4vgxsKy+US=bj45B&de>juMydEW$ylXp08P%@my3hp08P%@my3hp07z>(zr@J{!;WDSo8b?S{UrtF~rsgH0mO=~Z^JZ}LxrS;1nO)Ql^o1jezmGWgGr%g_R zVOlKb;sKBo7=~$c$%$dwWN_l;KPLmM(54o#LYuQyu;Po9RKt>7sRN;??D)Uq=FJBy z3x}*YK3-e7Ab>>6+!u9PB-+#wuyO`byzDeT{?r$BI~9+1;H@UwLT zG;a3>5d1Y705UTx515VrrkB<74tZ}ugB4jWsDT&TmG~I&(f_AhGny6b4ho^73ZxQut=O{eV%i2jr5lvHifRg4Z%nl~zp%^+d)`p0lDy01RmAOSi9^R|&R92l8wXFSlX-2?o7v)6+`CCw zK!)F9zo;~D^rTK^-(2&n)>|sW-Hq7H0|&7s^sf3zd#& zNx+jU^){Z9MB|6ckmAoZ`QpxKWr`cES82dvOSEXcInn^+1Q!(jM8F(cLc^LQ#EjDF zMCm*onsLk50>D_$aKnQa*4=51wO6f5cC>=jJO6!MxPPO+ay|s!VxkuCHWMX(GvW=E zzGLZ6+O2$3%{#EumEj`d!TOtt?|z9&vKYNYAB*s;B(q*EOH`88|3DpxCoh3AY(792 zsvzaTO5DyUN%zrZEN)m;WGsp_V^O5u;@RrsdyKE)Pbs|XKLeHWD3cL>twD=uXLX-2 zM8t@9*$Hu&M?{4_5E3o_Y)C}jsduRdPtu28a^baE;hjaD4M76k5L3XP!l6p zN1Imi1n)vi&B#WD(fN@dX-e?8>Pq7c=KctIBBDyq$MiH!PhmCF1YV>fd2h9+pz&$T zSrZ8RV*mVc8W1D14fp_!zOqLHYD5nRHI+Rm`LI#a1YWM9${r2)65*Ry8nALEGsF?|eL+=1&ky0i zPruxWmYXNEzitIpPVul^;8Q3M((M>b-l2KK9&MHm`7@J7u!aQR>(+dd>&JW~57CwO z4YuMYH&I?iZcx%IM;TnZSIiE-vnRq>e_d)9Y&5W~oRgoD-q~y0Yjz~#xZ5)yM zahRHa(M* z;q>bHyI9K${^WZ6?3LU~9He)Fm5{i|ZY3nH+D%gWszzpvHMud?tjvtDW@ToK^hK7; z7}SjjPE?E` zV+9ijXH^Ffuw_si3=vxh(L*H6m;2jDM*n10P`>9)`j5A};Uhnp z>H$xWdeaIT<53TIPSo3>pz-FY2Ygr5>n~_r81;akhH+@{^>!?1+!pl=+V;)Tor)e#EBY&UyZs99AN6)FXgnzD z0UsIlb}48)D(V6Eje7U9Yd20PGA>j5V%s)tGpCQRow>H3c~X$fs|%wRkg1lriT}1c z`H3NB7W3rpO_>$C6pEGq%ify@*i}?{{I9!{&cY%H3WTtQ#8D9?1P~P^jwo>(`@MeM zbRZ$PZrtbVd8_yslAcw^D!7d9XL*_AEj?@q9XRc)56M)G5hU@8=*)tkX#?v|Ka zD)lCJ)?)S>HSBf0y-b(0(CxX?#_hS&#_hS&M*5Y#4^R9_3BgH!>w*vI&ro`K5FQa4 zmq%UEeVOhL@RZpMH^_)hx$`>|n8!F$A($mU0y0%9-4b zKU&jcwH-i~0Z&$Hpt0uUf&%FyUHVK!)k;$}1f;bDD5^R$%5_tPDbgDd14^s9wWD7N zAducl6MVZbs3|Nv(hkTxkbwTj8b*TwC`m$3luOe0@Lb(MPO^<8`D*o8ty*m&$!H;h z7TIRfr$yFjdp0OpZ*D(rFgzCTjY9IO?mZ%cGT8vTX^A33z>7_B6ZjFO%KmgLJ?9Cg_3}o63G)_J_i4j8V?2;PY^hA*)MrgU-&O<}M&nQ(U zCIw8&^hPgHB#9X!bRk-Rw8~BcDoOK9@2N=QYfBx%=lt!=V^pfJh%|vTl*GMq&ET3$ z;;z@N{eaK%?e+-pSw@WlU!YX@Rm4a^={OS{0j@J@6nK+SBf#mOqtj8~9HT~nJ1bRn zO6fgO5csIin>yeVjT!;I$f!}^QlmzIuQF;Bc&t$)z}Fd-e6N)TwoPE4ZIM_8B$UC& zb|$dxG_cDE?59m#8@P=HXBm)a%9|U2gtF?C(!W^SHjrq_<^h~*p0|N97|Di~2BS$Z z4mInS1M9(PW`fa7gRz`oTyN@{z(<-x%Yj5d);A#ThhQ`dN@rQyCXfiolj6XCuwXQS zq?SklN23&w)DkJ+F;NN_1Dm{OD-CRm!2Y8}Vi;HtY|8?>+yq-dqAA}g0k4fxz-bno zVIXk}QEXWh4>Z9RkkCmFz&*|L7BG6Ad{dcvK8ELYH^B1Mw|bJ62HovUxfN zyho{U2MN5jIu&k7uauRQG6^O95RD(0|6@R$mLtwZ6BPEf2#f(qp3pORc@zVZJ*7Gd z$45CJ0WIbH?n_7qEdawnLL1(@O?Zi7;C$=?N1z%OsN*G2@&W}d$3V3R)LvQ@nv*F+ zEO0(hf%8!boR2`@x_OIhGG0JDtnz{q{%)=W!s)6PO1LY%0P*^7I9KrgFOdNKmhPiK z5<&82i68#JWzu~~#)T!B9A3g`GQ)$uOb#!}7yqMNl z{Pg(+w%)CCuwp5pR@9-HwWJdCogA*F^_(^IOSX4Qe7a@H6FDHE+N9MFgV_V+`V2^* zWV6EB7>{8LBOaLtG0=@7`Fd;oeEFKpYlHVH;kyMHz4B5HuTm(lE%WM*m+NRnMMtKJ zl9A8k7K#}5ZRIA~mJdGoU}Mhy3&N*-cZ=dfG|@65RFM1`>@@jsl}2T>zO)rChgg(P?H?%H$?q##S3LHQuT2|EGmv2#D`nOOo$5 z|K7)qH|~Ah&i6J7coxrX!#9fi&#jAe^XMJg%XGQ*eBgqDA{zS|hXe6KB?&okQ@UF} z*JwZCmlNO{l-og$Y5OgZqgpj#*Ihbs&a}P5Qx^ze!W%pNUb5@JmVY>;>~Y>X(@t&d z+7T70of&KA`MM2nx!;1sFbpE#?}W&mdrR$DA=W7H@RUxNC!d3LA?wt;Uq zY7~fjRa;7H&4)JdO-79ZFE*+T{J2r0!0U`^1Me_u6c~e%+}TZo(Igmk&zt7?v&_vV z5XXX>&4SVgq9Bk_tOP0jGztOT0jC6dT3d=NCi1S0P(vLq=f4t2*mG7kkU=&TnmWbl^`Wt7LC3A9n1MU zX8yk0IvoREs8r~5te}MLa$E;ucqK>)+eHv~i)xfhq%m-GB6;x9TP&LXM>Wt%<~}^} z#tmr&#D|SC)Q1-RD3j=84Sos*#La5tdh2SUiwPh{GGJX-`sUiyt~R=^v%$6123?uo z6-d!E-BIIl0aH=>KVDG>)^b`^2oLB;wm@}t8# z`D$AWn!wYP8VG9veVk!oZUPyUQjT7K9OZz-u9WkE!?UBc(;elQCdan}=dTVQ^Hw#U z9Nzx1c>cTs3C!Fjn73#3m*g_yh~drd8F5i?PM>w)UF~j)kHF5IMt7?9k?F412W(2U zK9FovciMmsaqif|#w7U?lU5vDrZ`({GJ5RE{BQ3+By zNCPHV)+dxxM&2$q^>qxkxhZwAiqNU#>)8*CC%r1|yH(ovyIc=sr`XxG4IDSB1$>85?N{lT z9;2U2rC9XoT2J+-;1232GLF?=SeC~Ga11P9!UN~ z;#-r3ty5h?(dDDHLUa3{EytU{V~t9#B2$q>@eTR0Jys{gS^DY5_IsVQxAtT;i9mKq zfsEK)_owdxm{e{5p`ZF50E+Kay8a#jxQn%Jd)@;e`sGCcr#9y8IWXBr?ftHPcs?--C8n9plXcx#5#O!h(GbZD}2O{nqV!E#h(Zet>XQdxz7=WaCk;>F?5jus3PX8@Q%lqifseY_)CyZ!)S4 z{E||G^2+USwNLpgCYmg0bbyMmGJ|QJJM*nqmA9MfWFh=fB9ip}K@HAT`k7g$KbD9j zT6WR3^JP=7p&7BHlX8V*@HVo>$car&Ur=c3Y7{! zpAQ;jt%JgYEm7Xstw^T{rqD{3G9U`3$?f&sk|l$NNM&9-ZtH zyjwHJO(n8~tyFDeM|CsIF9EZ3UUpGROg33**l@A5rZmtQOLN=v9JO>eTS>|cq_~ID z^$di2g4S6Ec5KhoE>S5mP?m707@f%hp>gC(Ityzpp!TXyh?5(WebNk$xm23Ea=fgA zBR1tuQB@J254jm48lf2EQls|fN18z4i|i5jL8n5|85)J^rikFkd_fW33kAgKe)+`! znEqr>8mB{Ck)}Xw2u=Mz?~NuXhS0;t<*ip*&R?&xB(WM3vorlknoye@@uxdQo5k6$ zX!3sFc9}NthenM6f20(fDCM7+X!7(89iSp?qF|cmHqm=jvN^v1HV1yh`R7!SXr~~TC!;&R+aClcX3w7z1=kFGzgv=gt;=n!l4Qbhp-__EYu`s$ zczhRu^F0L4cM!O4`Wt#rL*lX7z8xYis?(dTw6|PoZ#;%M2Z#^1@MUQPE>@TL`jY;fh39y zj68=z9C5T_@fGVT76Y-SVtb8Wbwixdq<47{L3j#g(Og@FNX<%~5{W7aBhSGM?kRWC zp)fhB!fevB)(8rZVk7TWtnr@xR6|T~Gq>ya(s<*htTy3xy^pfe*o#q0dyACz<}B2t z{zpshHt=~yjR0R{R2z7dQ6tRSA6tgDfepKFlf0G`3+N+k4B(?xc{t6Y9aXsvWVy=I zP9DsRa&Gi17i-iyh(>fq$ubCJ5( zLsv#DcRt3~FRIq*;qKmKKgF*oUEb1IG1<3LE@$9=pmpW_9I5NPqe$&tDplT{5h)oj zs6T2|7?G+5WOwWV#Ha z7t0Q#-=0(#ixFFm()7r>rt}7`+n4Fd<3>Ayw1A&8stNq6QOO$R6@_ILO*d<^+|7g^ zYTeIFb#gP|wnQZ9{Ue$coMbYut-c}=Nje(Uj^3f4T?R2DmYYAvsFu~HZAF+r6i-!p zc0*{%;xK>UFkqI}$pYus4b&i-S&>qaIO1ESt6*X0gcS=Q}n2N;6U}ZM~Zr zrD4k<>1#>@ol%YPFGpV!UVSpiTC{5Wesg zv5fR&9|WN)SFG_+;~lH@1LhLOySI;drKmcXgN?jauFWambDokE?#`5b5ZSD-=j)7 z|8ADlx>FA&WRoiOCS}BOix8&@&LX-9SyBD3FGACmo2#GO>q^skN~ZQ7Rf=^&zK()A zw61)WEId*Nhj(SOP?zpK*hrJiod87WQ5Fs$9fXC$$Bt&mbP2H+%TC=r)yW>ZGGe*i z9P9CUdgI8jXHa~hMO0{sX4uW)+FG}LXd-o;r(tR_bCybF_Ix@+G_zo*BFWc-UZhKm zy;wHJcj@h+eQ)6g)(CJvqeg+xP%8WuWu%}qW`ZNYe>Q3q_(r2ffaiRRPDg?NrBu}e zrOQol1bFMWtu62lqeg&3-!Unlsd&D#+0_OvGHMuzuhp+`C{3DO!$3@~1Sv&3k~hv_ z$8_to2|UoKYFwKh1szjNY9lZrD#X8jAIAGFMAsB0JCEmux>}o>|pq%Es)`t zB?frDscQkxH3yafF*!KUDkw4h(gTp;mmYv~ZP;7D=y@`qQqRZmyzb^$#?3MBX0Udw zz|-%Wr(-}IsRSuqYW|M_AEVhUFTI^8DBxh#ni4+BgDybaEn9a>eY}(}3xRcqmu4JZ znsaz*#^I$ohnLzoK41#NcZD~R2@ivr8fd?s72~s*Z3IYj2CFrBDNKI(zXE=L- z-)80wJz2-$VEqhY;A3pXNseD_t@?!#m8B8wt=L!r0iQ*__nTv9NeT`D|C$BvMvY1qDDZSSQ z?{LPEoRYTQj;RuAl_k_t;2X@_Qzi-uG4Ym?9vBlM1)gPwO_`q5YY}8uODw{n1V=ldi`*5z%y9Mw1tl7pkYfd|*sYvL#>yc!^QVfY;QPkCB4X zOTW)aVHxmPrNZ(tQc(KT4@?kv-mMi?2~zr?va&A$Z}?GN(B}_sjGN7EU~IVV_K#0= zYk0IZYyxpwzV`&;aj>?T@qakyf7ARw)dDpPjMfE>-K=|^S=Rz$lAMFFYPqQ&1!9k^ zXbi&tV^Ybg`Y!Dz>K;b-Vb*FGi0;8g^%hw=I8H_3y;Er~qjX4bA-=6ld`Ib%zA8aV zYt6_}U_HLAB)+LBf3cz7=DSsXKS4jktqbs>#_d~xUt-*%0=(9^{R;5AjN88e|G&l^ zP=LQ_+>;CN{~32+0p9L=m7V>Krxf6Y#yzzFzsR_!72sDHcTfR-y>U-3z~>wHi~{^I z;|?yspD=E50lv++eG2eXEH9F+pRn&?uI^KS3AIcfAd!|cG>|}x6p%QJ6p%2>54C_q zS)_mjS)_o(Sfqf2SfqeNSfqdiSfqf&SEPW1SEPVMSEPVfYEZ>TAo1Q(e&BVrypHhT z`K!X4q}{*w>5B4IRUp^fAN#~aLGd{@Fiqf9x<{xa#dB;&7zKV)DV3ynqHYwjYy(*Y zRMOelz(`ghSyF+2?gc3|#835=(~>;QE*jK_|sKO|lKFr9+VHmJT;Y$FCJJig+t?0eF3M zJV;W!DLM{(Wpq49QhZHx9QaSsoghhZ)Z7^X-q$HT6<3W<*cyt{2vgl5br1zYqgue%8`T6}XjBXM38UJ;?-|tsK44TExRY%>E#Q-l zY6F`_wSccSsttUDQ7vF$R2#_tCc7Vytxh801NduI@Mz8W@R#VqdwD5y@_0mJom|Gx zn?1D0-0E7M3bs82nh5mjX?D2ildk43d`!~Yc8!Id-31}vqY2Sv5c0ilp?+be%zZEN zr(J6JR$MG_l<##B;oHlV;!D+5R8*Igl(^s&K_FL~l^`W9I5+s(6H`(9(DyK_u}1#x z*~cg3FI>_o*3S+X+*${RN3~{9tPk~Oet_|hqnIqW;S5KSkCk|M{k4L_Ov zn-1w)^>cE~B&Lect+t7H|GXwJlq)+|2r-Qfy8_{Zc8GZK1YQ-FwyOi+1g0 zRi$DIt5iiVt7;3#Y3r-wuRPG+S5TOyXJR4y3kvHZIZ#kIKavy42unA;EOAM|VD_n1 z^2IjrJTr8>ub{BNx*zW^D6Ea-KtbWMNKO@->aYUbHN69nho3#kYtqSL>l1t>k!S67yGQc=qkDAGV|c@+Ti)ehjXAT ztMa#}k-^YiO1lTUi6eH4s@_>tcNO+7-K4JIdG;X*{JuwpXEs(=FEcJ+?$;XmRtWf@ zQoD*d@`mKZOMM@5fgE1Xn<6r_K^E5+OdTPbAx@SxGD@IS1Kp~(Fc$rd01R!MKlQ#D%8)6afcbHO7P+!m07O253)2K;NwmX&hS z%V!HRd+3rc*&=oib`$qjlB_tB%@$laRL!0#--=e3PyIZZR~JhO7+vxSzcHR_TBJ?=jdV3qWyyj-KTQa?ksfX6MkEg+}G z{>DTDd`Qdyv1M@GmEkLfbR+OR;|d05GXV1#te62XC!3K+05M2LXDWlbjSjP5<41>< zY(Nt83pVK_@^9+gWNmc7g*n~=e%!=X^?}(#50Xu=`&mR|QJp>JrGtkSJo`u8iN(FH zXOLbx(Zx3XjId2lai-~|&nFn8F=c`g;j=Y#F~MNoZE+q05{*MPDW>Ef+K9*VHbZRD zjZR|xoK1`|dVQ=}3pRev(30bk%$aL9>9ll-j@RWj8IEg!ud}gi0lCtW(^4OpL%~6E zfarOIs7@G}IfE9C7BXkVv%w8HUv$;kE$&n1I1%}%dLMHf<{FF37!d0Z*`yedFK8nc zlH;2k3v~OKu<~tuF4B_2m*nF5O*)HxQRDgz{S3P+_;!nV3ph>BU2+!d19Rj!NRB5x z4=C07C6kM2;ou^&$myl4&Tes^G8c)+SJnHNi!eX5xQqd@{*X z$+1AUj|nTv#^)j}S8uM9-v4P_|Dc~C7s0bN@#Kp%Ah!VJq}K=LU?tg89;`OtbXA=O zGa1<}Qd4FRKHQ&V56lenWDJPaGR~W9b+>V5wr>15(~_Nflh2GhXash%#YEm!sdz7o zP7BBlQn`E859UO2`bL~@Hsy4wQO#rtmJpq@Hu`K7PT+&pe_igwY42UnzTK_EDo$5t zY}Dy$(@wkHt@1OA44ZOhku`HQL*lv&^H|HBF(9LJ=3hKdG#UjKF*4^otnvjbv5ves z>tT^MSm*PH)qmsHHd=1Bv;BdZgNyYu>|x*+T7I^Gt4wUXAN(ohPTz=g?WSA>bf2eK z!kJjswre-;`e0+_)9d=~)^*E*DOVB3=jj^cxQf6$&qiqsNF2`mi;oC10E-wISw$X3 zBZHNGgxI||exA~Dvt9QbuA}@C{S5OIyxKCL1w6yV*7k$XkK8zzll$o#aUHNJS54h# z2bORRA#-;luOT*8epBWy<2&WtWz1G+;N#qdd5w+S7?3!e`4=A%W(5{8GBS4`Mk9lj ze(tW)&~E(PrRD0)b)EPYHRU}0409KJfn`7o_&%j}s;;hb-?$*cg_um1jl2-sSouwv zECh7Q$wH9MRqx|uf#F$$%x?l&&NhCqX~_X{lP?x|duGSj4?cjT$@y5r9mYeO+xn8Y|Bbq!{OP^E513kAT@qoFbdpu~o@$sPLX4~9( zxM(L`ab)y>Pf<#4U?w?(9aF{#-7{2Cd_Dl?k=ASSgeQ$RK1Q@$t;%Iy$qF^qWBquc zl0JPGs-l=PNruxWnzG-2qG$gk9x$6%}|23bmfG&b1aBZ%?S)FyeF9aw94W%q`&tYH&~ zpW^kUd9Rx}uSGWJb<@0l_MP|_^isml@W{IS09n2LY4loTDMgoZKP@H}TbWBOATG%h zevH@q@2IR13iu5htaWGpa-7ng#)e<;Qo>bvs+v)KuKFhwZrm8%5RH^@N4oaoxtH3G z!POXG^H0r`5J5_qBCjMCb}&5ji%A5zR)uL8QT5Qv4lxyzFsAyUG9_ZzGo18aHK#;q z#F~7dF!zrZ(J&gl%yfvQ*^ql(2K`m0#q&mlTYZ~1AJM6$m9=Uw9k#T%5C2LOTC zt&(10p$f0D++7O%mQiEC?11^tprxp}Wj&i^?QSOw2!rP-9@VqE@azWw4Q4aXAD0fmp;f5#&{A`pv zv7qqXCP2h(ppl5N}knn>#FQeJ{59JpZhVh(z)1Uyc_PUu(-v3s{SHknE=LxTx>~5u=C}`As6Q zrZ7l$Q%EG_@B<_qV)Uibk|K&l5?JdaNOtSv)M)fwveHvTi(C-`U#Z6c={iVKTxF+& z5#andJAx#|$15#5f$!;*q0IxM}Q30MbZodNj z-^T4|-&(XE~(;aak5wARdbp5Qjwyh`%BQ#9fgB;;l#laaN>&_!@qkt?uHsG-OG~ zd!c{?BJ*43y>uL2XydU@0d6ZNnt(4iY8d!>qlSRz8#N64xKTsEuNyTC{FPBdz**|4 z!~wXcQA5C|8#N3(%%~=?7OgZi$!{w;a7?PAa83bVVak(>HAlNu#o`@KNb+>|6M-^fwk2J}&)@{R$tL{>J`=k4%5#fWn8SzwzY4$ELp_ zE2+k5H+3kyuL*pxaZfG4D~x+u0VWvzjf0B+()tgY+`;u|o;wS*DxdzfDsM3Tj$6O$?)vR`L6PIT z9Grny)Q&JBNs&H963FZehaHjh(@!M5H9ytqzLkzx>!#{@AD{aC#hP-~ywXz-JpZ1bn$sO(2Pp$(jFB=`UKV-4^E+v}Ck6gY*+92XjTv`=9P%jQiyVc4zwMNyxU(IFM}V)5 zhvuM@;@eHK4ZOQk(u(mYJYC^Y@hBX0Qamn70&nV+r1*m<34D4y3J0AOpKnLu5#Z-K zB`JQ>B-_B+ybL=1yp-u0Xg@a@4YEs?wuj_K0{yIxHxlmp&3HkPyqAdtYdB2HtH@NTob@JAx)0fP}@AG04 z;MHB4z!NIDzm09z+l{)zfh3IY*xUXiwf)W*Kp-|s&fR?HI3>)KNCSyhC5Ta2AcDTr zv(-gh+B~kLTAQgY$t$-8e_oyTcht|!#nN3Go3EP45SD+&#%~D7_{^5ypN7eL&b0JB zmUC4TX(E@PK1QiX@_H3S9yGg>V{a<5v|C3t@4AU#?snqZmJiCmIWrI0HDvVJhs;f; zJYCZ$GjURp(Oz9=o>jS9-6ssW_2;uHbBlgXC3CZdvkfFRTgWqH)g0bu$y{$L{l8Fi zVrJtMebYM3(pWQI{Yc#Fn645?%Gt|{JW53#tDkaL5lE}>Zj%$_>ZV)Md1i;a^li0V zkrps$n%P)*VpRi!?kyS%PmtHF%v%$h`3~l);(g4vAz)jnEyS{}A)KQ9zNWGX+~24n zAXa8(@e@@#iX^jz@0VFNWK#}*>^ieomwyA>c~5KhTUq%0WkF^w^sQ>&7Soz95NcJf z>jqk|T$W;a?!X&fvEABYw@6)-z40S#Mx(G^<92!m+yKK7t=(VR=x%7cWofwlx?gtM zdC91C)^;1mE4-?Paw59B*Gm88h`Y$6G)SrksPegsrOFMnk9^(`|C za#IJ6FG(a{fv4EHei@J>zeoZ9GfDxkGinIUBQvaF97EFVPFrF}$cRtSXs#nM(B1`N*hq;ruRI zb21yDchRoB^_omKz3ViAs|A%3_dR552Y#eOkkb8jw_-W)4@Qju8L+AoKgTP!n}OX2 z31eBV0qzX(sv>&8ur&aK!@t5 zXb|>PEfK+rox-=S>GUkM_htIYTki0Op2C?HyjFW3lvgH&x7>YDuMjWM&)Qq=FzFXs zg?gVfdulxiCmT=hHr64jSniwSmpd^DkK9}Geg3PH%B;}a{HD5o+f6@ZS^?)PRlShc z>mJhDA~pnGY}7FDzdLkM`cf1Ge#@v4;0^cig;KJ4q(-zc;u2Fd3Z&cMP`jXXoe8#q zHybqy#F%M~6E3>BK`EM+?2f5vO-!TbU`NyJSbH~aHh~{i&nrPnvD4&gJngiV>$GK^ z?qr>|fOIYwMnHNFcC-pg(T?OwAhlyGXUCY?K~Hjh3Z$FR=~$-IEbMFzs}NL0~PA!#S`%E+du4Rl*&|CnkbspQ_~qP(Y*21r$Sd*-29Gs#hTOc6`8cnDAh0uqZkIJM#1~bRSyIDJ z8M>fM@Mo1v>|m&ZBD6H2jB92U1isRsm%EY<-t1b2V4=LR7Kmk?`vG=|AP|e>auJAe znfH_DhGA9LVQ6+82C4D>gkcD(a>EdK@0kumsKQx%3Q-L_WgM%9LHdTkQyTOhhB3^J zvvQ2?nToQo1F=BkCsIIWo#+A* z0+9j|jnY^jigmg~d_X^^H=2{vCZ|!nUFq3!9u438ggVULMQyCt!7bFft>rhK=ky)0 zfVS`lXWGIY(xol@QJJ>H?M$sxHz>Zni%K6zZRUeS?#AS316h*8Z`CwbB*&(lWJol0 zK1Ay8*9~WS5M#d#f?eI&d4Pk`5L zfDOnn7;|~?H393e!4Fe+8~oHEE(_Hrxf6+*D9t=km$o`GZFOX~zINBXX3CA`iNOh3 zw_u)NRty0#EDF;Tp^?Q5h{;j7n+>RyV>kL}ocZGBX0_`Mvu^}A_J7=#83n#rsc>gv zq@XnCUK8|2q5f`D-v&O-)E@~%VNl;T^*@b*z}Nq_>ah}}^j2l%$`6QXm0)k{nyGcm zbJjI8)-BIj*DO-&mglTX|LAfWyH;x5vYd4-v+h=l-7+9n%Fjlp7Zflu#Hp1rd0Ecn z^q0K%>o8-KBrxzWvt*cIKFO#dAa=+N5a63_REL0!sz~`EZKzJVl_?HvX;tMTMi3Se z1QL`fBD7)QNR6>%svENr$&~9i&78Dfn_~5TH)4wA`fVp)QqnJ-$YnlZW;BKJx-vHd zyrS}us>6q-zrkVqmQ4WoIoNnc_kLV?2|^E^|)sSrzW{$QR^E45}ML?HB^O zqXA{Qw+7x%tLW|KvNYqCk#Rq;j9YVBL4h=ryEwpAmN-WOF;S#2jx>`k7x*)4xD-g5 zi4>4D6J0H&X}qF0I>I&f)c2MDUko+TNht zO_@44LgDqo|61Fs8x-$R+Bfqt(`G(M3_8OUez`>yh;87ywIdVdz8w%%VaseL$|}~Y zBhhHPp=QvIzBjWOvQ+zEWL9sC_>e8tO(6d04IQscX&(*NbW$|zP~cbUc*~BoN+F%p zH@C~RE^uJ7P?`dnM!7xkMs)}Ea;N5Fx5}Lw?*zQ}g7%+RJ>l-lD!IC${A~-@5D+KJ zI~G9}7gwu0=%9>k;u25x9J_n(!#kB$ANKBAeF}QpWiOy(lnT|m)rz!La+;;Dn{}Y7J1p{)dI9rY zOV%ME0gJ-)^v5U+#N;U4+wx`TqjBa8y1Sm*_qNk}8~7xp!WP{wC_N(z0-vW;c_*8~ zXFFPZW7BR&=-{c=!3YqY1C0}=7Zm>0L$rC*hzWxt?SLItY{o9z>c~mv&=`=e!`}o2V}@{RQ!B>Olt>FReqXkNLtwnKP7yIYL= zpXuov5~PFp)@ea44n9lvxv#qW`ds~#Z(f1_WYjQlNU5+CjTDqFxu1K+$t}E8eH-<( zu9h`QpR!Kdz|R;p3jBG^j&?z5@^{$L2Cg@16xcLt+rSqaH41#0QElLAqeg*mFscoF zt5KuCw;R<4#*ihCT&5vw=0euA@DnFF+X0D4byY`cr3Ir2JWHuckkVVDAdu))f|QL>^#hLs>C!YIxF31cNl>1|OXkf2tAy)(U)i)$+r*H$jBEz7%iSh!k1 z0u)SNJzVMS|1=H9a^V`w zgljApt}zQ29?1hxK->v7j}??oH=D)x@6Z#T^nWjIXtd4z*i_G znvV84g$xdh0YD~QWho^b{)_pZTKxaVx^S_OWiDo@Vque3KvuPC0YhnMz`-abMM>Rt9P`0%Pu#%?{(*uykB0b zivt<4@NlU2lgMJC+|hi}9dcXSAC%xpY_bQwm)`KMJg+lwUSHt4>Ee_pi2}szkoB#k zgNpTCRIKlW6ZaK52r{*RpHXTcbb=Gh?3C05zF4VJj$Yp!<$%Pll=F7r+2z{lj&jqp zytZClUR$q<*Vag2<}SfJ{R=_|&QHVLJnQP`2Fu#k!z~3jrFxx$7nt_j2Dr$zgISK>6nh3+3+xM)} z!^~Id4Rz}v^Rx}bW4YA~MA1NF-G5B<4q7wQ?(7X{*P`75i{B`a&{iuSrMsgbkg!&Q zl(w?CjRFa2B}nOWQ4mNtD?v)cBy+$ZxVR%HG~naR;N?Jk3F^lzh3<`lKzykLDQ#;8 zF9+gFB}nOuQ4okPl^~_)S@PXV>Y0t%J(fwsc=Q|dVHj(__eYX_82AUJ!aLd~w+g?zCf?$ypUha%|{~n?lv^%&KLK!vEPQEMpX&V56`MxQ|gI3l%=XqP>g(_(G%@ z0G!TzGxZ!bCi*u7q?Lr~f+SS5Bc=Kq3k&dW>P>jQ0S{3Y)XCvdwQ?+*aY6F)IekWF5j#{JA^XV66>^m6iL47+E0`HM9cpb zmnpn1QoeIq9cCg)-Xro%6`9!>zp*hWZ~rP_o@Vl&niXWn3^}=sA2^u4!fdSyK46n##U486Sramk$G=IX&(TlWm(QFSJYIZ`4wbCXbFeiUf|Sw3XaUOGY8 zFry^Bhu_kEXem1(?e;bE$AC3e7bS~As;ZT$O0K?X^=ma`P`>6{DdR?&3b#u=isd-6i72(E>hCt0W~ZnqVPf6)MSY${wRF zC49O*z@R(xO*)qBtc?fd7!e-apgd1`nb<&fJ(pPel@=yhygIOVst7^i%$Vr?;ko3};X|+;h(1peYnP1*gT#B>;-9 z)Xt_i_795z#ZTKXtN>z`N_x{!c%8PC84{L_J=YX4D99uqeMGlbq;9Q9-AXo=?`iM1 zoBcz;UnwQAU*ACc9d^VT0uskk&Uaao-gP>~`e}{v-!}@nKwrgrg%#UNW0@~rVM%&@ zCF#}Os$Sxv+_VQ`UFldx*?O<5fihZdj9Ob9&?O z$^OZHil0<^cA$qQ`;Hr!?1TH7)@`SCCp6@}+$YQ&oVTY4QTcrpnx#TZC;ODKkM)Y~ z(@%*fa7cwEQovs1`7)TD{E- zRS{LK-o{7aDeA23$Hv( zAGBj;Zu;zy7}a{K_8gm~Q{{0gv&gJWrpmLl3M(?1lZqsnlZqsvZY3Qh5%XHIS%RKX z)fck_=6Lfq$rAR9DdtHkl6aenB;LMB2khOJSce0zP)g=Wa!Mz`r?Ah9WP?QaZ_&PH z@(PLq<~?Ruk`m`?)f7ggB1w=^kt9eT){d?-TMh?)S1Acn^6Jy`dfVqkvO%KzkEy;G zB$(^Xup~$vHl_$tDv|^#6-k2hRkh(Zv*mCgeM^ugY}rw8jfx|anb%$<8zj2_rs|79 zf{8sOLApR^;#D>yM}Uu-Rv9V>*c}!AOQcpQ{Pmzoc?Z$`7;Du8(mHJW&4SY5rmG2j zg;JFurPyh5uYpeSLQaN2x{=)(c!rtQ0!H1*9*%BW%b^)aYY7%GDoSo&U83DyXWcIc ze%h#E;J1w$eX@q*5R24uV9TiFjq|>{uz%4X%ELZ~c7mVV=f0-KeAA3-0i&wb+q_7v ziK^1Q?Gv7JfqG>t9dP@c+6-g#2BK@W%*>(`2qwGxkm0EL;e##$7soq&s zz}#!os0F+~5%%+h*+nkNLO|F^_{`dnwmLj*l|(CT)lOR_0i@NdVgP|G3G!Q>i6T{% z1R7O1TL#*<%SQhN+WC-v%3X7fKFo{FykuFRha)vFV`G>Y>ZI`rnxM>&%xI({N$69N zB+iIjXfq;-A0pV24b{JE`$_$bi3)SB_0R&Imk9erWg-2PWqF%9{{qw3CXjLcOZ?nk zrJrD}mII@rBfqX=^#fbkmII^iWU=$Q!y@LDhs6%-sWd& zU{sYZc5hahv(1_0sBxZFVOi!XG8IW;i^wOfqa=qAp`&aLeMsAXOh0EfPSfcI^FP+Z zQsDn4!aj#cZSprWmr{{rE~O$#91!`Eja?E4M6e|rhby)H)%q#ND~$upb=JcXz#9`` z9|z2$?@Sz0kt7bOND>D`{$N%naX&N8NNtSxQ%}9Bo7Tf zvij3?2S0lI3he#ZzQ&`sD-hX3MdyTj;ay~X&--Yb{q$3M2JWv^IDMzL))#B-@z$^n zd_%2ayP)*)nN{ENH7f9Or9uzs%e>Manz9g;dSQLkw6>?*0!wE z513#Jh+CP5AL;SM+G=`Z;dmqcYjj?0Mk%&1{YoD}%nth)c5bcF4IFrp4Z_#~3SSy2 zpB!k%S~KGcHUC@Y{1EU5N|i@K3Wr-C$+Yu*gosRZk4SWMp*e~*VPgl!F-V>yJVz}= zG{gsdgE`a!e%2g1uCKUB3pgfMEx>!~a>RRyIX((}fl>pF6Z;AuaoWvMp7W znEo+fP5;SW|8y09gns61@G|mJrlZS0y0Ib3##AxJgbu*VXIC2Jt5x93m6Gpe`wI$N zX=aJsBNaX-QhO=fM?a;9qrm?Z#fB7~7O8U;5}?diyo}NFEvzjdo|UhHp%7IiYsUy7 zHm{r06~;@G_&ufL9nb3jCB&W9t?E z(WoPUjFvoR#VAD=lP}~q)I~oSB-1<1fH7b+^9T$WwoyGI4SRB1;bM*1=Ph=lz}>C8 z6=?`Y3BfMb-3o6v*>RWYUIyG!b%&>+7tWXGeihq%k?pQ=2)%9D*i#=x~0 zQ@eUjn8*1z3HSA{fV_?h5WVrwX&R0sN*qZb_#JE_ycArApg?%i& ztnwYo+n=TtzRS$rc-qm69Mo6Kr3*zm9c(O=Ueg1-Z!Xso(p)ZSyaM?H-a$d;V&?%D zs<2#O_?{Mq@cunM%fX5@Z7<{^NmqsS1ciT#h4g3pWG67ahsw-|iBa;1qBib!5MLE*wkUNk}By^*|Ng2GRf z6cd3zS1OnYp1+l)Gw@eR1u^hlky`@(P2?7X?@_Mwgu?G5dFg}|wyruB$%`fiv@k7_ z7feu?sib%U{EAY+MDSN5w*>t4$SnqcTe;Fi3b#h`(g_Mbi{wQU6z+)R1rroLvu))} zum}7-<-}~@Ganhnz|U7s#DGIJG4OB{1CG_iz(+(e;2t~0PAe4uXUdVpa!~^$kE+Fq z(zkRdMG(j^Rf3d0sbLa9Ai=5xDJ^w=WdS5^dpw{-|o`gIfpGES9^mG=05I#SKtNHq&a z>iafQO(2t5vUDYw$t>K_U?#J0M}e8k!i|BM%EAqSnaRQ}0h1HLEe4Ym@-#n?i7a9( z!AxY~js`Q4g*yt&JQi*Ye5Q+H%xvdT@Aj@VYNa=Q4L|Z_nc_m1R@XLV+h-Z~xZ|BBx&W)LK%p&>j z1Bg4JhcR=Gd=^0C}`40w-H^7jwd6ck1^2m_6YH3bDyuu@40-=%Nh5!+R+ zmntc|&PY*7;X)-Vm6UF)i&7x7OQjUvZ)rXTB(+O93M97}3EV?-x|E}EU~TCfD=0Ay zN>K_-1W^gRysnbM4IL^eF=a}X6qq!k63Cn>2Z91=CUQQg4Fm80_a91F>{0Xa6SadQ7T{pGEBpH;OgvWf+&R^?ZxS{17lTcT8rYhQi=s{^wdp|vVMDQi`JavrLalWWXymC9AoG~LL`n_r3vS>TA#3SF54M+7*E z2aYwE5CjhEf1yiJ;8*XQ>vuEzEi5Cz_zoon~)EybWT)BijHNQP>Woz%#brh8tE31=A3ukRoUVy)=5(FT=XM(0Ho^Ut9j&`uc;hA)(Y5Po z+zf-{NkcF~2)I8bcS?X<#mWHw zMQ>FRd?Fp}A)%Pkfdg|x=+AO*$PZdD#HVrK{Fnu<_lmHQBgP^Tl!NcJ+#Vv3%+MeP zCb;tS{=+V|XWKoeA>i}ux@qN_f&vpqe3)to*s(+mxTPHpP8tHo)urb^4a#F_;bl60 ze!=@m2G2}m{gJQ0hsw9|KrjVoh6-6<<`pp9PLi4>RSgU`m852IRRhDVrCH(0fvO7X z_nIYAb*WZCaidA9F48I}wqmK0*Lalnkz=fs<`M*n7G3iBW=;HrCTvV_R#s~O!QvYG zfZ;AMswi_O*i=DqKbQuTS+}i$UkGB~J{PTtUlU@JUMoX*IgNauxM*&YaxUMWslN6`bzsVy3%;{I^7+Veyr{c zYz9huKb7`AD(&67QQcy}3$LrFxp%3#SJJm=k)=+hw^BnHc%|{<+vZ29#BXXUo%&Hq zr+$>Q_rsqFkfLYK24o0NmdUgeMwSawhK9wOno7jBy(QK#mQWs78ZhJk_#(e z$@_rnct%+=D6=}ngT#`hUG$=_Qypr^xD@h-?*SPx`a&}-r_^RDS9|DrGTgki2 zj%&0Axyr%c?O;5bm~!Adq`4f9j1F;w~zQGc#Ez@{QXse--y)2rKit96<^X z&q>ZGl&1P|C`m?QGtGQrM4rjp>hB7_;bv=NDS~G;!ccf`E4()s-mg{gez}78v+69J zRm>B4U>=i?v9HFC05VDBmTp6DIzw@VC{=6R{b+m?-QGQ+yaz&Yy%n#ml4td>n>74; zPL8E3uZ}G_S^i3k(Ip{E(r05y`mriWpNA#shp;4l29~5xKMsU{ZRv;Ie^0-1M9bOd z$M!5e@x0GYK8FMU%cvIcBSs~^S@ep+DFaOxYqRisCU3z1YF%7^T3+M<72(jGIY?ce ztdM`cLj`92wVxn};a|aY@gaXG4hw{Xg;_S|-*q`Gl)SeIdA}IL`}G*!FUjzJR$$*l zu_}9C{^Sz?MiRtwf64A{zROb5$F3xO;7Za*tt5TOZcJ8*&3BIS_W#Kt!^po+!*ckx z)&uSn?Vy`nbRx2+iiE8Pyq9t5*2Bzsctx5UY4PkQ_jUZ2{@K0!;pxwFbp$&6d5XJ# z)_nm_C$@ycPwe+-AD8K;e6b4rAEl)TUKzJ>->Ll#%Im?yD}gCrru-Hq-<<4QJ22S?d8Jm)YP2T%rEbM!AJny4 zHB+l5`(Qp>Z?tmFWZz^T6duhK6RH|-E@sh9=H>B-(4Rb}07Sq1g|Fl_q%=b&J8e^; znFl8s&VaKzl$ub>>QItA7et1YB(sQ%_LB79j+Uek&L8v!_#^G;-v(}PR13I^QSDxe zLdK(MjyB6(AWT;SeyE?Nm08a}r8A+!b8m)MM-F?9@Lnmr*Socvx1D}YZ;XYMr?{ii zvje@XdJ8t(VOlp!>yDW0gStekLh?#0sevIJGpZV?Q}17)pEYezXskbZ@LMX*^189)*Vst`I%ja zK9FnaE$~7zTg!_xFUPlh|E)^HH*zvtDPEy;`Of;WYPQ0CS?kKz(Mz4oNhq8O^Q)@y z{n7!Fndy;=B+p%Xk!-wDq0F|z7b)0NYI=&U#=ZBNbRuL<4Q%X8rL47HsN29HceQT# z;aQF_a+I%Eev6V{KG}DI94#Qfp_Mx{PHpTv|K!HJ$$pVg9RB100=KH*tj0(wB4-e& zJG5$cs9L$UJfFbbU2hiZCi|dpzC4ALH9j@)0A-m|2qM@buhj>lUrr&(OP*;~PhAHU*9!B5;@v9N&KN!G!Y7jvDNqD z-dqvHT7FF?6p&wzlwV>~;MZg#2jsV8r5pu*N>+7DiC>(Qq7?WwnWzNvTe4D)0zW0I zR8k_$rKpdysPthLs&|xIX_Q+;nO{@Nr7*A_KS;4~kbMC!fvL&rbE?|IX-Y5Gyf{JJ2Wob7Z7+UXE<`L%u; zj5fjGr}MH30-28?JK6;$rlbf0nPrtAC4M?Dg1~xiwv*fxK_8MzW3R*vkC*$fl$rh8~Ab8P{arr_T_sxnBPyaHbI z=!%lpQ?4l}kVzs3B$I+AVDd=BfUMm?3{2oeY^u1;tvz@TW?Ph|y+Ub^Qu{oM6>f*L z3L;Au#)q$AcqM|==p*(pycj`h^bva)UXLJ;(ej8r)U9fXRPl&C6faAVDjuK!^CNQiX)&zo~k4=0ASQ9^eVv}AYFY(}+bwBp? zDmVJIDnI`9r<;8SNS!_D?sGuu>`8Z@1szYi`#k7)(%ol5$CK_p7doDF_t{W$(dR?0 z%4bB~d!G}vWHyiFHDr7zdw zA3N!Fx-OEzD(y{F+MB1ex9r1e88dl$V`<&8yUc`=_N%c<6Zl*7WjRXUr$u6~^eOu5 z=jcbNJN2WaQ$I@D`{55g<%Oc3dlji3GmT1;0`o`6p@ISvN5~}w1tyG;iwg>jZ}sdr zC1yzV>^LPNUOhYRCrW7*`9Y*g(#NSJeUwVl$LMqVnD{sJQ?@W57nwbt%|-*_nVWrI z-Q-S&bbyZ0&&*Cc)?0+upENC9Eeaaqby;My`ANB?}U zF!9;3VON?2i)m1%4+W-ykbmAe!N_c;^v6jip4=w-{UUi*+IIu*LkaKa8oZxwD>d`p zV;MLCBq?hTG-Cz1SF1`hZ-gb4wN$=40dmHzDe#`us=PC`s&4*}7k`sU!l02P<*hl! zCMKfXs`NoEC4D%vclJ09kPR|;z`x4Gz@NYA#CWV8W)8L&0xktU)2J5k`9>}E)5t3d z-&UgOP;HjG=Q^r&Y&w}oUS6puE0%)H_WVv=j7T)uAAqEMwPBI-tVi9pMa*H_+t@u2 zTUoZGA0RhaOO+J3zgk&IDb7rJd77V@8@2}$nTKZ!mE-_~pC`ketLm}T>`cRaT z=^~Z+j+1bZ6|&1s->fAS)Vfn;562lO`9s=S%C zs&1AN$&M|BFqWi`VM$_N9M?_Se7NBIdzzQy^c=mP{|dXWpWN0%$?m|N1xO6rc5{`(8HHiU6AtwcfT4$J2^RgR>f(S zIVpTgd-;KWX3iTZyg@}eVo#!#B-z%nW3l3reWhf&qs!3woA0Hhck>{%q8SYa4ja`3 zE;FjtOHqg^nwDy_+zIVCt-D%3JHjzb&t@gLqZT@1wBh0K2ETcOJKlVK$i7@&Ju}tt z58n3#@5dM3`@}ubcdBpU{-r!hL-D;z&knRa^0Z2xq=Do9=`5`qo$P~RXu|U{(n@Nu z|7*H_#2T3IYRxQZGtnp;K~2+f42?q7WFHLSh!5uYfW(e`NMel-3f|*}tS8uyNkaLa zgz{Yo)m!&+bQ1}Ab_j@{{f&tR7@xxY5>;BtR zCoAbUb$;BYpP6mucAdy&k0H?~b0Xau2 zSvFdeeNc0?s$6ZtPcEbm?r~bTRak98-EocGipf6cC)Bl8zt(`l<#LUXhLe4MXs~-* zRU&_`5=i&5Ed%Ltn!MI!x8xLnE_{;748)M!4?bvyOm=)yq0FQp&QoggLNe0jWv7n} za`QDZgYw6}!bnqort({qygbWyii|Ymp*AMfqn0w7P;IT+QlG-C1{3x$t=+185>p;} zt*W69uj{RzmW9IUa;&9cI@VbJbX6kb1H?EPYam_b#~NMOEzfNNF(mg`EzOY0j&&-O z8S7`O6H{vPLfvLHa=5@)Q$Aih>lkau6KqUmdP*M1SVNtnRa=IcD^=BC!k($M)x4Hx zYpZJLb#=YfL%C2mU5<73pr^N+ z*52)`CNI=&UVFKjjTPldEuAk&UT!fw9C(FN`GC<7&t;~Ot9W$R0~)bHS9yn#qSE~4TtXU#*msagNr!qqZ9=M-Tqrhh`cS2=F&^tu63=qeg&7?P5|Di2tXaxAApmdJeH4Mb$N{~{tBN^`0jwW^-Xm%_I(sNMXG&|ZR*aQ-GSy_OujZ(l1%z@=V zOs;yMMA+q#X<+mznUMH2-Fj>R4=`#Ou-$<;yH! zz0)!4l>8MzAkhn*jun)?Z*GhMv8NKGbfvj527IFCseCmzQBWX>tJajBVEH=+#ND!W zOy>%v4lm6)yfowR(wxIfGY&7!IlR<{W58zNQXt+0-gp@Y@ZL-jL=f!CN9$&&dpt-8iUn%)3E(vnV$GaU?ublS78 z@WU#iUK>+qXUn-3aDh_O<cA|K>vP?C3gl&ydbR<7q0J0iYZ7F@!#-`0M zkuuWO2Y9M{S!MaM6!@>^<|z{eg_u`M8GVciAp&n}hE17%7%e}UCVtWWBqPbGDU$Xz zE@|bj+pdzoZtEj)vJ=Z>XgV&V$&1M!`Rz-(Td6a9!e;mg@Qm70Fp@MpTWhb3 z4S}ETXh`Y%Q4qNGu9f$)LjoV6R9IF<3Q9lN%>;q#m6E?YKfVFRwR6U`3yk}g8P^8> z*r+ieeFo#wzX_5?q?zlqX`Oz}dT0XaNwN-D@3fhCx`W1RZ~Y8+I>G;F+`a|)Fyj^# z;A4#2uK=H6-2Mgl-NqeIfG;=h$p!ce#vNFIZ!_*G1^7PWo?3vn)M4#!JgorlZrniy zct7KwUVxVv_lyF3m~jUe;1?UWxB#DFocw`!#W(6_U*m8;O9}e|b77wXOsM6c1|-t5 z0s{%ONCAnnNC646ESW%}EK)#%EK)#XEK)#1EK)!sEK)!MEK)$?D^fthD^ftBD^kGa z7QiNucyG~IckSA9rX4(gmHcTBAKm}aiNW+4XsqqGS@R{WEJ+GXmcholqX+lw@2OFu zj&hcOFsV21-1KfpuyUPl-c^WBKVgf@VZi9=y7g-;o_(6aeT~C_EW6VhL6Fk7Y%~u8 zu2-rO>^6)nZX$Kpnt~ETCPM+NsT09&>R6yf-RWx!O6ZUv18eF;u$wwIBT;wr+JX`~ z1THtK1w78ECh*NhwSez6stx>KqgueP7}W;;#HbeV zexusJt#w3Y76a!Q)dn75R15e#quM~`yyyZl{bkU70RLSTT&$mi@-@)R2j>)&S&bjj zSSMe9&6~YXpT7B@3dRS4g$VTOX?8e2Nmuh1J|^jHyT-!K?t+l-(S+zS2>D*mRlnY& zpW%H}@;wSgtRD>YNP54L`unuGBqs2WN>yhjN}t=kvOolZcPUi~Qrcz@6Rbe|pH6>k zowk90Flrb`&(-mWQdFPZctJgmNLB!8U9HNLKA{eXAQ0c=?%UB|JgXX0iiR|!hZRZ< zY2^%Q6&P}f8PWpMP3W{$P>P)<-!7+}j^#QXvrezKPRD?FENA=if&zU9OU5#mB$v0h z=~&YJpCz~Gl!W4sKvBk+z#lJ%1j=OF4DoIxR6sB`GebrB_p)FJ45}y-C{qQ&sL{aZ zi??xxyb-7T!?pDIX&ox7fS~JHGGO|`f8VLK>D8-T8WI%xt%jTuSt5c!`rovK@&3!5 zuznjLdav%U^e5?C3VVBGkH9qIL3_#!O4;7WhEAD{q7O*>4W5|jVu{(@tIcCIIrh;{ zxxMm#3h#>)a8Q@$9=~nebPey*zCRdk01`@P1N0TzVr%)6JhJ$8{hK@=c*uMBy!ODD z%b~X6{*gxe$88UIqEX9%Cn*(@XQZI?p?Q_k%$J6!pKH3>z|}@A1EMgfZ)emmbH+DN zzrb`gfyWs&3aqPdX4EI&s-gZVrmF>fl~Kc}U1DJvev85wnPj(rkt&FaMp5)mtr~2c zC@D0rpuF5VXaZlUlpNQu{KQ1sZPQ2W^{RX0jHMNpJQS9_NiD;r8S?jG%3Z{JWfk5x zG>mCbrZ?7o@)HvUMHI_o2xNRz(kndsPYM>T9?D>STrIm^KV{K;n|l3r%aS4B`;-a` zmYfl~{gJRu*Kn|Q^qgn|kc8`D!?QM7!_lgNw8<=bm#WAY%*+wscZ?ba-lJ5t=u!Iq zf+}138pFV|9#>J7Af>U#n;`J|z3PILRz1N4fqOl%E=cLy%1YP3-JVnzq_jv`5v;1? z|AYW1+c-#OVQ9kH#tBw#WhTxs;=T*CGd;vv<{NDMx`g zR0dKa6w(?<9Lm-dVo+AZa8xJ_N;4OfWgiVh7Kw>t~IC0zpNF+oKNJ5r!6i7*t1Cn~B90ih3 zDwvuQxC8ro7OkfNNZpQYP}HFa@TaJU9kqrj>FO7z~jEGSf;q3XF}& z0hwu~90jJG$oWz8nKeT)t~_kw$g^z~)(Qxms{KY73g>1w)?LEw3^;tWjv(-~-Utqd ztq}xn*Bim%y)}X$9Q8&%E??3a#W>Gstn7@-{eG3TIeQN=+ zm2QjHIRt^W7DShw&ssnbWowa!R?SVrrxiX#UrR!CN@s{T3CxgRE0mFdAR;s5x~?Nc zuP}$T1;pN#h1vpwNNiP}GWMIRH_wR2UUmXdg%Af>g(`%IfzlRfptMCAD6Q~;`c@LC zQ#%8N6Eox+j4~b&E6wdSRRw-UE44*xrBvD6? zLE9-w!BUt4`6$QZTgEAHHZ3J7a5j}oXCQGX8&kk@x~I%^ER|B=R9b4FzzCO;6j-@X z+bbkn>Z&C-Piiq>rRuDOB`90!`c<`sB|6lGfuCoGSyr@8&9E$Mw5>G565`Ibu&jKu z)g$>6-vt~aF}8MJC~OUF9fogZ>Eb#&)PKC8(mz_u{~R9TV-+r~UV zgmhp(!Jg7Hgs5b1=@7zwDoL!Jw6juE_>W~U9=6G+tbbIpF?Do`WmK|BbqHY@l_Y3~ z5SplDML|gG^GUqD`lvm1$>qFMuausn<5}%W^E&Kc5hr=4l0}9bo})JB^diGzhHO30 zwOz8bO=W}6di3UO@X0@wUOxwROy&UA`XzD4WGW!(X9Tv^X9Ti-VsyxAQXuQ+$L?L{ z2QBa|gYC(sal=tNq;Rkac?f6hc<}?Z{xi9qinhl~YfsDTRB@tLcr2 zfqfU)x#s~rgDOptRcvPYfLv(6Xs|%U8t+T2@w-~A>1E~2(#r64J#!t)Vhs$o$}@uU zFrU~Ont5|OD!s)WmEP!%%3gNQF6~}8KWBNp&f6}RDV6P2Q*XVi_V&AK?}4k{Fdr7} z+TDk$uB#7Kbypv%>aIRi)m?q4s(bk$H}pvy?z{`N?_;WVKh)2G#_@dxg>(L)suZ%n zpzx%}8aYr<7>?vbl4#P6cTFhoj!^0C_T)PVe&t!uN^2ToEFJ6CLL4AN2jCG$El;T*Pz!Ym=93B&WFg&4}$bs3(_*+ zz7~%)eFcRdN3y@5@KlS(nt_7Ce?~G5lC#2x$yNJ6x#|rIRW+CLp{nY-hN`M&ZV@v1 z-UN85Mej)9ODs_1eFcR_SxmKU?XDpTKQ{u;Q!e~5wG%{^aN=>0_3i}@ z2Og#02L^5@y>YU$e#%2+^8NOlm5n)tw?2sD$rJTUQOxyj`CG%{`FHdbK`K>WmQmtZ zErLK!*!B$^1V>tt1#%*lXV4y=Ujq7j!g6b-zsn7$eml<)iBgyXQK-VQU;5o>DV+(- zn&ifYsbC`@gQ|N({kCy&bS_m>;4EueAjrK^nnLI8r}XC(UvoD)rSY^b0h*Qz(6k~S zpnbCenpR{2G~OjZq9PxlX}JJREAjynEtvpKvzgi%plQi>w{!I#D;KMFn;s%2#^3mt z2SkDamsA-{M*pw=ek@)N>&_rgr(HWD5 zs8kP^XKu2FYg7YYrEG?8RXDKlB}w>JMR$I-`|S56n8F}{E30@h05twc+1=u zKi)o9-Ob9^XhuoGuOBZcon#r>1lHz@2zpbi=FFm(#`z+$lvc+10<6s!5%e}Tb>&;8 zpOULUG9~2diQPAxrJJl_7v0$&8#g`RlWUva6gDi_WDPh=_V@mh-{*?8@L6INKKAs< z%;Q_0`chv|iD6TGljSYSf7Rp7%S&~$b#2V$(|y%dGRq}LSQ zje#m0lNW7xRS#$;)*@0%zhSjO{1KLj6fq*geo9NVI0>7Hld#0;4;rsZ@78+DrWQSs z^_I?1TmB)&1-Q3SM*tTZl|;`g3eoeL$~TLs3WAV{-XryXS1o!M#L4?jU1h~fVKdPa zHWMddGjS5uN90V^f02GlJ^?>$)Dggs8I?rDD+&?unnFZSg=3kBFvD2R>Q5uGsF9BK zI-pQ7f@5A#n zt6r*~@>mvdty2B+=(GRl%`4x$>76f0Z(K=w+p;h2#I%1>J7zZiKlaW%K+meW|L;ta z$s~XvDk4V2Bvh|wIBh|*dY8fsIEKH!Q~D_a$@ zLfq<#VzpwmD(==&^(SssE5CD}dp_qrbMAd-CPJ3}+We9E-tTknIp>~p?tSjFyzjei z;L?Hc%>^*eF{iUT$~eeuY37j!IlH@V@GaDhjs%wirE`c}3*}=el#irP{c?plDFoSU zdW*_BqmlJ?BkN5^)>}VQNA?=aRR{Pkr3U40zyCK~{w7?orsUQQpn1#>qj}5^rg>~8 z-|Fi1PSvE(VNH5BYSP=UNpF}~o~r>!GXmyS7Udn95j?e{{FQ8S(wYrj1ue|7bGr)tuNvnG8sYtjd^CVfzd<(oACX-2@@WKrIs8Nrj-Jtd-2 z#oi^Md^Cmfu@uTjQYarsUtChTvI+DSBkPSu*4vG&Hyv4T{qvfm+bvfe;E+BEA8c*% zy;|Ps=8Slss@hZXS68ohswRCnYtl!vCVen#(np0@{<8)k%?OwuT9kKaMtpM*i>Oqw zcS$H8O`&`&h4PUU%E!?cmsGB7z`e!DdZUr`b|dReN7h^atp@wvTK=*v2mX=Gh%*O# zH}7}O;jBp?&6@PVtVtgfV%gHbr5ORUzeRb6W(4nk?I~-FRIztS zC?8Fsd@P0Xkrc|u(HECgu3SfWi;?w4BkS!()|-y3x4u+!w9InV0j^R?t|NA6M(m^g z_LTh9)$5(ANgvLd^wF$IAIzHU7t6ym0BJ_RJkFxLLoyTg!0i8%EwYD zA4#En91Al-t|Pofsb^um(a3tck@cn{>#dJyj?T1Pb%3u`O0FYzXhu9z`|TSlRlU=*)Nt|4M3U^FzYPJJ2WGBd2>&Rs8q3cNhlvpp?oZb@{ttE$FVRY zj-Z#vfgN9 zz1_%q(~jiiLarFRMXBeVavd>8%l;4a zuiR?_@^LJ=ju-?#%Vx=Bc1zX|qqt zOT0@vd6vjkgtsWOWT#wPJVA4`+H%zao~M*tTkOzzW0CgTQ}S0Azjtb9%m`xn6b(R{ z5ik=Lx z)_!|R{_5iQPVI~tK`hs60Md+rd6`9dhh_wyHs4bsD&_B8+Q~CQt}VR9cxB+XZ;|yo zxXAiVTx5M|zfN;>jpeEX{DM+)ZLvc$Vzc(!Q}S0Azjtb9%m`xndJRCD5inO-ly_)G z@YMo)N<^jny-PcJM#xo!wfa@7GY&~KZTYl|J45${x0drJQ5;`dJN zj2S^J|4{>wW(3S&}vp;Gs({-hAoY;S&xDzsa>W{4&>}M=X_(?kv&wYW;dP z-Q}GBu8HPp+^?+7V}5;Y9`mbg^Vm$j$(->n?c`}Bml@uoOrxE0@$p+t?Edx*?Ooux zHjUP8=3^1R(~n6Ew$^PXro84KxS3*ESFa!2&M0Zd)zU!>%P;B1SG-{!Y9r2m=F3;7 z1Yh>K`)CK2hsgShjI6K4$PzI=0uWgrv!|<_=a_>lfu|VN1+F)0_;Sez{;oz z;G2{RUyahMXniUP0&kuo*w5%wtz#E>g;A@3mj5ZE-D|5Py}?ODe*@C`nKX7m$D5ZE-D|5PzXUuYdWz{`vp1vZU#%;B5T zqj@b&s*cx~uI0ct8Z{1llT!6>h10k?$pQa9$!)A?d^E`cKau1nD;l>ZIp9~5+yxbl zA0|2AkCNPmipGB>IfF;UkjpsEuLiWSzy6h#0XHSN^%adjO>)3jCAoDK zjcrK|cv+HLThVxTl56P{q(%!L`kOc^rs`zM6S&VA3_qg%n-S(3Hxy=<+ivT1j08 z$S9-WOv4nZtInwMk+uu!M6jPaqz76nfDJ#k_wTqO4QtD~iWWNLtC4^Wbt35fU>bf- zr(@W2g1=+jkrnt(~8f#>V0G1z)+1-_qgkE_5(8FxYjZX36(`l0SA?$E!3trIFR z>spBE%yB>5;!a#;y#W$d$t#ekiWHEbiWHETiWHELiWHEDiWHE5iWHD|iWHD=iWHD& ziWHDwiWHDo%4z{5tkRK%hv*6gB)HNMnBts&Emfd6dN2=K>7^?(bsR>gbZ5~F&+ z#~RfJHnNe2zURxC^E2n)a5)R!X39r=w=H_;ZTgoVsrPbkadMJ;FLINLd|3a=fPqw# zoUFHHn*sWLSNjFe!2?G}D`F0~*r+b>1fzPu38T8e^Ns2OUuRSo_+F!Wz%Loq1^&XQ z9`Jx;aJ36O(x@KrnMQSiFE*+N{9~iKz*iX62EJOUz2zR`OD~j;z20b+C0*1tzND`~oJdgRNsKFmWAh9b184r=c2b9an+hYuxb__)Eq; zvI76YxJOlBf;-rHbOk25gRRF@V8R>DFJR(3*m_(ACcxpm0VcA8t!34Ajq<1TZ-^iG zdzO@JVxi&+6{$;~#*KDq-0`!Bbp1tA4g4#m7Rs+u9U{BlOXn`G;Fw|Z4P}>~ap_zR z_tXACRV90`Xr8jfdc-k8n5MOU|!_B z$@?4^nLp6Kh1q;3;Wd#FhI*?Fbp(jb1@eQaQfy&0_cs4f6_EkCk1?}IwicjeU-Kb5 z>*ONYw}>I~%`~`~e7Tp5Z0|ee-$BDqIyh9zmbr^&F~+PWF`dFNHj`<&dFVyv=mBRoREO@3g4f9wmfv`*Y!yAHh>%*#) z3mh|Q0!52V(FE`irH162b)*_9+K(~O4)94%ls4||6CM%DPyUFSO@-^6+|_pO>KLwm z!n_{?;@q z773ZPK!UNyuNOFcb6WGL&TFkY+rSSgb&oK40q$mI+0 zu|`b*mnjw2-(E%Qxk(WC{3(L{jDBB@%LNhe1k<$&h?!utYet`!1c6PX`HN|c^yg-@ z4Scle8Ur?sw$13tNf6jHn!ikh(fQV~1ALTGqrj%oju|~k1$i-z_LwQiPQVB1Ax0xF zrU9Ea(!e@qBfgkMJK4x~7HhT3)#gSI_(`Kif!|UpRt~MVpFq`(0-yWDNJT+f$2`dd zfp33uQ;^m(o??Q)Unvze(z^1gCI~$6X-$o^j#9R8$A=KS+s4}kGH!`HkU_^frp35L z5XhiqPG7%?2E!IPAY-oOXfSG#12W`Vjs}AkIp7T%l3I?&r%p^k=~lD|PAy6!MJl_7 zAyULbRy`nr2w`bgv>y0a($fYWrj-2-AjH*47Wf}Z)xQHs{^wRu9pGt=dC{q8O`4A#VAGXOc1p-y885jqZmtj` zxs3-TK=r$$G>C-A0V&B^js`_3azIM5mZL#YiX4!U)p9Ai5xqu3#%TU$RoK>S^#6aa zynejl+!$_Zk$skt;jmGWZNp-#XBJs{(58+TgdVc)4lS;rsAsy+rlcEPpeSv+(5AW@ zU7)CPy3l4JZghcSHBO_1VM%Uuf?^h7$4?|5TA6@8s$rG;@3%tufz=m|erQ}}Qb4SE zwJ;3|LtYC7QWLct4XQ!pfD}S4M}ra&IUw1uKLn9u02FR5=WDeleU(aQKh%7S)n%yjY(Lork~6vIFFzP} z?4nBt7s^wxY;MaXNIt%9K9sKaCg1b+@Mgfg=>eyDb98@idbu~f+?(uC)BCeBSdKTC zmKTtLhuMZ`W2;Y8mgjyf&;3~Lqw>+rkCpiGgU3bTf!0dkPn41q`-X}JA=_6Ln~ps4 zlE-7`w-z0{sNw(_zF!b%{uFyJMbXtOA^6O)5UAdcoiSV<5rvFnJHK&rhS zj*pDbQ2qO*rB$fCWXMihY=LWG8f<@M-h6oz4K~2FBn>vea`P8RMC#5o7Q2v4nn1}hXAW*WVGf>$0|E-nl;9r5EGu4l6Js{Ei+A3yT&GG*w&@F|R>3x-(i zKV`iLWs)`ogdo|2Ar48eFP(5kZpR^0Lexv+&`fdLE5_~YV(6wE^wHak=t1vO7so?C!pKydq7;&d-lw+6T31Ihq=Lf){DFDEgDm~>=udIhCt!+4J_mbi4r2Ix z8m1ndDHHdwOq^d#hClA}AvY~$lrgTNAxnwx@3XSEWhM0XiSgQfZVoKwCTQ~MS)8TE zGE=8C^zrM3_uVJSd*|~Mmi{nTBu}DV773rKOkk#L&1;=A=g32gmkHe^;Rt6bc1qfT zCL>%BZp{qUJ)oHS+Jh!zYV?3&_oO{&GRQ^`C{_yXL6dD&qX!h5tZ<1CM+n1CtI-LH zRb0)?6x885sNu1iKaYi}R9ls6s{9fx_4DDC&6cT!P6$^sDr0TPVC!1Ioi3LoY=X4 za9^1)~x%fr$0;b@BI0m-hVo8oy$vg_=ocpj8q9+aLBN>jt^!V-(O9F};D zuuMLl5?uql-KusRNW|nlIUs?G&PAngshZXR2~<-Yjv^B6KmyeihhxHvcpkhp<=~ZP z^O)Yjc!B;MlJ}g~wjNXwoZFQTkI=zCHrey#-J1thtz|KQ#3ji3I0ViQHgMPLVsL~0 z4cAEPS_iFd9h5yP3*tWa`CYsJNsq<+jXb%%Fqmy27} zn~Q0JDFYhHBKzOzh8qLgR~J3`FqnKr7%jpnZ~XxOP48z$L0Z4CyPRDh6&(eAJ>OR! zz!OM$8IlLoo9&tP2=Gsp+9%%3==U1gUDxnob?fHz$^h^yO3mQIr|z-_|Lnf5F&{rK zR62WjnQ|t!=*@L*w(qg`-Pt*C8(ka)vT-QhQD=bM+wymn;8{cN-g{nnjf*qhlEU4m zrx__<7)Pt8th~MeWYLxMiXe+9+p@^AH#x@M)~RK+^#uJZ3n{QMdkbGz_Ytj`&)Tz^ z5#TqJ3e#k=wfML`Z+w3GE^D|*mz+y&=@|jOS*aO(c-39jz?wh_$vO#Sy^Lof?uAzV zfsHZ_g6`kFveL4}evVB&>MEth0FFt8yG0gTZ9O3s zF0fI!K~Qt_fK<4`(t2-kyY?j$QL`05ids^qfds!iV8 zPif^gBG@R`N2qr)eqo8=1u25U79uEYA%emZ!T&pD8hD?+t&`>d>R*Wg@V7>-0^X?< z5u}apg$QB*6(NFPx`zmYQB{oK*YXHno+kV|)jNrxu!UF&TZomgU)LBlVkNAP)v2og z91F~9;G|J2ft!^Yk{eqaq}pg-Xrin889+sd6`1ZJR$vBP_=qyIm8DU_dEF*IS!Q>BxHP z%;Y~@7LODjI{0|hUtI1ZQj^}jn)Ke)q<5|+y>ItbpYE@Jj? z#JgP=3jIGV@68$N=z9H|DiUUk9+~YL$6k{h4LX4$_G>^A5NisFnwyJMr0G^ortXWBeLF=$a-%g>m7=$_X+FQTN>AR5$|?g zDD+Ed+(i}nCUx{}`ZrZ1%+-2iwiU_H(@=I_MS5SPihLyq<;y@QUjst<0uZX@{`u5O zjmS38I}us$M`XP#k@en0);kng=TZynqfdxOrr9zwmFtS6UrOUHs>shRi;jLl|E7wB z`Klh7ZAJ1zPa4YZt4QyQRFSU)p?nz#MI+Do`$mf zD$@HRRpcu{C|?Fb`5F+)7l2SH5)t`$NR7xo(mN4Z??+_4E0OiyMAkbLS??3pmsuKT zw~wS>O5-l7$Oo&V57ob^B4HkBLz!JgGW0Z*-B*#`7pWp&2}1cY5X#qpP`&_!^4Tv` zr0gTT6H+m7=$_X+E7 zury{L{Pb?u`$+nwH148`{4;g*)mcTtyxxW~yET%br=je=iuArn75PdK%9nvqz6ONy z1t65qzE7>xi0l}>6Or|PMAo|!S?^6`y+e`pK4JY+md4rbBk7mYxQil-*a6-WRDNUkO6_G7!qwfKa{wg!0)hRHW=9y%SP7!g@a<>s^Vg_a?I5 zp~!llu>NaHu7OqxaLl>BR3@5`>QNZChvC!`{U^)n-~ez}gUU$Gvvn0MJwW>=96Jq=~|RiyW2*H@(M7`+oxk-KmoiS?gb8fUkUq+d$oE~?1Ss-yp8 z6}evDMuYhm8%i5^TPEz!6@IfUS&+Sx|Dbhb3+FG9Eu6oEEu6oEEu6oE^*$Y@JE>!B zN3{a@7^7AKPf#knCMsW)R?&W%&P9_^~O1{fs`< zI(C7TQLBKM317h3HKQL*g21NH{HxFzY0o;gfq!Gv7_e!yZAP~zL15Ep{#EE0eW7*i z0AFv^D6na?V@BVlg5kSM*Gb{hey=IW&MfEYAx0zLcMWXXNCUgZY{aiZr=4tMUxj{) zcKMFE(F6WasgSW=MeCIR6YD@e_XT{OQc;lB8A%X$?i4{^Pq`~yT=_TiZWM?=v7Tr> z*Nlz=@lrlP2*kzMmloa?UNOmCYvUT0gRZtkX`cDf2A*V87l=RdiAf;NM0K=YY<=56 zyp4i0$Ffs$xML1uUtUT9Vmw&rRJ6#Y2m+g~z5C||1Vn=Jf3j;EIkg7Px}QAg92GvqK28`F;Bh@@c2V3VnR{PlgTMGO>!H)GoULx zZTwSlnz*IPY2qbXXcIQo($D1+G;ZOv1%8Th2go;u9DV58*5ddBiRZR1o%`_c6QCS_ zj>^4lYJj8kQ3oHjbk0e0o?IJVeCgctTL(``LqqDsB(=Do%#Le1`cUsT{u0qW-0w?H z)_CJDDaUUu*hVBvQOc$Ac;_-ZuF2wUe&a7Cd0i}YH=D}t%a?vlnR5-Pru+5G=`Y$C z#Vi%(qriJ6RG2A#Hn9pZgPOC6{^7^UeZMU_gI=$H<+KO9R;jQ)_9|LmOoG6lJu_;R z6ASRDXGJRZrS%{^J&Pdle5K_3On$IYzEQM-xI%vqCu*Q|m9nA%c!XZ+L_u2DD=UJ) z&rK1eb?!=Y9{4h)%qNIXC0T>|oFQ5t(AItQuUH3ueu^NiXP#``RM+5J4c-5Cv&b5Ml&KHAF#L=jgSTXav5(sIEb)im?HOPd2Ix zw1R1k8Elk9x5{g0tW-PQDzBZfQtjl|7J2QoOSRLs+PTr@dK>sqE0pCxYG|-^ZVUXm zBnG62vd~jA}U=$5>4*2fk3LT8;*F6!O?ERMm2-idu+*v?vGp6h4r| zM?qRYwK7<4P;*}UlNZ8r3V}RFg|zmv6=ONDxe(;nYrbG&ON0I_wV0l~(Y^JPZDSM191RY?ys3aq7W(IgDvypK#D@7fCpPKi~~0rH38h; zb{Abrg2swTz4#wD3=m5sP_iJfEIg59y3d9dlxnT;Sh@sbX_K zSGUy}oRg0E(w==N{LghwX0prZfMhaM;keCvk|}~flDX^hdQr+NkYq-0X%RZ{7Dzs$ zAgxArfdn!N(y~OiW|?zKe`Csx7pB~JY08afJM27Rt4Y+y?fkkL$VpXxjr3&F#2GYj z%rMTDfnz>#W(=H6-K|TF#2VFX^_z`^P^v?tK~9VHCF){o}9YY z>oP&X%l9{2q$zr}{ykXksh)6<-1=CybbU+y9KUq&+VFSr@F7cw5BY-%xl2_3kk+Qw zr32ITK=NAcvAEe|(?I>z0Nx6+6wA_bpH1;m4!-%I{6zEqJIntXARCkLxPo*1^K21a z13X=+A-O#iZU=F){yQCue0i-mdWmLWFFP}>dMhUbI|HnGm_l0uTC04T3tz7BH>#U? zDk%m2iyoTO+PtyC;NPp^gXON(Q8B6ON=ZfTdX<-?N@S<&f#gTEha@%i*fgh@RAj%X zeI%)+KAYy&y-Dg3S)@97q_R8y6#XU*OSnq`zDcx}!wKWSrtFk8#QSrSYNw}^HIPx%W$pX1NFS&D7uJEU z%eI0;LFty%Z?zw%GqDTgGA~?eK0_A))(H_~iDy|C>I_{HSR#b-{^H9|3jV_3Q&^Zd zxtyR5G{*;rQ+wCZKKjr&di+|cPBnYN;po(!-si&RINxDr?@7^X7j0EBx72p<%X24} zvkmU)`Xkrg{YIPo?sFT7GblHT?)p6=PIcT83VX|LMT^xy>~PAkn>+pP^RQdu=8p6Q zvi*-nXg%}f7!MHyHZSt>`#i)wg{#kfS;F<8oqu1p6vtj6j-$kpo3U~W707Ma9ltM| z$NcWTFH6jSV70y)NE(J@Es#f96;bEv5Z=#?Ijv0t6|*B%+H5WoU#FK_n3fF@R@2m2 z;|+>bUf63$NGj|#c{tbP#hPuwa;N@x7S}U7PItfPO%txeY)d==dNcD&=Ah+FRr2=wF7Ae032$h%HY}Qffm{*`J zqQ~lfP^A7=A;l>akd;(iU8n0Si>gS?lnc4Xd-3yzA z@8aWi;W^0`plpLlMWtF<#Tc$1jr!#3=7S0+bo(F_DNo5kUFStDM}uk; zIUn;{E)U#t0{5B}xLXQ=%XZPsp;UqYyDg06LTTUqkL!KR=eEwg`0$Dhun^Cei;CV7 zIlzj*hCGVXq9Em);pJV4B`-TXds)I1`NbeA^2>pVvnWvIb^wB9KI);O*_cO`7TfgX zCJe`X^gqA0CKjINsWF}hJ>^RW~oU?WCX7GrdMDMr_qVkG?w zF}l7KBM~jc==xHOt}n$%`WIq!eJMsFT8Pp0S&X}TV(N*>9` z&a2_>i|+C{p_&5_oG?Vzpk6&iI}{3MQxOEF+6ivvwG&vMeu9(+K65=YgJU)Nfk&3H z-+fhH@btFg%oCQG*YH%>f~Ue3JQbEOrPXANm6D$Ut}=F+p#6Uqv*WiPi-dJXA* z(b)4VjJ)A#{beeC4T#^btB>fYubL!Nq@yND2=kyOeR0*KudSN&!F-aI6fVZe=6#H8 zVG?XFP6A=Qn^A-JGqT>%$a+ujt+!krVh`@Rz=s;u0iIw~_E^#@3U{v1bcA*b4_>_k zkJn?Qqws9^8eNT3VQYAkuIDJjES3GFT!PSI&g#YNbQg3!NAZeNbon&d0qTiq9!--e zrgLOzF^iK!FiheoZzI-oiy0;5OzbF5i`CG?eJ$956v2py<6SS@6pnDFA{=RWf%U_? z>*ZYV|0l(>kFY+YVqMrgoq;X*FKoeoVaW|yh4Q>}-tRxV+ZNN^Q+nT=Q z;$kGaeA(Fns*L4RR=aSY)os$#tx8{jridbBmPAFgC`}Xh zF9q*uz4-c?tw~=9HR*#?lfDdUlE6`~HR+3>CVdUutSQ~|gZWeSbiTcX)d4Orstw%V zsBDqF0aCgnOVI1p)Av{qy1=WA8Ufy5RCdYX z6@^ft>3!O*6smXX%MrXJA-CeMpFBd<@2_vy)TDozrY7wPyKc&B(g&m_sc{~3*QBrU zf;U%{yt!(!;LTN&1#hmJEO>L(WWk%OvOC8GZ>}tPb7je!D@)#7S@Pz}k~de*=uLP{ z!zU_mJ~DyxfeDP#9IP+Q%xi7kPki**PaSx}hWuy;pPtl# z1RbSwdrB-Id0%E>E5!-!L)x)19w^qzWX-3&sjy$_$*rs1yfn!HHzc{C zipGUW4*2RMH@~8>HOT?rnB;~l8gEW=z`sj!3o07dCOP1ZNp4|9<0DB9_}L`4S4HCs zNe=k+B)4}(hFo2+Dtt%x zK~icg2Wxd!3*{%eP=10_(@gK)$#F7^D3BdN;W3^+*GMm>7L!LPAoB*Bw3#=49`({Z z0y4EEF6UJQ;TtIkK;lx%5t+Z<4UyqACl~yeZyY*LRtXpu$MF4if%k!fCwiQPhDBb^ zcteZr?HAVibi^kfUHukneA-J^xZvmFlA{aa@>lY;amc$HRLY`R_IS!c?{q_HKfj5g zoM5(HKo7IM9@K7){JUU?pp9^uT=Z=6Z< z`j+gTh87hxuWy|rZ%v^2op#QOJZ)9*Th1zfAtoYuzkDjL*^m`q9{Le;6JY6M7ciAGYjCCT+u zg)3qWhy%4({rX{70tm!f?Sk+3F?EFsmv6ry@WfguVw}R6eIdpYof>VMB3Z$tuYlF%hlwMwVPKs>m`OS@I&wjGX?3wSuRYr-!KxPlYXbDr~`1 zVGEuLTkuraf~UgznC_*uaG3s;x2=GWRBE_&cKk+*3WjsA2yhy4P5k!B3W^Ad9v?oTh}}oD1{w;WKBb5Dm5b$|yLFIS-~cbah&ms1 zp@=wpU7=XDDHoyEogAZ0tqA4AzeHpF3Jd=jyU1HrSU#fn1BG`aiioC2j4|xwdf~Hu zx^qZ={FPdMlm2~(JQ}k2QkoN_EoXlk_^yNvW!8um*;}Rkle(J*;14 zbAT05{HBbl-P&&&AF>5}+=sK4^8x*huFi*QRmj&%0*_H@PV20JiU!dZIpAIziZ!m=Y>f!}I&rIF@Um!Kb}k*!V8AgV%cs%WGT+c#`} zyxExsi;cV_yKU2uObduKg}an+T0}G+DgJXby#HX4T1})<1XdG)G_uu6biM zO+lpjwK{hy+n3@L9?0qQw&E#JSno%a_3K$={Q?kKDj6RmOWhK?$fm$WG+)EVN;urw zI(YWr)`6`9tPQMc=?Lo^m79+EFsCDaJn4uJXFB42_2=erDIFktHGBEdH;?)8<1b#+bCCuv*>v*$ z)Y9G+llHdOX)JHhzp_oGAV_~d-C4DzC&*V2x=cY+=F( zTbLTc7N&-)RsYyS+n)Jh~ zNk6)p^nDSnfVXk@Z1}tdCM;eV8KaMY>J`c9e8}zQqL-eMf{jBUs<-xG8A;r4^=VwbqF9D?iIGRVXD@a;#W*& z?xTlJGH)84ve}6^H5&7@d#l3xRF|`NsV03*)TA$on)DS>lfEEo($~WiG$DIZgbF^% zySY9@l1pKIj3VoU6j>jo$oeov*2jtZ9-Ug}G{=-)7|s&oTL;$5)PP&1$L^!Y#O+Y# z7jKdYC1M=jq+$zHthaRlisb0oNsbTEgg`Jhp%@RY3FUnf$~z;J&pGBugWTuA z96=V!2T7=Yitt-XY6HQ`%~%q7=|Ipk;!l={GM_ePFvEPn(Z zf3Q+t2h+f$kSUMot|e(;NXYy{2Yk82kL)|Y?0fOvsbP@!+JM}XkdKl4%mKChFl641AC%vb0{Nj|`KWFnq`%%G2Vx zn;GR}56sLcPK(koaR_Q9iqqoepNY>@5z<{gBUMgO`0$i|p9D#X;dPsT1LYLSfkKLeEu=`;J}DA63n>!$a*E_YAw|NLQzQopDH67vA~`TqisVU}&rkgF z2%j7g^qH}5q##w~A!?vS9z+9>SrY|)J|gRrA>Dl@gz{-HpL`ZX)~6ua@$nZyAN(lj zBQJtJ>`|~^%vq!4&H7e&bDt0s4S3?gN&(OCndR_|fcXhO3qO|j@uTJoe$;%ykD4#| zQS-C#xy=WncN z@C~Z+RgA#&-H|JSOgni5xZS=I>IM4fuO|Hjs-0n4aHg*%X}rL`8=(zcr<8cOsiHw; zHNI4m$x-Vgf=q;(r2fBUk26;Szh~44a8CMC$?cOg_>QT8)=J=CH|1!&K}mTk0OT9Z zYFB8aZ}(gY{DM9|tL13yvnFO(R)_7A;qf5Eef5AC0zOiyS^R`taI`kC;H|R42zw-$p>tFWAtiWby$U@fBYq&3WlXf`U9z>!n439 z>Vojpv@ifS7}b7+&i$kGuP6eJ8`Zu^EAk_0>H`1WsO%-j8`TW!%wX%-ENWgxb_hPO zUSv+|%>4CGg!rC483z#a)Amb3!^>uLvUolINgXX_!(%lt+9h*-bbj$KWHJ}YV&p}N zV||S#bi4i?Y#oyg)yu@o8SivK>f-z}^+>(h3TXt03*q{ue@CPw-ZKjgww{%Z<_fip z*^m=3$_qK+C#)8%i369M06FDJfdA|e9~Sx{R1RK2st|+6tL@X14?xbE;=`*C_dd`M zp?HNj3#mWPuYCNnSJqXKnnu?_Cr4jImq^k2BV{GhKu&Z~&=0^Hl@i&KrN1DwITL!A zn#OLZ5imX!YJ}YgS19va=eCA67G6i^5@V|ku?PI4=fq*jq6a)&sd!mL>rav(@a0nk zXY!7lJ3qsD*? zFZQK1`CMG<0{_&gG2lmx>H@!F)EMwqS|-?ot-Wy5R( z8CVpg^;by{NEV|Yty_~IkRU`sTH9@8Z6I-og0#Mq1c8Ji3ewu!s-z7hR#7ljO+J6cjLe-lnO@2D_YklLEs0c2-0GZ^0WoW2<7)P$H5Fy#DM=XMIWs++U)6# zA6nr74_RPvA_8P&ah$Vs-xnDAQ6@RbHT6mS%(~j>V*bb(0!Sc(`|a#>ll#?07t518 zH~^xn@D4cxBk2-AAPQpu(Lpx*Ce{xd3#gM2=DKW3P!_>x{^QrVu6C}gjf+iPc?fe| zU37iTCW9^G=$a;kX@H|BO1-kZHp@8+j@&yOf3w`1oK)Ln7#>x>*5mL#++J_^eMYY{ za9&^OcO22hQYone;&#lFHxOBGqMTDsTPtm8ZUawKYECEyY`n~p(*_c&S`Mo}PjWzF zSIhapF|u(rdSLhrN%)=K`o~NcuWKC_f9Q2_g#>OsAh@@LAKcBv5o0um6vRcrr2}Tb z54FE5J_7fj>hx5#J~I8)`hZQV)(4VZ!`CdKBfn92Ps_=r0 zyETG0>)&vJPUUm9EnT$xelpvAH=(@mBf4i>V|%wA@Oetb`1+1_=+P6O7mpNfS&ZnR zaTDnQPgbg|o1-sE9f2?2!H(Th$NU#NZ&S6uF^gT`BcC4!7+lRe{^a-V_)G0LPt{9q zfcq&m*gD%sy|up{TV&lvfDbjQ3q0Pa5g@}EY^|H?wJ%fg5$o0iK3}PrFj_YxLEv|d z8UrqR0VUD{zQ(9A;0;Fgfcw6Xj$^1!<)aW*6jn2-~F) zwk?F8vJkd`1T7eCSF~1I2-`s769sAAo&h72}Km7^*#$s8%QXkAgvUb z?2dCjT_>(n`^*@5;pyIL=>hupVEM4}32`5FcHCVpu6J4dE;@3DsQe-Ihqk8cfh6;2 zakIx+Y|5ruNi!s(*K5w+VYwRxGR*K*lG}8A@3c6N0vYm9+<#Tf{l{3OvkQQSXj)%j z>7Cf9(u7|E_!Fg4CbE~v9;FJHb8;d8qFe6qZmiJ$Dph%~oE_?4HD9(=ce;zG{F*#+ zuT^=;g*=Lyt_PBrX%ERo>Ty`HTF730l+~=&h^?QNT$*)v8g(cgVFwYLs z@Ps?EVB#|*A5{*YJ*0SWBFK`2y>)#J92m3o_XUORUur=OOW7Y?Ki7p>pk7fhv`J>T z9Lv2r#=If`gRP^p?Ic-e0V(XPZcy`I*S`lcFOOY%cI(LCDUn9vn<{xgQ?en)@jFdp z__$>~bA!%?BoZwBi_(lec4?TgQ|u$fIbo5UUQF-0O-|Y;;%4=R-ZD3UteW-Q@cjtg zd$s?E=aBA;57d*wsuYWJ}gu~IPJ-yF|=p=fKZB}}~QKiC}r&rPXmm~=M!xu-bQIJ+! zS$T~dc=Ah{f-}3)ExFP)R}L@>T_DDTE8U9L>Ldt!;AzQ~Zbj?O%E~P|VAJRPA|Vf7 zyX14*d_LG*X#?>xp5bYYB|#uwMnPIfn9ptCcw=zwiWVc1dzrxI;PM-B`QSRG!F6nK zZ?M|v02!oA7a$`H!{}I!-e=>T05ZZTNNb~wcLLZPMt*NHAI5lT7~?jK*V-_~f%}@# zF(Cc~@5Zfyj?)sC+mXPgcjMVPA@^>1$-Cv|9buF=)qu^2F1Ltou!t@P5?Of>>*9?x zngdyG1EEC346r$n<=H?Up;5zodFaq$(RigE8xk(OY)1R77U>b-wN8}wH3@UZ;W4lSmhF{@JJ>zS1`13u@QaXa!S85#7k(sp1FHP7xa$bM)n77b8 zHX9qfV^*@kv62mrc^jN$sc$)*0)L#Q1dv$DJ)jn~&e>OXn?TMeK@3c;!#(&m9&$zr zg5YP`oDe}yE-b}C(Er<{X}l=SL*QGDYL942yEbb^fae?4{;|S4j2Z#n%a-2kn)qSr zGTs(8-nmG&zw;tGA}Zxbxt1DLv@O= zwVCxU$9Jr?qlHsNDBSTv;U3|vh7|a|g%qInI$IZdz`s{2%(DDvUGt9JQpfyiAn({N zb+{GeAbninD+zhVPu10P|2 zjRA2v=xSHA#*!cqm!lvpTq(SPfggA+cX@z|P5mk$mV^3^`LQAi0{epH(zCg#}HD z*>5Jz3w1TzRQS9bCXyHVHA&`Z_Kpo9Wu1_h=0AFM_SRynh1_kLf}z%Cxxu`sqMhn$ z1W0BOB`sUjtMElyK?QA=6uE;0yv0PLHO6{RO%AI4@&WJT%P4Sh$`%KI<*_WTVqYk4 zS}1Q-C~pu$=*F>~H9+jmluqREOI-IZ{oZ5#Pc4x{HF9Yz*;NyLyEDK^BK1>zgw;{Gp-s7>4CLq9A z51MLuimZ>pbh17+{YON|?xC|}I<3eMUU_+lHv;1;NmKmwf$#5wp9qYUC0Eu=+(c<% z8O?jGpclIcSy_Q7D*Z$;nqFrs%K5UzUR2TM{2+@D5Lbs`QC=dR!X=-eOc2mCt!sV@+h!vNY9t^1j`tAGq73evjO1lz!07_|y`+LlmgW$1gb-z&bL5K00~VPL&pO3Ur7*1z~sgE3o07KEB2=KRSQ)I zNXX<`2uQqQUs{Ar1cAgW3erl^%Wj|M(Hkdv&6*l7gle2nF_vI-yrRW$vEgRZDla(GFK6un!=5;)w&fxuywPzH|T_?@~EIR6iuL~?hn z=7;O}NXd?j(~k*VBwbz;0HQwTpFomc5d;$3I(-C}G)~DW5s$^w&le2Ki43cl5DEk6 zDxB^Z6pESw`l2o|fdeP)tI>}8a#9Cw)Y1s|>Ey@OD%#1B?EH@-gmlK}e$p1e<-p|J z8k|cO*Z3Izt13vnM)F=GzSm<571d8>#ohURMMk}kOlh`2at9R1 zsuF6pTQLasJ0b{do)EisLZpJjomb5b2@wjfS?q-B=h}TWM-o0@)3u(t#x79=flb$X z1=mJ%*G6;KvZ&2dmso`5`SAMeUl|@dRM{Q^o5S0I&1z{R?czwiN7>R4Qcm>z-vwxLILqFo?vKUTLghA zYir0FDU$Dc^$U?uKF*?RCRceNM%0B? zK``W`3WkxRiZ+ojuY5f0fRBnD@Ug+fOaW?~2FfrLP$fSj*tXMHFhp&mY2|LQ|NnyfPm<$jyo zs;$CfM%-peX&xgrH>v-sjwWkKvx^^0pT~?$HsTlFkH!;bOyKZ1uOLiu7w(PesK)rX)oxrZRv_&`jm zq&KTHrkPPn)8zhw=mpNx`?hmh^4ymOC8^!L_kXEg-I`yVArCr!n&e^+ zP0Oz}Ei8rMfi(C^Ti-|g`X0n6rvr2~3>^6W%GD3OX{3eha9hY&*+XaW&$V;m#s0(f z5UL7KFc4t@@3TGh^+`80#|Po<#hc~6$xO0^-?{Ya3iFxjWk$$@@Rz(#9)uHDJc0M~ z6yAG>-Z`4Su;&X;y=k)qN_c@}98sTc5yh)zngK-d>SK=gCqCr+d5a735IDDPdw#5o zls7Dtw<(l2*Kfo-Yk1hZyOWIGM%erie(xA9(!x2>=6?r>x$sQf+Y1~?VM`>B(}5I7 z5X%$Q@rjb}KdJw*SL^@a7YTgq^EjQuPgli#AEzV0Sf_iK?PD;VtdC9q5fQQv>ztZS zD>8(4`p4-$@cn)8`Cz0h=CbnQCQ1v7ao%esz1S_w`V2%-xCr90h45Ke8)#{2&6dKmjom zbhT~d=jvDGi6HQkI?O0YD;doe{M=}#WVB;O?`K9kK#a?Gg8;D{{ODA)k{{W6misYY z@?+fmz>-`_12Gefju(t(H>623mP8k@8S$NMboWGjk(MkY-X-ORQ6euT17w29JuP4} z30=!P6HEkw%&w66Zbd7VTQ=7=>)^4pOO2+8?}pWmaPL+pUYFJ0E|_?^(cLrgazA## zI*-v^TJRX%!)mR=y0YEQ+MO>c{P#owInB*!t)ExX;G84B$~&*3@qu)11b)q^mA_QD zuWdy-z|)Od3A{k5u-OA|Hn9%y3Zqs6uWj@M-)LeT&XTN-^|1VQ(q}Gw=U;gALMDk| zxWnI(v+ZJAoe2;OkXtHWk(YnI-a0(VU6~Udr_X5QmFTFiT283IjaVDqtqpLfk; zOtx3*lu>mwnPtr`ewOulia`vS5_0tqBt~~-3r+CRxs%lYT31NDkxO%L%We<&MT=nl zNhzBDY->>unDSfrq|_a1a3_rm^$cW}jccVwi~G#y8PWCl^v($esS4^IUZv{_>0h6`}iM58qZ z!jgh&znsAPfG`SNoU#|eUs(lk75hSY(?WTxLV1H2LN|`>tN~(grgS2QU*fuV>GvM< z1}%}d*^F!hF*hiQ_x1uukU4e86*iER2eCZx*h85pk@Yc{PS(e!|A+|Lt9ABFrxlq?Rd!Kp{nQUnP4O2gl5ud7 zULSwKR=$J1@j)`5sQy0ti9=*qqGgR`SST$lyLqpb^ulFXiGe68t;A?Lz*e9Q@?n2@ zFG|~-Ph_zI;%c}%97J==Bi?enJ6u}bv5!(&-O-2Jg$4gM_3qdDSDq0A-=ufIWk(Kt zr&8e_P_Lpjd49}AVe5|iIi{-%yvV54KorJXOtcO=5A~~oA5|&}(z+!H0&y)0`jW}L zYv zh!ccG)&?L!nI=NB3rFsZ6;soh9Vn)f-Mn&gyEes|cDq(4j94C@0SSMcIVHRK7r$W_ zBeEb6X4DDgbrqHkuPAJ@y{5oq$;`z`b=Pm#-NbIcPVUMwpOzg49!O!Hc42iCHITjDqR6bk(%57I5*$6MVdV7H* z_)O!#fjKRRIv~Xn#PTF%x2}&;hd7tHdv}empF#q^n|DwfVjNzfiko*(r+h^d0*v3? zznmiL<1(GBkJogvK92oIPk0}%b8kAW$dHQZ->vmY>hFUuSU*x(p>Y$Xg+)8>wSr#k zE@T}CqNucvqv-**9>wpUqRrVzmNXy%=c}r6%Ol=$|F5dTK1$`>fIi$VoG!Mgx3AK_ z@@N_O8l~d>6IyRfg22C3s(!LW;~^Vjf?{u*eqf+xK8s9Uy5#a6S z$l8q+jRoq6#C@T{dnf8Zg}5tHGyB@bSA3DEVeUW8t?nY#eTe>*_-w3bB!gXV(8pD@ z&m=NaWPZ_%%x|?Zn8GkkWEh1+28i!-S`ugP8ZjX$(hoSri@VX}bvmG1E%xI;Ob^O~ zS1@h}w=wr(JTHlHKgJKLp)cs)1LgEme@$Wi5!l7&rLQSO0)Gx_t>sx$CW{PyF7mQ# zo6nh$Vl*NxKN{NbVy*m~sC~VmA7_7$8UBN{I{Hf@K{?3Srh3qbgOA`c)xN@Num@ae z)F^O`QeiRaRkVIO8MPKJO>*^J)YCgIOtfw_qg~)vjG6%cuHi?wqV|ZxP|L8=E^t_FN4qH6)hZ0IeI6An>Lsg0#Mp1aTPe3eTZ(*ZwCKRUU=^U9#$R zDinVH^p$YERO?!90t1<1vL^*{atwE)wFA{3)c^l>o@CgJu`uaaPMPr{0%W?!$^S{6 z%Z)khPao)_qpY~}V&(0-2+ypmo$G3&mer+PBe`E)bg?Co8UdoK@Hin?G>W3sE4znD zI9Mws#K4=TESu<Iv7ePf0pfO?ciuo`y@~Q1 z<=Wz%HXYi)4=5FW#M$=Bm)aa@1Bq2Fht-cJIUupC<@{h7*_CSaKzYr#zJ9NN^I&n~ zMO-9s^8vxVCEQ5J#1UgOhZMv`LF4uS;beUMT|t5f+<&UmQ`P#&^jGTxHmzD8NO`@W z{`VaHE3be8pJ&u4@KmKjar7!$1Dm3b!c@=IcTrF8psQ=TUTsFZz`ru1;ajz7y)y{{ zZ&pek&`(x0a4wpedHC&;e{J*c@62o)c#D~ppDm`*G@GBE&ekrdkncJ6EB)-OyiuhoSjn+hjL3`w?AI~r&bA`%^CH{z&}U4f z2mFyyqre|46<0;C>yu8`VC(D_%5E`bJ>Tsodg#2(@rW$Uz|BfUd$c}~1cCS768n}N z%Pn+E7P=J{-e?xOz_*(#VQr@Mp(F^rO{scqrqT2-f1x<{uU+!5ZT{V8X4}ATn^{?# zX*A8|YxA$O!0#rrdsa3G%LNwXQR;$t7S8_{sWkbJMG#0f>JuFeRE3JoQ-iiQnh(3A z5HWkbnQa59WI1~Q844#l+6=95qQk(f$90`~(gps`s0rY(PJanH5GTqfx=*T(TTIso zkdB4R&sV85v%henLuAGQ#`P-wPsgM?I}6-}xYLz@$%zg~z-Ar0B^vB2Ea)S^Hz^fX ze*f<(J@%jaSC)C;>;E_k%c}>#Hz*ZfWf(qdQ3dm&KQUq8=}IMGmyPqC4$0y35;NGpwORmyTi`N%A1r`X8W zFrw3J3~TWAB=dF+@N}iZ`;BMHE9$f_N}>$m7#qSGoPDEG^!{6I?_;&U2DqP6h~7ur zX}oLvc)ztJGT+s|@(u`)UNR>x&T5-}%qh9O38WHcJH4h+b*hDYf{;JoLOwyrFH02g za-(`zDBNt(o*>#!vS?2bZ3ZpQ`%wKtLzKoh0;HFO>hdg9^rLfW`P-HFiUJO**VMin%i3T*rk>47+G)eB(;Rca>l^Xe>5k06y+z)iQeNc zimFERuXOafuT$w?Sg+BeRGQv}O&+~&wO(0OBk~~?kzaeir8RVx+y{V3j%Jt1bpBZC z{E5(caRrk)cl!+~iDaYxGYuQLAC`;8P1{>T@?8_|-?>h*5W*j*$o;M0J3yv~6y;T$ z-%b{ENvO#IZLoe6c3VGF{c}~noP#gkJQTh$6lO28JPv$#ChP{*LM@Vo?-E6X;fQm< znMc;o-oEPD{MM$!ht8IiLZG@3Vn&rj#(@n-t}KQ4%4{)snmX`Yi`uA9O`+uJg*}M{ zy*@}{3lD14#&h&<_?49KyJ)o8xrN_V+4SX5fH2tEOOAveTs>)#9K_R zV8~~IVcGo(!9%Qp`NVHoVhBWD-#hp+UIlO&MNg>Ye+*oZSeGKG@=jJp0k8$z2iT-qeH zIk}Lv!o?tp?rG=$v0Twuu4v2$+bfdaHXp9Ms+#o5YSOFoLIu}zk<2MCQZ%(lhyED- zJIHgAhEK?_Ih9>Mdm?tALUE>GXIZ#5K+%KM*u2&;VJ)C}jHO`|h+8VzaH@tVJY0Lq z3<;~olCvrp6cmaN8);ah`LIUwVP$)$lhpdN&HoYLYNaIhKWL$SjqOfGfW)zu^CoN3 z4;`a;Kex5%XRQho=!>jZ7}=RL*7f2Q)}+^0lV05l9VJ7Q%LgFd)dMRi`+?WhLK!`G zWL&=Q?1yl;Zh}tGzfwfNCn*)K9(&oz<8U3oQ}nO=b|~;fqeg(wHfp&d`{NabivcuI zhv8xXe5$D$F&zq?qkoe~_D1~kGLdWtthc0FuT;^(zEDrOtv@r75g@%v_n$tX;y3Bv zd94*&2e%H={IJps13j{JVEvq}18`r|WB1ZyXSarqTzdBK(BdT`#KEtt&;k`&zI8wu z`}1##uGS;6?tVkz&l6?h3i4jvyjOO9&U^LpUfBi0Z`J0V`d20ua9{OMV!I!je$wE) z+w_}nBO3uWRJ97Kyr!1#CtlI*+etg{cBSUa4pJ_ij#%XN`M%-%_QH>nc?J8Rp#H=} z<<(rOcB)=?@y`rHtLe+hk4QCwqFs0_o_X0n# zqJ^1wCyrLKkgZd1Q8n*33#))P8Z`?1C!@yhr4dxADqvqGoLf3I>4mrtoi%=|Z%m!{&`ZgOFlQFYR?25MTZJnEZ)i?y_=GI<=;^4w;noH5<*N#Y(a2Zrc|Vzb zOP9p;-?9_A(5akDfpk5m7W zQ=Rvy<ogOAF(jo>id3g!AV)jVor%ih@;$wac@pQ0iRuTYY?NLG@$NS4F(nV}hd zDV864&QX1-{9v}~5!pakmKAV@n`Q#$BH3tjk!-Z@(3JjzRqRUOtxCxR-01yd=1k+C z7byk_^Y2z$!+cX|1#_c0mQ~Gr_0Tj<{?&Z%0iQ509+FA=WQDIw)L$!n$goL8kp4&aGp$z}Nbj(p zXjinh%|}-o_!gz2AgyFHyKu!QqmVHJF(ca(;JeJV4lwD?Y60E!mTf+e-V!WeQj}fZ zeNoN+o0(q){E<kAiXZJ;f(-UYvg+?Hdnc-EYPT8uyvq)zhd;yQsd`YK-X#XVgA(I%hnq# zT&9&PPtvqA42xFb?mjil#4BVY7s*1Oi)3*|u6y< zW)?cYPh`TrP+6hBV`rf*i^-(v>k`Pcux8_Yl1filuT{XLX!SmN&bqIiyH){{?rc@@ zy2I+|m4{Un?WiiPqU+SZv&_#95I+Z7XYKbObug*QS5ev#=ca(@fYGCv%30 zQJAE;NH$4xkt_~~{LYFZivuF~QjEj*wg2tL6&0T0CTWeSP48d6ZUbyEo>L!kc(t-$VIX^Ao6VU zGK&Ku_)?6+6Se<|+0bB4G7HOrBbl&|!!liEo@J|O57<>I?Ar6EW^486=F}1RvS~ZM zR*zn79lOA5l#(xdTMNE6ai@a+K5-|5-nM1%{0DX1M2`8nDjk$=3$L9EHs9Aj90XlE{G~*#sVAq&-41>*xx6vPGdoj4&w40%BW!HuU@oiz)2c zjWueX$rd)aeu@PSFOD-W#(-OlS_vGnFpdF7jT#3&+o&<%^NbqbqVSDItpt)Sc~nN0 z(h#%fUb7vdPsT^Ij_m_?f+hPIuVWs4g zmy;EZ-zhm;v)z)}ZiQLY%B~C8w2*(=Hy>ELWT9;qS~jpY5DW4ly31m#e;>l*&c;r5#4;RBJ6uih3wt-8Psw4LhZQ!=tYXttE zrko!`p6;Gc_ulfZ?NNtrY8`Q?yk?gL8U-&^E5T{-+m(}hOTc%UGZ*CFc{H7_zc5|x zzf_2$h5M>_uu(;WS72NzY-n$CnnO3Nd@?xz{EAYNfz1^S@?7u=RX3Yg=L~}BHrP5R zdzy;s@0$zR{@IJnmpASf%IP-C*CG4F5_pFGl|>qOno{NU*mtZ~3*WWh((rU=H3XKv z|Bt%sF#rm?X}ikYwi7B&wI`>PqM2UYjx8#CNctik5Y0%(*l1)Ir);sm6J5yu}>Uu5Cp$Z zxmu9M#~zdf!JkvE7Nl|Fen~K*^gkY7ZXS1m?=@-|h>s!S-HO(u_lo-DffC?{_l{H) zq;;$gLj(;zpefj=t=H?P2m-$|O_0{m1I;w>M;d}?n$}AmY=Xc~DHR22{qw#i2s}jN z7X@h%)xu*m1oc%G!8Y(MMsz4Jtw8QUOGi z7ux?ONh0(6oumb=OZvC<*ZV)_DpPVn>tZ(fR3pk<2>0@Q#Rl%HTD?!ur7duOq?j@b z+=7Qj3shO)j?lG_iVIvXKbWXel|WpptJnKhlQ_pzsYx14mYL=aBNI74FroaoY7>57 zBG+qNq(Y96fQ&1YRVNF2WEmOGMb`TlS?^+Gy^oQd*-hJHG~H|$-E3#o5sb96`iN}F z&30BJQMQ+xlt$wqpy_6}=w{d4q%?w&Zbd5<$}o^MH44%q8DiaMLa5UCy4m;&d;EMV zu5bBr4+nI=l}5Iu*N|oI0b&_G~L{`1F|;ojkcGS^IIAuNFEkr zR3FkKulj6`#U$NYul|}6iRdJ7L z90J~JR2w*=$AKZ9;MI1xI0U4wf*6>p5|@D#SP&z#U$ljHeGAOOD_u$UGj;tH3iuW5gsL{=b=+}Dcah5Ix^IF}h!KuM5+N55T_K{JVppaapz4uDv`6*iO zN`FC{KAanv;9xI>pKm8Q8}=|{`eC3%tZN&CkO-ODa+ z9cYb%C+lk%ORHdpweBQcac14oFMjHXU;bvTk4#Wgs_zDO1gJEbc&cThj*ee@R65T@ zE$uV$X|1U5*-7n?7x}zyVqVJe7?4E$k35KETnu7vc!x!{oJ5vPF=ry{r%YtY6H_Cy zJ|B_o6>EHwcdx%-K%N+73b4q*d%xhlXYk%Pc<&v&_wP$u3y13jmK`0Cnvm0?mYy-? zORR08RlRk(w>0?-vK7{NHDZu{`wTRP%Fjj|;VPt&}BkQ9WSs%v8 z`Zz|`2lA(SNWDP!JMv5uaFJ0Xz)W8rvbcE9wz zPqN>6S&#i)%`+wN3wiGuyq^~EeqzA;+Q7fvUSqt2jop>^rLn1(JEX8)7yaT7A%aAEKJ{KGmf6hNJl3li=VE!HL!BqZ3&lp8Ko5AFNC3W9+I$2l!Z{+Q6fY$`&cF zC@igLIzqdZme!~0wzEvNU70vWFCfc)6UepX;%a+}s%V?4At2hN-*uCD?G#>P`q#SX z{=I${E{}XV6KVSj^e4WvNjV3lQPyi*t0$9J>Stkpfyfmq5>f$vopJfa_&?We?=;n` zy!AI{BCEZ~20dxzq_YsXO_@lhrL8-(U70I851iL1E9$JyT+oWVA1{|WUZ@_2D+=x&dSRef#dkcxQ5_3(+lr|^D8@b5s(Z;6)`@d91!)V+g}P;d^f|vZ)hC^xxKJ88 zg-ug^FkCXIw`>YqVAgAvk|7vU+b{&h`Cl@`O7^Wx9Sjj}sPjQ4nuJBZd__DMqEL8_ zGeTh<8-;Ie<+@j)o5;%JXF%fCFE45WV^_FMU}7a*fy5?s#V=wYVnAFCV*bzjy8G)1 zwRnF-*6Xf!(!`c%)Fk0x?8V)=zok%KAiY$tBPTC6U#3PU)W46p7u;JOT0L@^eDC3) zIpvph`3{98n!6rUOB@PyOK8di^nDqi?+x$AbDJ&`pV!ZYtu^)A=PsQy)d$B7m<76R zbgB=Ei<5P?^TT>E)d#mg=Xj$V6mzB74UQGauh6xZV+?0k>V~}2x1MA8fzjf`<3ct) zcg5;_&EI1Elv7~f0Y(i0A8k~2it81HqZ%|V({7~$%%gSNai)3*_#~x@N0VNta2)d& z+MnPRo>IOc6B+V3^_5kg^@V-{h2L9a63Oc~+j9LMFyHOn=4$7fJzn+mSvNMwo zkc;Vs0}MRCMyOmc%0uO%9aTZjkLE+_-$%^PAt0*iYTi#hL|3uIeNQRzhr51-sy{?; z?>tsNMFe=bQblWCsG#};?N21i8TC@Vroxj|+lYP+$d}>5p*ih8RDMy-ubJu#dqBt+ z>Bc=;tEc*9+lBIS$EiNp^W4@8>aDQrbnBj>@A(_$pugS>ynA_`6;nS3#)VG-s6E;J)sWQ<(CP7_#*qE?A3g9L#CW4FIC*R5b&L= z@>0DHZjfUT19Gqya!&h?HEfG&Uh;a8X=Z-PwU17tNJZ_ViL*Yc23DQfp_?Tho@2QY!-E)e&=0>>SZ1sapUu{-9_C2)QK_{IbBa<>bJyK>5cw>%0j6p#Vb4?Qpf z9%Zjd8a&J{oq;^cUdz#7_#)>=U(5O7i<}>QE$4?Xax;(KzKg)6 zvL0X4A?y@Zu-mO@QCp%BNM%JqT4YTGODXGS^GpOw`RZm9O$2)-i}8lXE*UUWC~%Ao z=K|-w3!KkL;JkBz^VtZT|MR&BGWd3YerGChV5YvDT|Y%PpK4#NY6D-VRQ;qYjjK(r zV^E*{sO9>!@%TjcS2T`QQqBc|?{3P`Af9q7hj4yFFO1hlXizai_FMTp(7w{u1|DkE z2=GNpg|L7(BrzZb7{tJopLhq{S+Z-7$#z$70tJ2G4R+6H1UT6c1GBS`q){hJ9ug}- zJOO@DDY3e-qQO2z&)Wt{0y391|8V|bqNHsS3ST!G=_0ei>^YLn^^bu=z^D%35^HFPd z^Kol*^O0+I^Ra7m^U?c}9!s6A3(7pXq$v-xZqnwfr_-!8hw5^n(ySsBp3=mTLQ&ce zSZP02r5Wc#bVe!tCT&tUO8b#3?MM9(9Tv-O_=y)8HO(t+)HThh_qI{j^iplItUgvA zQEWabe8jRgaxTlL@WT zBzor$dXC$iq5$j;=G zoQ(d#zVdvr_X^(o1@AqB_rAe<@8G?Ew{pU$^IG<{{R*jx#^X^w`Hf9^zDMwx86moX zOpb;E?{Q;OFQ68*XViGjpv0$E>M@TFOkzAkH$1e490^c7i?KDjmN)5;E)0{kCQ zi#SaP<1*l57+D|0$oeQo)`u~&K8}&~fqanO0y|Vc<%$dNaYl84#~L*@lcI2W2Tez6 zx6(bzKhSNgxrN_8eOe}xy=L`fJ;z|tDBPfXMkbPJS>%>8*>SM4k+XUSL&{j=YLW)K zIOz;z!K>wbW@{3^nI5~bYhw_29P^ILxDr|KOJuz>k=>mt^FGNg=2^{M8(zVCzu>)R z@ZL9gKL_A_@p}L2HOGgnCcS4h>D{VH?^8{BhicM$!*TWRQw9&y%LymjWrZ&A6r62Ah zR9>u$%6k1Qh+L3~w0%rCPb?>#g=J?`)@%HPu4Mn3j@5dJd`U$@BEVlaE`L<^-*wyF zraF5+<$q)%*#Y?9>m{%+=%)k@XeVI#&$fM8rISde_*Auh-^J2vltpCPaXI;c_e%u4 zPr#UZx=KIiwayH;YiT}D>4kxoUpEcEkP63jh6m`jwLHEu)yF2zz89#-$W$K`XX5EW zRhW;c%|<7vkLjjC)i>1#^GV$j9=8!c<#$|_;0kE614DA^EpnG}s*lC?qnhD{2T+tu z@&JZ%Fb^O|P4a-`d`a(QHS9ofb1QU`@kro&=136%m7`A|L81JRgu*%|gqm6RIA*XR zw@zDNVlA(^0}|vQ1}5O)t}A|7kXqvgbwvzab6>M}+!6=0#}x|w1b!Fx2;pGq+Y-ey!6h>xQ%;jtF}Y+b3ooe9}c2vAEErB znwMvNr|G8pSW`Al^}&obJB0^arJqs^sXp2)KEaAxm?_+u zKm>d8^ac?9vTw+qN~Iez<*<6G;_7~)8%|l3m+G}YM~*=Z$ni@ici zAyW>kmnvrHg*w8WtMXF4=Fr8e?Unk;v*NU`RX#o|er9~?5%Mo|WBr(^Y?D+#y+b#J zV8<;m@3l_x`BUix#gOU@)n|r&dBiPx6drFv1bgAl zJ6GkUdd;C5)Om&+?)=IWqIr{UUKnVJWw=rR_ZbUy+$M>D!Z)2kvc-oQ14tHrJunx^ z9yaqL<*<6GVpx~!mA>cN6~7VSdZofsbt4t6*Cs*WPmIcb`Y~7EEvfIC`YTOW7nsy% z-yqM`w@d2Vrv8nlz74$2s17h$9Q6szEq1W@LQ^zMu&yzx1AMPhSz4c_n@}h#C5o=H zUZX&q?Qd-z@?PGi;y<)*nMmtlvD3-=DFA}nVEj zQIJ;hBTIMgM;kv*Ge1@Vu^iO5D_UopU>iu-<@v6fAkIWVTFH-W-N6q6DmP()O`~IGl=>AxAkhm( z$0}O?VZ#^$;!hN$b(;-i4EU6X$I_J>l#>+=ia7SB^(3p`F(AXOdq+i$|J5xj=W20UIG@OWXs2L;nkVO2M;u(yJi#Q-1d`)9tqqff)OLJo=d?CWT4vvE`a0iJNQOg1$m`Uw8mYeS1DWo0>=fyiKhL>A0 zj03N-VO}~}(MUxzzO$mio5IyfiiR{mF=^4m}s$Ztb=?ebet8|4Qe^B-V(mG)0zXnQx%g@~G%NlboOG5@(EVsy1F z4SwaOn#)8=AN|dg2Jy2U!_IO^qJF|=(kz(Ufo!eG?**{t*z$_9MqpJ2wtNc zNTZzfDNO47c5dcIe+#P+Mk`xf&H$#7Zr#4o;`z-a2xQrnOBy0b>n6+Q8NjqQ1wp?b z!4(!aQFqtIiWVu8Q~(?5M6j1S7HCnod7`3)4k5F|mqv{M_p(yx0v~DA2#`51DFHJ53rDOkPz9Ik=Ro+rjQj>5ZC2yGTU+It`qG7m z_brYu>tX}>LEs1ke5$;qh2cmm{HWy-kIJk)sCC42b3w>kG$Bj|A#e3O9oNPBIj=Pq z?y}It`+-0g9?rQ%yDd5}Dwe_oK18Xof{j$PKAHr9OP0jGQIOUuWkn!ZO6QQgGIOl z#KREbPDSgaCDDRZJrEzGAT8V!L0~h&`LFQhK97}r9y6ccVLp!m@lq~nY$~`s=3TDk za*xNn$FB4u zZr)3ApoRT`P(Vucx3z1?y=&tgYo{D~1M%+u-_3fz>d@lZPUHP6jrUwlpj^Gg?Sq!Y z)5d*&kv=?Ji^l>B;jUq$2GL{3!i)CTKq;ZRX1p9$|p zT@v_#s+#mmc5z;AUJ?zwK|keU8t_`9I>4KZ8vmJk|115Jj=%@nv2Q0TPyc&!;o>s( zP#!MJp&eC0E@_GA-Am#p{p8Zq0<}!H!qup}SI772mvbincajd1^cFOGk&>E!(^8$I zz87|*UZj||pJjT^^d9GW22aXjh@PLTzQNYozF7S*$4?`3L-gQQ7?S=EA(f9?pmPhjvtz=HCJ84E&DkD}0u{NNM_Aq1_o?&x-Yeg^Kke#fsgg20vwfcYydH6+7w2lIt$aU36cq zy3^c+xz?->0pFz50dlAI((r4H;abi_*q0xCMEzEGEqd<}doJc#&&YYfLgc(iF>blZOz5~PuiQL3Yeir9Cx?iQb(_Dvny;;rX`fGI)Ui6HF7c4}=ixeYqof>?P`P2d8 zgG6GJk3_D!P($c`tLjdXfO)rB%_6}83|{n%gcmGC!iy9m@p1L#hvriUh!3UNl&XO4YCHh(Peu-ZHJ5@jB;12jIqgDc6JMC&At#2hkpKsJZ#zeZn zEk>;ZqA;9cbt_s){iPuRTw@Y%p!3XuHO=U0h#Zqy2Uh&A#kGR}AlhBXq^efV}N0T)d+_3#%V8 zhc~t+q$}-P&EYnXhz!V8vmolz`WM>eqxv~texQnrWU3_&B)lVp9s%kKu|h#lFO`K zQ7hN922CF_`)ho+yw^&4{n~o1^t}$(O=MXftfSX6G}48izVWu?1-1hfD(D|l+&*Z) z<_WXB7N$WwWTyvYAr0TaoS;E=QR>IxJqj08<>7#W`LxCR5H77sA6w$d(Smx3U*9TB zRBv`sG&@UKz#4*3z;Ue!F*MkLXD3pZOxsRLJfw z#6q--%#7~LYh66&$b+p2SOb-&N$zDyJZ<2h&|#)w4%T`v6c4C3*4?xP1}pL&MNut_ zZoU>vjHBUt>rBQM){2f;Rx;n*<{en1{faK^g$qY4+XW|%I-s``aa-AmqYl`)6S{q- zNXS0a$G^vJb%QP^%xDq&tR9k{@#tt~PV2(HipCET*JC@eHD#o*q}D_S2X56h6>*>r=r0*kr?qOLPGgD%1qm&Gj2dmd)KxeX=f-O z(ozi9y}|fi@9VbBy;#Jd*6Z0>5C`K>L;NjzzVaUweuFxN zRe1{Yy21SlUZ;|~Mb?29MIvMo(knN}X57SRM!3MrSDLEXu zKD#1IzavyAj0}v=(($&ekBoq-q}CYzL)E#oDBm1$f(u8tCXsbh_=n$`h9tF zd~pT)7uF%WhKU&zg*<`j9&QbR|4CI9bN4>oc#GLv4Q%wgJ-cY_g$h4{G^{zGmJ1>; z$V9R)vlY^`eUr^4ia-(~ES4I{6P8)S-ZaYkudoXb+SKU&1gbJ#e;mq3RMGwF5Hn({fy&f*h zcpdOF^^c3Z?;Tm+_lH`(JM>eoZ~{4M?3YVz{?EHy-vsp*i+4sbc$XR+Y+a@|gkgSZ zu4D&QnC1AS8C4GG@2hNJgQ$D`@P5?ry+VfGxSD&dk@Y$w>oqRY_FZNL{QbJ)KVgBt>LaD#|E1_OLiSxmSpDB)#Bw35S*2sFDk@Xr6 z(?FeM0qX!cA@6Txi(spmQ18d=6N(o{W(kEk(fr$`gi@6ES3t@5AroC`r0* zvhrFZ>vcxfYdl>8wbBCC0otVoc}&Ltc}MEpc|T^KP`v17358i>{_RpiDa!jRpCDi*d`-I}fN3(>& z{I&VF0>~AtedV~lepX(dualbr2laTq&eve`*onpH{>r=8Pkw!l;C+_7?~+S7KlWN9 z>vcxfYn;+}Ru-lCG=vQjxgUb|3-li*Kxgbbw0gsvrjUfJxxsx%B!{F zcavbc=5zK%7w=1R)>5eVSCYMclIary?^EJ^m&D1A)@zNd*BM!_ajgdGDht>ckXz)k z`<>ma^?uAgp?I+_ODN1H^KX|DN>Scl3HAC(oDV*{4}T${vIq5AC3UmjuWr>qy(t9@ z$a4U)U(GH?c}MEpc|T^KP`sd)2Iaf1afZUY+5F3HS$SWYGnAsdzY^;8lepbTsO(q0 zR!OL^UT0)|iQKG#`m_bC1N?zfvR|EDLcJfePbgk|OiiYQ!hF*F+ob~~it_$SsMmiV zBvkgRUaO=|*g`^u^$Gok2I@x^ur`o~0c0OKyM%f_W}i^J_%}-^`%svln19(<&386K z{bM)H+3N$;+~&jNZIC#xAKvSRpGmKzPWGW*Yh=C7$j+ouzF#`Imp!U-AR6;2GGC#&A z=Tfpjda2d9RAGKi=2DViuWLx}$X{q**ck!7*Qk}i_bC;>>_h9TNf7ui(*%1NJ;OS7 zfj1eo3W%9tv|G{odJ+USjplDuWu(uyj&0yy8Z`>UO#HqHtviw+uxYg2Gra4qV+Z(d zqlST)2}V0M(pyz9d^hBL`7R&r&zXYkSnxHviP6Y|AHb%Kw%Pc5vvIN5SX$9eHnOwC z?b>D06JmPhaW~-MMhyc`RVr2vty7Mn?=bMG$3{w?YPkA58b_QE$=I9Ln6mN|aNL~! z$GJzFCtV=6gRX8xYi$w)t}|*3h{f19nTsel+AbMwo6)n(LK}!Bd2kNcG@4h~AFGba zbgcls$Eb1O`;?l~x@=QL<8P82@Y6|dqN4HTBnP}B$!)A?d@soXf0*PpR5X5(({nrjORi)bxOgGE%n zM|bsk10oGUq|rh_8%ttya*)PPCXEEivNtx{d#81=l0 zR#MjiGRi18(=bKq#`CItr0s$_5$vT7X_+U$h99@|?)c&~tj*_Fw9p~@USLC=2zo!5 zy5H6*93F85-)r2F75KqNL=F9|qbl&B#vNUOpJm)J75FUUj;+8~7#0os1t4e?CNfr1IU#GF`q|$) zsRFYWhM3MA_ropj#8uWPAYqlf0*R_f0ST%|0g0(d0ST!{0g0$c0STx`0g0zb0STu_ z0g0wa0STr^0g0uo7C^!(9a(sYu24XND;sGpufRW2PP7C6+o%!X z{#sEY1w7iQ5n#utA>d1m8Uen>s3G86jT!-d(x@Te*Nqwh{-;qxzvkM>W)7fW-r>DU;oAMz)3$D{mf2*IomD|hR>f|K(UgXOv@|~mO z7FnM+udir5+|DaTfKOH`3eq|?2?D#*1Zka-1cASvCP-_cp1(-_04eWi+K=R`I+B}o zB+?Q13#AsymvNWK(aN^DODnh+sr5a>es0@bHbY}mz98nyV=M3njXSOaf5o`tEATzWolt>^ZGY>;3QTbOTTiUOL^teh!Gt&b zo(h=w_P3r=feCQfi-L)4e`|Rb&u>$v=I`NEzv2V*Q_2H;5b#WMmt&ZsVM*r*ZU%Z%y*Uu)C|kb;y$XCM(QteUr~g0GsQ zAs`EfJb(gZ4UsdYUu9=XJ5j~?g{bl~0@dHQ<+R(<1vB;P_Ud)H_6qap-HJc1pYpB$ zyJev}VsQoYMcpFq-+eBPFHaYjA9jP;p2V*_52u!?bERWC9RA4h(OBb6?B2YhhwJb^rCAJDS>+kF>c`r~0KD&ip!R=C z3L}uNGJx&b{CSMZBP8E*_o|Ook&mk@a?%LALn+z&Z$Ee7$YopQH%!(JF2?D5QxD)H z^o+&zfPR8*78^i3b3MG5&rmPBDF8s+kpSF%E)5)#OamXM#%nnmm=`&3@?15w$xLR) zhiB-f6D$%#4E51A)FB`?7s%IBq}al7mbdu|RY(R37o8FLh}x9lqy-`M%ibJFzcuD- zwr=JkYpMtZT2<@LXvT8v?$`s9`)yiiZ7c{klr0 z&<_C_guJBzxQC5r2$*zd-(){orJtL60aLGRJtjvVPQ8GsSN5j2JyrU|)XNX`X5I9+ zy+xj}meLM*QzdcQm)_zYijox*P&NW?K;A4y$24Xqh z3#Nq`IXnQ~WqrHA~vgM93;G~=g0`V}I?3llAHh()nyorLelF@9r&y9|ijEw18zc9MgbP{elc*Wn%X2lTk#95h3 zc=%%*-w+T3{jEz{-~pQuLwRbmL+{*$-&K^oC-Uylxs}g&eEMx}mh)QJXI;vaFcL%h z(-_q< z3T-sV8!J^q`AmxaSCyOqva3i=z`R?9K-+IK(d=3^U4H{d=(-zLKWGleUo)WnNprXjBqF_j_~6aj&z5_N1PGO$&wJD9CDM ztYl%VU||dk*V%~WO@cIXt?s@)xo3lEHz`lQ2>E4*+2LCje^Yi-$@SZqj{f24*eNEqp5{y zP%|?2yW}?t=2j3C5n581BYIRe@|753)#_`l1hQ7hYVd(gg(czxoAM=Mr8jwrntZE% z%KG~@g`Z5+rxb2a)HfCWEKy!}KAaT{hfvA2R^ZjDsPw}Hxu!K}y3y=sx61NftLXJ} z>$S>z{RG`a>g5_7y$-k4d$sT7ZPmX;fC~DDyoVnypv}~;g=r8E$pnzqRWiL}f=08L zg3_5hN4yuv z3;M$Z@f4Cs_ue>jEte<=-{aBQDM}MIJFIX6rrimF@Cwu&Gf!CYf zDUa$iw3O2p7_2njHV=hiiSbT;xWF54zUtkZZ*KDrOnVKycVVUZd}j0W#l3YUZgPevTqs zUG$#Z2bI9?=l$w+y(rN*YdL-o13zO!S^YVMUodJ7u(31SF;Uo=?U;y))Bis1%_{T5 zZf@JH3S{ux)8IF6qVbkesl1KTp>@tgt zaVb}QhRP#UP+iEIeDxV$o~#NaL&Bn_kxgN5)`Hqets;ssLIco>J`8{lnP_XT>1yyh)_ZG z-a0cI(@4H!PCE8f1rlOmQPW70u!STEOOo!es~$naFY?q0>xcW^KZwjN`Y9(!z)u@B z0=!+RMfEh)UkZTyvB_trNmxNoNHm!4VJ?D+g+w|}{x8(dug%^Fu+cAneY+PbJU?`| z=3oz-i`k9H|IS3Rr;rZL^L$;(^9$5%$+NInYUEkiLY{>!X$2&MM7n<4Cz{yP5tHmLFuOrQ|{%;IlM;G@un;-FxOiSR|0QP>HxVx zv|;)yT#j5OXY~hZ8TbL_zRMiqY$*A@s}sv-)@vk#J|OTjxhdJ&JXv|Ik@Y$w>owAs zGabone&&)#N0La-6rI^-_BKt@XS2-0eAzO4Ch(svvlq!X<_1sNUmoON7GKV@>zUMdr(j=v4YY_eD~2tjZ@}b~jI6Bl++#g-;Gf@^h>dsZ3Z8;Jo@lh{Y)8zIZC(4of+VBlv*NB z>x8QpySav*rRtLRyPH|Bkqr7Qz_;9=53*!S_Ksd_WWCPFI)yFzzJ7V!BG>=Ch&Nhi zmVPNVv(4jC!C;d`vW}9@f^iH*yk`D8J^Ci1$-~&)EyZD}b z+1)&OjpW0}6y8VGyD71kW!Gzstk)S?uaUmru-L5eBHm~{Iq8>TGuu3URxN!&KU1Dy zzGlA6E>HNLeA(SRd5z@5#}t01kV)+20LN>Mtk)S?uaUk3c3)++7b)h6eknGy&C@-a zg?sfgmk`EtK_?bc`v6r2l*BV)`GqPSIeIIYJncdEgeknGy z&C?^)(j)aVobUmtk4bm}BKfkr zdGZ>`hmR?|kE(Z5VlVqTuQjq>XJoxb`Zk`H@FL!5J)h{8Vl&%3y-l<5t}IV5H>NxR zKb8s4?4=~@bJ{I=xw{$k8p)K8HvCL(N>*eq=(R@H>x`_|NZycer6`#hJrcFp5y2MFH-7}@@k41PT9G0$ZgtLbRksS!<|JJ zAIAZ`p^qwww(Q-!)B(AatgT+^q+Ckoua|mmE+w_#rIzb=KK@WY<-R)b&y89MyiTe3 zy%AcUOoG5qPZR8A^i=EE1y)9_0%9f@?N+pIO@hFt(foDy%eBLZb!-E#HEI-ynfTQO zTDK%YVAE*+t&io};f2<*1N<|ihJly~MmsjrcdKCdnn8HoJ?#&fg6yl`m*^%&BeNFR zw9z&j?=u_mb@#NBjkZtX?b_wr=Eex{-<67GLhGEkzPyZHPA8^$mY$Afp>iq;Fw=rEAs$YVo5#uEF|!n?xliaeZcT*GqE)iysC znjdXo+o&!Of8+r(AkIW}v|eg`+dzUZ!vYfd*q7F;Q(qt>je;}hp;H=c#|De<@?|I> zjt2{!iWZKFAP_I*D?Vdj%0pWdgQUAjuKVbTXw`_Wo z$UIZ~JwzU9J^J9at)=lbSzDjGZSG;=?Qm>o&~Kj}j3|B70Y@#H^U}fiG|$p)b1!Wj zFfC*asdJLl(q1yVuIcE5z2Ept9PwL|)bOZeEtksu&SiF8Q|>qZlDiMYyt@i2yMI7$ zsiymNA@vq*jAE7w^HKO}yHTs`6E@36fVV0Yru#@%rrUMvk5fnB-8=1g`O~8hcj#x~ zJGk3*)Avne1bCNHWvAGBZ|VsA#V&U2mOAFgIIU;I(JazWS@nT?Dis%GTF*{`z-pSH z&(MLo6=|^moL$G0Rl&m6y1Co>jy*Um9^S}_zo`*id7&yOJXU>$Zu-CaIj6O5Ugp3S z-S|4wGX$hxzg#c%f7hw_r>t8Sc&}0+{CVDP{7pN4Njv_FSsVi1X;c@u!>A!3e)YG` zpX-OydS+zy(N8&W10JkYEPPs*B|+fZj2Z=g-KY`Z>CcLmme4; zE|9cEL0XqtWV%4&7zJsi5O!l;8vkoVh?}erK!O%@wJn6Nu@JU_1T6~EdbkCl4J1BM zkk-4CAdpZ*L0S*9z_fvcA_~%4Z-Hq82}Km7l>(EU%;nQ{eNg0Q_jhN44I!}VDA4aiF%_yP-Bm>^saoe=7sP}0B=+( zWn!gQ^b1wQoRc#YAiCwpj3z3yKU-BEAp4`E!tE1RJy@Ru1xqlW(Jd4VZIT(zta7i8F|PBv{uMr|Ed?vM?Q|*++^6 zf3fU4bh;nA?0k7K@HA;3<=3e<^p?U0vLx1p@B6W^Uab8W*4e2GyMjYO=?s@yg`UD0 zKz?YkvaL%2tB6c)Rt=6Pg!+U^v49BW-N&`>7Tkx!ept3x^-ojP&Edk~)E;%Zjy^aB z1HW!4^n}CF={>#A+jSB>$Mz> zr>I=N_);dWGBoMR7BwN0aAW|U}&!$4vcjE-4RtT*pEz&|o-42X-dFRhQ5UjEz_Y81!_gLmDE)}A(uE|3w*!~rrwxy%n7)Q5;f3V4cA zb6OiVRy3MH$**b$+{1Fv0VXH1ojj4CCgjH+fK4aH%!xlYC&qwSme<__u`KF<$%$1nM`FAb ziE(q{By(aMnAGQwnd$sSr<_kQw&P6EFz^*hh4-#8oNqKZGRwG1?`K<}@>mQvNW(jk zyjgf`pRE1r_p^l#5I7az&(@L`QG3^{i}x>e@nQE~?mRBky&UTHRWx3&4$I{VAd7~q z?;HTDr9y4dN^&fRD2h9|x|qu3L9pwow~9Ol}nTOQihSI&26pS+*i@MMA<>9jrbG^%xjVgAVCe+c8Th@ zk{B`VPon@5#$*PDXv%2?keyU8117?A!Y+{5u#*ac;7942UJJ5>U#Vp2qBIqAy+z`( zR=n&E!?42}&%#+Pe&Q26Zaqk?!PCZ@LHVv}v;cFi>p`20ryhRx;5|NuUw-6-#{1da zhOby?ZkvsbBlV=IWP@!b8*KA7ZqVpcRC1$o02GyqYYVO zMQad-N!h)ywN=*Zux;isj1zermqirF7~~XZ$7MdjBG*ehdF^kVkVQCfz4{SZv3KPL z{Ng~bCTZYdIKczk;a}*Dw`qkCk_IzT9-KX>qCw1SJ2Z%CEjg3hECG0qWAOOI5&=&e zPtpPa!xBLc+PIf`%;cjS*Aiv26c3Yenh1*^kig0TFp#1AUL9f|rn{o&r(F`z{FWbZ z+u&owV2E~DL?{3zS7~AK{pup|9gCIYdp4IHub{cDm*;!hI#D{eK_h1`y-s9i(k^$0 z=T?|{fSR1sx_E9yg9ysETro-j3txP_Z+c09mu2ayNk1NMPlT42dgebZOHURM>3LE` z;~5tGHt=){&iRuSjTD?V!C7d>ds?!|<^5Czy~N~_lCw*))s&w27q0Guk2GJ0fXAAz zYx}AvIje>1zwcEM5?jY2`5jR~=&_lj2;HO_gQMV|C@0^92i~L9oYrLDF6sJ~=^6rl z*L1BNsc7tN+0f&YG%DQs2kjSJ1>>KbOs5VJq46Q6xea`@QgTmuvZ6su3y*4`YJct5 zFJ}{Ax(l}ve5jes27^elmfainBJ*WGyHI`!GK>2~I%X=mP%ybjX3mS$D~%84l{!aQ zJmsj7{U3*ag6xF~y$+TLbC9Rd<>v9>uoe)zV^-;??QvDV+( zIN+W9k*1&P0fnhU_b2F#k!!WU)23a5rS+mD2s~7O%ttP@0M{uMj^0KpT0hac5JBL* zO39U;9h(M@UR*&uSdaaq23iaBojjre_%fwT1H|a_VmzV-T8I2$WJLq;7^O@D#H*7m z@NblgN@@K=5(I8NyJ?-)waS_)h!0K^_leB2)GcMHTal%AS(duMuP7x2cH;z%Une=> zdHRmVT8_pwN`{Np3g_x{t>tKRloUB2B~i=KpdvyTyH+0`wnFLxDTlf@4XPpbru9~> z7_kkc6zbkIsD#*?*70Y>px2@_PE%4;0zcA}qw%XG2c%$XeKe?-Xqpx!Q;X7|VxlN5 z%B2>i(X5^Pwm@Dx?NaTut#(GOCfdL^T3S~Fn;C2uGPs%ylAbV-c14Q}$t?sRL63s8 zh^GhwiCYw;Mc_mb*bGemWuZJUol;;r7MQnLU^+nNk?hZagEmzTOC)DaS|&R`tpf2My&tLqBsU5z+w?dG(}e)!t8*K+G9bN zDB}#Pv{2LlORrFVy%h?}tdc_c^-d_iPh6}yVa|%*K++ZpV$7y5nG-=EnfvV}@jaH; zl|Ygiy`@Do#akdLj)Js^u?PZ5d=#Wb?nMwt?#pFBUMPpCjuw+j)B&6GEC1!HJYl0G zYNgGuQ6MRjJCUc6rjAaXz%iYE*x>x+335K3fg^NFbpZ<;OCK=|9FcpEPN2XMGJ+F0 zACkb$T&}~6e%wYs3~Y>kd13U+3!`6N82$3X=$98pzq~N|<%Q8NFN}V9Vf4%W==1&M zn)m5)Ms>?|m{T+#&(_ZaNdlR|wpHrdyfR?ImCZ$}Q0*>jXIjxC_3WMv^$^r6KMSTvs{zX>* zOfz!pRbG;so`Wv-K=N|!AxTX=E-pVgjqH^6k))RTTs-~B>F=v_ic}|$RCbMy2|Qp+ z)!`NRY03@Ag@^>LRrx!zu_U? z+^b{ED`HZ5n3{9NkLf7)V?xX0%+nvHM&fT(a{n~RGFfC;`uA(fPMf+u8t+o=^pu(f zvU=7v>-#Z(-k|*#)=`Yl`NE-~^w9I2+7Fq+g&2TbFZhyAb!vDetF4m9QAaB2^k&fEHUMu0W)wdz!}CmfFMQKyZbKI9x7 zHI90FiTv)4L2(if>eF&|XdDOEB!lKixF-Fva7{AHIU24>e>_~1GsmOH^*B7_BHXTD zk7GDog!|2}$Fa1C21d?xx%=c29ml3bFdW2=RJ6E+vWwT_$on3)mmURD3OtKNtRJDL zQ1Cu^+jVZzU-ss7rI3Uo?v#;GzKDb(kd%;6e!xP_Ueu)bqb9v4HR*k+ z$(g*7%R2b-Lc5H-8c2vl)%J;kd)p^)@9*@ga4kvWr%H+uAaz;GF$^vY*CdTKcJX&L zkige+h2d_Wm}$7u^Tr8k;e}RaD}gN0GDiqYBL+8?VsPUGRdcl-%!!ekC;sfY(U*VH zPdOI^60=&4Mk6pcP7s*JfNz|j!7)nR-Ot0CEQIF93FhPtdP%&N>=iHKAVRlH6e4kp zMWSB|zLuP?jc+!|J2e805#C}WoYE}U-SNBO;w@Q9zmv`AHQqtlSNmcW%12r#A7`QP zfp7`sgOq#lzdMA=7vg=gLm<-sWk+vqAd`4*>%uMjRkWzv`C)f-kZhnd*e6GETFmY6 zOx~t}gRa;wyH93{C@6N#Q4uYcfTRe9sxw6pOoyn5N@OP;Sz4@#Nj(gULX@Y)5@AMt zc5H($_hf+yz||+~(rY@989Dj~k{0pqihX({Nj4OcBy5i)ZQiFxlB7!^Ny7F>k|-)C zNxBr0By1r`!uCj#s4pZ*SGq4YxOM zD!9FQQ^D=cn+k5{Kauncb(f--vru3ou3JhDZYepqrR3n2l7m}H4sOYQ7qVZEbMQtS zBv>3i7tWlKW=UNrzW_QhAcq6;M6}P`0ea~8WIKe+o?W_!gDn*aS6liN?`hl-75HJs z9a(`nO%K-s$Slif;K(7%U*MQjEMukX2#2UGRvbC{15!jd z2+A;Opv8DZ1H)l~5emr4VoD*Hbs`HS8qp5%ptSFk{WJkgL6KjJ%~B+93l)}8CAR{L zTcjz<`Z_DvxvOYL*n%Bl3wDI{b{?w*@qhGF9^V0SYBX2A&9bDTMY${pU(o5VAGm{} zVD{8UadfkCaLwq#?(B8Sd)|>>p@NWXMs|PI zxkqQyLcL8P2e`mJjA{e-H7dJf;T45_AetVi-NJ=pKSLg++a6=8+rSh`d4Ij{=YL>bb>l^#Gm~F?vL;v9?8)7S*5pBq}U& zQJxkR8_$gH=L2{`TrT!B%;qkZ0mUi6U~oIKex^k>r8c61b$MAgQ>|fLSc2TJE^NWN zum$VF_OkxUv^xgAL#ensrbXo~4CkFa_Q!k0bNfbwb$GRM*zJwLP;Gq`}%#a@7pW1l-0C#8po%t(|A>rxI#_Uq)%#1dc)OD7jG<2 zU%WBB`}KT1cYhb}KFEAJlC(o9Z@@vC9c;zNCAh z!jgWzuGf@DVM*VZiDXOqr}8CTE~hcE+>-uVxp3yEj4a0Dk)JH-0v6+^Hthly^~qeZ z4G&)O)`X4u6ib!0g>_565DUwthcl0%FXU26)hH9S=P4fr;9h7TY#@%JAHnMwrI{U?RtQT*4S8D64 znyeq|r#!q5{E1N=;4h5Q=U~!cyrQsVqUq<_t+Zr5P`{CKgne11<5MTgc(N1GzFDt! z-YZ-E_RP4rkk`|DPf=1ZMU21EJC+sB3UXPsC&pqq&+3@P^F4{N-tDM;XV3_BW_!Yo-Lbh z)q9tXySZC7aqE_QnYASMF4@7aDJ`2V0gzw zVZ=Ko3M1YzQ82t?qF{JOcGtFG`1X?F+e?OTFB!hQWcczMwTaH8?yqxN2$;SJvxmzZ#pMUa6b*3cW6rH^eTk>aOtH@~%)* zYoaR+tx)86Flepy!4Q>ZCtaYZ_SA{coT*P1q%AP#>6V5%DCSA(0{va9NTY ztY|zY$pMc}atkUNCnP!GDM@Z&MdRs74%kg{iz*u9Ne*~+lG~%A@uDOLT$kkbtY}=C z0(`h&pTD0$G7XZ&;k5NXVjSGOS(6E^A<;9~3T4>*xKNw$b_4 z!Z|#U1@;bGh#LoWca(Mtca(Mtca(Mtcg%RCAyp2gNjAAMzw)^ zTkg-lq@wZAB-e6(sG4sgLqK{<){?3NlbqAkLKUva-U5gN^5&dNs$TuDD**&zt#-lp zTcfVL#9SH!QuGqg+iVJxfSM%qM7butg?Fmve>MwQDn759zHB1d^Dl^?MRp5wvmpFL z`?HUch2oBhfeQ-n14MFhn$X^FL{KI>!kI;ekX~ep+rYh*s`E!9IgmvKRY_#TJAeo# zLba$BdKIgM>Bi2saA=T=WW%R7r!?{|*}O0jYRqguid>|psrTD~!VvJoO4a2}1D6T| z!;vJCU9mv~6QTu3ui|9-$%;6%Uq3f~#tT+lW|@A$YD(Qk;q+^X%&5p-tA{-w)X&KJ zFcz%exTz{w7q(zsSeD#|bzuwEh4t1Ss6Y07rd|~4YpviAg1xU$oEQ@!p}d1aVUbf= zp}e_6)!e=MDGzJ`e`(Yx9-M3^ki)qL!?`(qp;0pWWYxLVbhpvHpXqMrin2+KqF-8= zR-)(&rf6mE-MwHRdidbE|W6t8;U!b8{@W7AMT@c9@={ey1?4 z!SBzS-)nNe*W`Y$$^F(P(%fzprp{f(sUvJ*>IhqyI>KVQF>!@Wb|d<=R`-QE=)u-h z|0VrXeN%lfOse^^WkuLPFdn4ANL^rl_)JtL3|(wPH#Z?@Q)P^+9dcO%INnw%)aA;K*g+3wuis5-GO+ zsY=bOrTTl>xk$z2wbnp1Jg7BIb^M`?UV45vxguIvuP3WdRZaR#)uc~UP5M05q#yQ_ zh7H59l)UB0def2hwj=9}M|Kw0{A&*S;!4Ot0UPi{1ETzxX(`W9Y^eAM) zmG&;<7Z&A_GjB?$nR-b$ZZGc7ge`DgG6V`r?+Q;#rt8K%$+`MZ8aXbpr`d$(Q;&>FKHDtqR13QfbwJTa{k|6LW(*$W9 z@}8nAN9mIQhwk?hFL=SDp86M*6(mLdWqkTI;Y zj8-rGD3emlHT9_HnRT_%#rh%V)IjnU+|S=U&bX6bqv)bz;f2RYQ&18uAiZO>&_Sh$ zAdoVNf_{uxK%K-Z*JabXsVkqYxvqAut6lOdk5{g16kRy6lX#`k4ClSF`%moASW4wF z+2>w6uhn7+Ab0RMaF$L-cpqD@H@x1#>kORNS9%Quy4VRwo`AR=65PoQMAn;#taliV z*Xt^5?} z0CBog7v@#2>2`+AHfY$=Q;(B72l@#a3PDI^xVlL)VC*W&7X@6o#O zSJyAt))#*W)(>a;RzE7FDWF-`5*T}c{ejbatg*T<2%<}zY z;q%>u>SgT!&E7(JqKQa(*?sf`@7Z>uH?mUU*@*&fG-~vf3g4e7;65*kR^;Id;KP*S z!ByIiHGZx9OzYB$_WP5H&nx_*QLBJUUTlU=RoK?gIjzh3DjF+Ht_^%alIyQ%ye!E9 zHzc_^6^$#B9Pr8{H@Bj3O_Bq?Ey>NRXuK!M0Y94L1}Yk#N^-!@C%O3*jXRPY@Y_jl zu%hvUBnP}F$t|d8{36K#2VN3GFL7R2(O9IUOeo+(liZ?;#^NLgJS55OQPDUo$pKGD za(h-Zo|5E%&q#86RWwdda=~ zZhl4MACnyL|0TJ>ipIYsIpB|z+=7b6jwA=1cW$HlXzZn=JW~gJc#>OG(O8z`fJY>` zJt`U}COP1#Np8=I#!!+2u1a!yRWx3ZYEZhuGvr2BCSRU4yu6KhCv59)KeY45OU>tOwZ5OYzG)>bR>HV}WJ;LNUc zO0IOwm5rKtIc5TK7!xdXDq62ig21LL`PqLSm9dg5W9AA6MslqJc!~x}9(lNVk_J@~ z{2Qxik?2~K25F9>w1{pkN`pm2tOJ|l%deW{;~OuHZ`{Urr$ulai1Faxcts2Q@(c&C z>B@L^ALlgHjD0!Er0|I3ZX9@Svua%dX%ghc%UX($PTeR)e0Ye*(v2Cp7mnx z$2=h4>e?iS%M6@_2%a{p(0q}XrDUlc;jot21_+8tzTnUrsJr-)q%91Hd|}HGZiStn z$d}@1aIz*}o#?Mt5 zC$d8Xrn8zc&T2SonbW$YucE;zije&k4bDe|oKw-@fr$IzSZG-{WVEK@vtVng;i?dk8SnJzF>H{ZrWlZ*%doP&>}|_`F9ZC zuKhV48IUg^UnXm^zd#nAzV&X>7umwM*APLy97G|_^|rzeDDO3df=}6a+rYn5s{ZOA zjifz$L>E;_Br91&Fd^q7KzbFws5pzUajeH`Ij6NL{}v)bjn&l$KNrcqbcV>ax^PQ= zfK^({Wm;XP7ZmDDbli+#BcBx~2BR22YDRs=x2lI|&rH=WD)_1bt{Kg$;>1u9qmchw8 z&N3Z&$5|G@K3X2frpf`n+^AvTYo`?xtzWLC>av?_x%w{Z=@kkyuP-_tYEHC)uP|y9 zh$lf;yP}1K!i{~@cz{Ln91q#2(3F_!zW zy5z^0`SEn~V>J-V!H=YcPas0r2oFHgw<5Q(`-(x2Hs%Q5b(1`t@xJ0Um7(8e5ft6D-Kk6bfSRA8#M&{ zL!(vzFHow#wY~-3lEi?2)(``~&BTV7%W0Nm3kz{(Y7EF_s*Q3rg*Jf-4~vY+im=qC z9%~srXtTD2M_a~%DhO7XdPBJL4S}1<1`Lj4BQ4AkdGDp-a2Vx60nsWe0T2TsHIwde z*J+{=8LE7~@Uq)kT0!7J7*4z10)r!{qK!AE3Ig}&Fq4lmGHg-KLIFv{fMoc5Ig6r= z8v}BiC-Ai2VKqJuWJb(sT{@?tk*3G^541~~3**cMN?Z0dKq^~KGjE=xL2Zj1kkYQ@ zXrwwH_u-N;`$$cwV^jJ$PhK21SN>^}AuN=)7lz;H4%_^^Zf;{x%XgIAsR2sL4r@Phb#Zut(Zzb9e=2uya}q z?yhK%n$Q_cjtZx5tQ?5Q^OZ-xb)FwUtA*U@nzXPgtd)ADP2I}DB9IXx%JA+q73bF~ zXp?g}r3O;Us2~~#X*A#&JUN8-`GNQ2V|2J&NQO|}x=`M*P~N6c-dwK;{kDgVA2TH% zxkh-rtM_Gi{CcsU+-0aAjlHOTH1>k@Xe>*}Ra#Q}${Kq7vP_K4*QnS-8)9Ly+mo;e zFwp&{Tb?HCBd{}BUpYIIrKDK1dXFB7VVw;-(~8Wf$}X0-{MBD6yiAs?G;UbAB#4PW%f?n+FC1yYrm12gbM``W` z0WI0`(?s4g)1 zm7Un-ezo!I1~az`*z~Ji@T;BswF&Y^kwSw zoYv&Vibm7x{2{UZ)dqQ&@0PZ3_c0cOtu1``N`w23mf&~wGxYU;=qD8S@oI6qx=*Nr z3%E+!N^a4Qz)P}#sERp286g%yAOW5hZH&fT`DkDk8?ydkfR2SF(f4W&@>z|1fN@Or zjlc;N<1~_XvYK7v!zyu^F+@*Q8zKly>eu*CbX7Wu40)0NS4eIznHjz8)v=KZFR~lK zqiY&n7JytNJ6vYcu(y-zjX>sTsIzXxASfmg#4^*raIF&c?5m~pfKBxy1@*(Z`po;c zxur!OVaxp-L(%EBhZtkekX+BcAfD^)vFFMyuEZicB01&;VmU;0R{Nyfj~<5rxgV?W zgFU*O`2d@K%xdS3AC$*^Tq&rhJnr{OS((Sn@^*q%%1Yq1MhyY~+Nc#@QusrohJf?! zKy=043gv6e^`D$l0uM222smui3gB5vg+m7LB}okUnuZwo4JI~3>QnJ&2SQ|neRjAq zAAUZ^#|)lAqsrlFGxfqPmQGOQeW1htXYb7eB(18u|LUHZW`=!H)?ph46%l3F5m^En zNnGfvu4=kxMU4>`5ExdOEx4iJG7cKz_MoC7ZkV_bjS4PN6n#;n#@9qKf@0i=D@F~! zbDw)Y=RSSzt*L4j#rK!~qw9OmbM|xYefBE9Kzc`{>%0$*m1xS+KMBOpNNW`^#ho33 zdvo01AP+QuydlzFR6$@h3-?yKa=2VdRlH3@s+fqFV2qb$0s?L$FgR74TMX%ISu%kX zX-rCeZn%7{*(8<13jAA3^fZuJQQv`KkfzA=H&i6ehiT>m?H!JMv`y*m`pyai`dj#b zw0F(NAocpRPoBi!^Y$Ed>=ON)CqD@#w>B6O!j^J!z#SMrx0~AtikaQq2JJA7QcW>D zQprsbXe7qpfgv}i7i4`t)SOQM^(WGBcLdDBCTHosfPsx*7pe6?0~2h02ZjNq35;NB zRJanr`T>jFfmwQO;SLOZCu$ADauV*qFs5(i`U{WyD#*r8 znhod#PYt2_`at&uB05|y6hlbfx{$nKA$glZ^5*(w=now>ehex;e2w&YSMSU8`1NAf zR(((9sKxaiTIqhfW5N>hLT#xB%Qu1)*w}ojvhC7fn{g)r1_IrErsWyDJ_DQM^_{ah zURsJZtAF%x9ILZob6Vl~bLCw~WW_rwa50|--GT8v-`{6H6@INHVW!LZE@83o3Td#^ zQlxNM4*h^AD&0-+_Tycac6cSBLM8IeZG$~p<;CRyX|U8pE;Zq$=q5Tv5`>EsqL8y( zYAI6e>m_GE6qRy@rk!mSUR%E}#dwj@WlaF$YIux4q&Bxa?k&SQIQms+!|6h%{V{F0 zYr=sd$9<(QCGm1{qKOzgjeB$MiTZSzF1htlyDW5g1!g6d7fe{R@m@$iRbKm+tu=Nt zhbXuh*xcqX3O2WS8+)sl>9WOrnxI_$FqbWmI@m5}b%}R&8AK7$=~oA#jNJG6MxlFF zHKgu~{`!q3U`JhM3kM^Aue0@&`YGR5kl$1KkYg(7xl>uzoEKX}t`Y+;Gtva`)k+Fi zX?hi-tuI72Tl8}EUDRK0>L-CHjF;9Jz1JAKz+^EyRn0B7v53N;zHQ#GJ|FMfz#H_Q ze`I8ojJExB%8hnPMmuKoQx;PP_yr?P1Ak|v4)ApIb{dF(!P`#7=q`;xP6~l3fX?6m zrb+=!SpcL;P8@+)4t`9TAIWHThWX2^_x>nG|HMkQN}GcE$Q}g9Jd#U4z-G^Pt$NHO zkp-qYWm77*c%QBjc^llP>d(*e^#|y(+~}WJm*oNc0hZC+=>6T=xyAdr_c}~SmVJ5n z709wLOE!>YU$}t1vZZla1Y4u(to!31NxG!qbT8-YQ-%TWwIFsFnB z#2hyO82^LKv1uUFufE#PAkD?;U#du&c+*Tg=8<$fkZDw3#b>}o5)Cyw1q7E|h2(7t$(!q!p+9uk_%W#X@HNuoUA-^UKgoO+<)SNLv7`%Wu+%c7a9MtV4~U}DHE(afc`pE!$TznQ_GpzC zFI>I$*5%oqTq?U{MmNzZk|1295QUuOQp=FyC_!=tL{TYcXgWfBDLl9sF8(tGMH@~RGR?soNAI&fsp4>54yu4G$oJ;5n)fT*tKdxHYkFNG z(7-J6a-5EN4rPVpbLF)+zeF7?KUFKuZFBT`?OgqAZu2%irt%N8%;MMQ)jvrOAI{A zdcwaqnaKUHHfZ{h+93I{GWX-2cSi1qO^c=uvpC3H-K^#({SmY3YM?KXtHv%B%)H z(MU_X3ZIimz_W}r4t$l7mIALV)&2@U28t zi}Hty;ql9dd^V9FrR?CKGsc;u2@38Z;a5`cf`hi)x+jCT7xK>+n}vcIW@)bl7c#B^4kmBpfR=m3j;nyYR)Hs+7W3uG6A!8$(lv+ z2s*G4?6cInKm!wOeH(!RB?^pSY9u!Lxdw~eM!4|(h1&>(aGBOH+-D585g0S8OGcC{Hg@-&Xgtz#RzE9!ixqWsw&rJ)TrE{%+zTjZ?S$z21$St$r z-Kur-KETEN9dvWSPpJMr`+4>l^@kNlc2~HH(!w4rms*MxF3SV;KopfOlzUC_212*~ z8YomE-`qAhj^$pwaKYbOms8eUD!al$KhY_Ypf#FCqzgIArIsS~DwUR;0Z~-S8JbqA z7+=`LnEjzFEI?d^HMczOEyL*c1(q%qJ32UDRyI^3p`6n@p>eq)rk@K!x@ae)p83xbgfarF?oOPtniGQ zj$das#({5AlKi^F+KR!~65sz-h%dv1iX-bxQ5W6sNu+Nn{15$z3SN?Nkps>!`JwyXqfBM-i4^5S^?x4VAIJwf0eMw$Tbt)wucdljP<>!Xgs)%9F`7xhTT)q&BK zX0!`@ospIUKi=@8TQT~&F?NBA*W%i8;A@T41%B2@%Yk1tQWto@I@1L_#YkP?w2_tr zR~e}be6Ep}1J5&37kG)0mIL2tq%JU}e|dDU@n17S&gFa_NW$ZB2_teYj6kv*85w=W za@_`!;K<16m6pghkQ_%wMqf#cK++r;8Ifu65lE&ZBcs2zT(yBDJ2EmlCouxacVuKl z%EdI0>_^5%>Lbhw&?yz5V{?hD%T+-jDGwGp*6iQ30(5|+JTfwRqvf>&B;}Ej5q&PE zfuuY#G9v532qfi^kr7!JMj$DVjEvS;0XjfZ9vK;tb z2Gm2es|Z~#;M0Cbn|G7I<1CET*IhHh=r0o^@RS*hjGmPki45-wHx6>w{&$AdYfLD4 zr^-t{yS%7d3~5I)XRZscKz=`PK(BslXW0#l|1|hObf$uzmFxW zD=g93iB!&#T|1yAEO&(!i2WbSpRfQ~`{XN5WhI=}g{`r`d8g5ZpzP3)VprLBuQ^y( zJJ;3c5NNQjF1k2&m8t+)whI@U@&G4Ll%%pMkPdbnl42lxjhJF|&@zQ`p$kFDCsOP+ zg#<)VTupsia$Rk7al|Ocgute*{FfbbU0rl>;4di#qN|YNTeRUh(vus5KmM@X802_@ z5|?xv%?W>EXN z!cnwbYrX551(kP4$!}D<>?V{vNU2=0@Ct?Gl@+dQkf?@tH(WDP@-C3Q!amoHf23Kb zhyGBh_I78NQ!l-BYxVE0S-`Jm5ju@AbE|KVnOl9M%-rg0izQl7vIoTYz3`6Jt2A5j@H6*0VLegVO_`V(w!zRCyS>S6H-5GOV#S;UIRHz=irU{DVP6%pEA+} z@Uu2!!{t9lPuUPXC`^-FeHZmehZVhRx_)9tyTD(Y(eMrajJ7hPUEo1VlCMl!Q!&7~ zXeK(E{^$O+Oa8UZzirKI8+fppJ%4V+plLQg0{E(igbHaHAPVNRX045c$-x7VY)Rwa z=eZ+@`6(qpk}n5kK;jBd>764FB{3#Zh2u925X`{1=9(v6;1i6r9N05b4~P?zrYo-* zX}Q0hRbWpO90wv;n1%nYHKqiGWdW9f0Xm0T2qxXxf%Lt|JE9~^jy!=ROqK~CDJbXe zZ!~8wNx1`#{Y?x@mJQ&JMj8k1WTY&_H>%Eg#xf4vLP?`?SJ3~xMaAyY&(e)P6@P?j z?g5`{qzT|DN(xI%{>$+PI9;Q1s}f}go3ftwa$Gn4YhnR@TS?I#qdhK(Dx@^PbCpzf zEVs}tS?E?+_#bnn3*5pig!P5dzUE37c!HAZwV6TFzx=1yoUfj_!qWgpqNbk*}5iz@K>MtPLWbXwe6n1os39YcB!vMhyjO#(>Nf)@%NXvnz z8>tP%iSlB7fC?OJg5y903p=_WYmG^IVMhnczySR!h+xv4?FH^d-Vr5XvZDi%FbM`o z3d*^AJ~?|n%kcP!!cjhD<@bNv>$asP)&t)90yIqk|5Hix!|ybY%|C5n1!di(#@x@u zLNn0?9%H0&AlBsxHsBA;kySv+zXI4)-!7aZAz*|FV}3n)v`9eQda=nsQLQ& zv9+3(wd~ebg(DqxBvSDO!;y;RU#jIu{k@Jb9;t(@i;MF!<2BLSu8H1sP4t#)Vn4$( z(!YAhT;GetVq#_rX_)&gh2Bh^18-&a**e44>TzHrfw#2^oq1cUkgeN%IaqV_9?<~>JJjD5$(>sd=)Wv1O>zlcxps>88c#X{knfVoH z-n~HU=N&=InY3qlS(c)V`2H86W{| zDa?}cvB2I>F}$TfF^=9sptowEztvAUG$w@)u|v(}-Z4l8)7+32B2u)&?PYTW71Hw_0to zJixMfTm2NjSQSeG_8B`*5&&V7SYD8MJy+GqRb|Vrml~62*u!hPh#sh3Kxd3gvG0P!pwPnC?}oP|(ELc#9O;bF zTsLJ9@QUA;<*bKJWD9d~gYK%*$wdeFLETi(MJ7E5fyD`mve&4jO&|!UmWwIgePl0F z+3-7L>m+}SU!`=J4?s?gq2C1c5E-X5CDz4dcmYw4!ZO`{AKDS6949^W)_&YVpn z&<@jWJL$IbT4RSSK5xG4#Ds~B$0^eSWt!fU?Kw|SmXq{TQU^R)N#*O^2tM8fyTGPk z;Xk$0|6O%{Po!QA3ARp6l-z3f0vRk${hl*vAFsctTnO@>^mmrG@h zppVbUH+RH8xrqX-lq5Ggwm07tz@rOKOT>T;F)!7E=T*8z#DK3;(iXC6%Kmf7LNELZ zB_J>+OJF!YXN-bvP*A_SgEyw_L7MknE&t=B>1l}sJX=Y9z4Een_aBb2=!@aIaBZy6k^80;`7@(FQmwM+*`&NX4nlY-o_PVU&!+_4Uh zCEZK=>E@_f9&QJ|OCy{suj=nzF~Ur|6vrr8U^1Ga#hQYnEjufK#~En?*fG-NCe6(2 z^;6!t23~8V>T71 zLJ)`*E;N2a4SzQ|V0l$=YsQkL@RaCFYbIHr zMq;OK9(mnizIo+gUZNd0OY`zW>R)oI^C-2P#PT`#;Y@5ChY>5Z>rI;FuUNJ_7L zTCG03zl5~R3#N{?4d%#Z^Y5da) zcb*qrmjnOZ6ux5%Bl!T){Y6vM1|l5}1=|&)T}@XTxL8S%kx??*?q`%Jdbm`ssl`_&fR{ldXrS;J5?_yrP{)Y$@!2k@SYm_ z!z~3_y!&P>nU?dlC*NSy$!7lZRrnc%l@)h?%qz)!k z`6}w=WEH*Ii4|7S6P4v;OF?!j{J4xITSeQ-Ldb;^|C}W&8Z6Hp3gt|*FbzB>WA@F-oMB=VCTY%+P12kt%L6RyYzk$0fCXQQd3c7(KgZ_XIXX$9 zoMjf40WZjyeI9TN+l4&jELk3MmMjmjyvw}I@&F6I6!Y+Em49s(8k9Ghg=N6EX3Rbh zxP|RP9&(l}4>?Pg2UxytUS@fK1z(DJ_@v5zHVX~P&1PX5cuU6Y^KhNcoI7n&^?(cJ zN9K43h|xl&%3(9`(+eV_Ts#53W&5TqqbKcXjKEJTDH>t)xt)v=_H3&eN14hwwwcF}~~HNAcv1EL9@bSp*#B+@{lsiheZkMIGTq2>3l^3d9) z(ApN--WFOLc%+i#Fzk61gR8V`TeLRbxpwS`g%yORKlzITxj&tfKOOT2d$LahHZ9~| zGn8AHDp{B+SeU}X(=6hRcdTX5G@D-l|7bQ9{baDF(JhBMz-1b1=?i(kl4)9wrqx!7>`*h8T7}fn zwqmZz*N;^!70hhw3`LxCR5bhI;j4gzHt&RP zACw6>#`F0X5+;wdMRel{Pu4XqEk1U1$dQz+V!%~;z$nW6;gfK&60!^m*T3nC$U-5c zLB6L(@gbxwL^A6ZBFu}ojBdzkcGp@4tGA9sIp_nPXQU30T-2gI3pH_2gw0;`F*djP zC}-8DMEb52>4tR--fCe@1K)0>4zLmFhINHVH)J=X{K6Zfxt}Kkk>ajgV)D)k$rlBw zIhvnv=?FZgwD8T+>stF(`|9{bdZH0{y@}GLC!dZjTAM~!Y#P}#Le-br!M$#Pu%@v= zo(SAD0^#M24f43*rV$7nGAs}q8{fY!Y=VTVW*!2^6b%_j%w>~-z~l}aYR@2WU}6#w znEMSYz8F4HWAejd;fXb$C+I#+;TLoQzt0lSnB_^56veClK1rC!b7UeuM~L_o;g!!2 z{CR@R#GnMpg%E$39!_-c>yxdcbWiaNJATeCi=3{TUf7@IMY`!N{aLQoO&{sc^7p#w zvpNaqwl?0du4436rOLC98+`ix$)L!^8_KSdDuk4BmX{*#i%O1|20)vj=!Lr5=(Y-`Id~O?IUjKzW71BGm_SqUt|IKjk_Y zuxq5Hz*ChJ-WIq>s>^t(v1YfGF@TCtT`=OIx?ln>+?hI4tt6JLm(I>uveG?7vvh)$ zF4M9^HyvXvZ6H!o-CMKj-lo-^XO{~%ZdhM2+Rly%yTBdQU8#*Q7OE>$EV4L8-q)&Y zjhYK@q2@yMHQ!phXLtRSTQR`BjnoD%Qc{@F&y$)neu%MVw_q@UicoVf;-ThXb|G>W z?R`B)tsH08IzXhP6#4t|o~ba0zpg5oOa+$zP?jH=^>H9lVtvDU@AdDs5%x@taMk*X z(Zf?C02_@Uj6}x%Q%Ww>LL&(E+Po2jT4)5JdN*%YN4}LD0)E#>%YffgQoPQvuwwiJ zW6dTK22c?i0gQNP1TbkRl;Uo+GHOfBI1ni*#dTRJJX4{^I-0^&Hr>a89H|sld!~YF z4xTwMmP*RL_+~wY;6OZ53so0hU$c4JM9M<_lvDxtGEy7(2qSd{DGIeg(*Y_LY6E_v zscL&wN9v};l0CG2OvaKm%Ne@&yCCfaft*iB8E(l|UC&f_tE*53*>B?R${9OjY&D7_ zyr=65(3p}!^@%ul+eq1{pR%$7HyNo1e1nm)71S#V#X-~GsaPlu`17VJyQz1bZu&=K z=>d_FwBDJe^&gZUrb1e8RhA!{>MrnZB^6bBrh@8!)6=)c#M!XkDt5AJkNZMwHR>q5 zg*pnAI$o;vj2hmaS66s_b??$1-q*s-W&$h^R+dmV@Lon+Qh^Di@I5AnDEr}NZ7C2b zNp8O9^Gt>3A9mJ;KP1gGAM3suOV(I-<|Vo+mFRPtEJ?6XvD7G$PzxmzYN13z^$xC6 z{a5Q}p@(3(QdvSVz;80zk_wE=g<^b2**|R7mI0BHV&r=t&s2E3eUSWYc5KY_alM9AxF_V@tDh|Jz7-!gI9`=L zRzKx=WPSVu%9G4&_Hqqo@vxCVoV{<-jseo^wL|wIL+@7(^h!Bor{%Rqs@EB*UL&$x z_LZi;D|;Qm=c}#=l4NT5+Woaw=Gj^8pqy*9TM4{SN&D0{eB`3n@w>})uf_HIo%~5HVraB^}KfIUN!VVdZmIJ)>mF@qNO(ErEjTr zekt4Mt1c{(WNP?oU!jG&GOIR}H(9kOfLAMN@0Q%uJNv#VHn#u>p;1wvy~ZY=LZ2Oq z=e0xks-X|kD+QFDq}Lj$UT36wjmUo0(%bh|0P9IzRpiLi@TI;%EqzKq(__j|zF@v& zr!PLSjT+pgQh3zZpT-kY1^VEXrPMqNO&}?^bb(+?Mk! zKA5^l$dRYvE4Qs$+EG7K<)G|gzRad_^d25H%Hvbf*fc0uD&e(5_o|@}(ks=tVO{05 zMyl5tsa_+pRAMv7JTvATa^z|F%6(tw#t*YO2j!RdXwK1lc+@D5Peo&s&xdbQD&e(5 z_o|@}(ks=FU8C0;sa|KKdX32bh4p!M4)24hi-a6`8oqKzs-Z&ZDOu&9 zJlA}gP34FHj~eCisc3BS`RG?Uubm2c)zAm&m1@Xp>9t0x*BPl^BeHL_Y_0MvKA5^l z$dRYvD|e|{x;(2Kl-HRr*_|)%OQQxvfJco@J{65kgMy_JUORNJ8u}o;QVlsm@meF* z>x@*d5!p{!wuU=GL5@5PU%B_IrR%fGLAlXD34D?W0TK^Zw)HpwL|x+ zp%2m<=aJVMsa|KKdX2~)Wluc~H|LO>edQjimJZa?lc2_vNYT3-6mV zK)RI}8U^*qYi#lv>{n~Aotk>p&VftTTAjD34D?W0TKEzsh;-RLHA_K1i=rL-veb zYovOek?J)fJKt_BOmZWDyCaQPu(&lc>={1V$>#9&WkH?H>I>zm_o%*HIL4z!5q-iM zn|#*#)z@pMj$SqNL3*WTvZi{ik?M6us@I6@_bpq)t*OY7r{ODisam=`s~nWqm@l)L znOp?MqegjrDjJ)7KKfP8Yo|h9HS|Gxr5duPdaaS_bw;Y!h%66l47;WxN1leS+()!7 zH)NHAa+6hV3iu->?bW(i9?Bd3HpfPNo@SGNE|AB&&Tj2~R%`d{R_lSv@)rG+e94(y z;SnRD`Yaxs%gTz0>`(NG8__6KFWbvyWi}$q8xtZcR4;o*E-U@uxmpj3loRw*o_+*Q zD=EI9%jmg@5x8#Ws6+1C0(aRnlA8 za3GEc!EVK9t%cDA;=AlxfUSkmgxvO#UrE&g(L{xe2uP%XL{m#MARgfZHbcu_7j9*t zwM(J3Ewpniv^MaiN~&KMX7Jkw$Kd2W%6ky*ljpR6O$+(! z!nuX1l7*>)g()n2&?4S=-C71sv-#`7+dRM$o%=IE_ddf)JOL!|((A%r6R!n(b`Og&|<9|LeluWc>!PuOhoL{6KBZ%z)yF6W7(R3qzoqXlbklHQXz!=TdJ-o(#gw zdz8EMx^S*(Ihy|2Dv`Y|ykBO26;iL#sV8?k^7UgCOU3VXZ4|Bt?-Qfz>1W|K$uXI^ zm6*HL%&pALt;F2crh8>>Ze_He{(E!vOTncAhnIbEg!QOqW%{GIDy$gfhATB!*Uz#jC5x13{&^LtpI??A1V()|;_IcYv z!X()Z`=x7j9pdmx*uJdCj;nTwON;yv*GR?Shlw~^F?iap#yO{Aa7iMruUI!(#l%cn zw<7W}q=xaKq%A~5zJ_ikV*I5YC$9v))FNCrQZd-lf>}3OF?d!Y&Z!u@IT76z(t3G6 znfm-u2E|qmSsOf9=om4c+vs{w%-NO*`1p z^0-gIJl%AJUJ8iUff${nRJr&7q>mz_SGP#7hdj!zi*$j98fhAMgpzo%nDNoZ+U;im z6=C{<5f3lXfLEBREc7Soro@t6IvvkgvbRGn(^Hza+Nn*Z!YE-vy@$Q4F|C9PDm)3xv zHPTYx=as~Z#f-mbtUX^{44@)Z7mRqQE|`D|FBboYT1hNfFWsK8WTpF%=I%>Yx=hO_ zbkoho(gq?W)y-clezaEiIjOpG&6CkZsk*=ysk>4!VJuWvs91b(s;*G?*HNS9LM_x> zsJtS;dvtnwa^Gc^^I_XwqHj-WuE{%)kw>LOO?cn#f(=NYqy^PRD?zVBOV$7 zOd1NMSglshH*4cSq@)!2t6iR{&|}}#>fV{E4CE}WsM<3XRCCnJ5v^46hIQ4$^l+LZ z=twP8U3h&;|5|0Aub=X&8t{ci>Hx1WQuZpER}^Z4rdO(1s15jIrYc)f-=>=q%cOVv z+KeUJQ+!R2_Bra8U6I%F9cB5ceoCJJkt$spEmT2vg?Rf3RVGysDta2V5$cb0`Dx5Z zp?deewR@z@H}|p#VEMhWhOC3GZsS)O8sGL};^ zmh6z}N`Tnr zG!Uucitd>T-+uoNEd!}3w8956mT8|3RwP!L#-kX1H_+HLNDDWayb9=pkZWn9@`b~OuQO7;#zQnq zCs?*RKyFZs%9juNKcB(6aNdvMtBDs!X4QmplKD5AYSNJRPc^-Es^&8feNYmmLbB)c zS|io#j8w02LbLQ#%T@=-eV0-BU0nZHtfu#4_-f+CR8~zW&o%$DgBx@*dalK~ga?4f+$ektGMe0w@CSjk!x^Ui);j4)kmzWom zKzb@IzR8P-yf2Mv62kpcO|PA*`OHHf9t0x*BPl^s>x5zLiw5bH=F$?%gg;!O|PA* z`OHK2xi3^x_M2X-)J>>fXQXgKa{~E*$u**2VO`4PRkA8&}WLFIGdDG@rBk6yBFcg=y0Jr@~%4 zmGsqs?(5+lm#WDg(`$`XuQO7;#tSt|FS2a4fjqA#d(7eHtM_C0YU0J3teQ|>V*X{f zQhZVx)ubWspK5yTRLy4|y3f6LTq-2{Mz1wez0OGW8n4tWz1_0a2J%Fr>@nASG4DuS zIPb^s)x?X}XVrxA4)bp|`_eb7)csRUubry-%tIfPM5&PMOTE@e^*STfYy7yz`(?{k z2lxvm$-Z>BYI;A0uO?o6Qcb2ECzO9S|JuOMD@nfKLEeLrr}~=T=eVT)KF8VbeQDwQ zr~Y0$b@sJ_?yKb|kuUo?VJA9wu+1~0|>Juo(na|k~+284-P`l_; zJ(3XF754UtJ~tj+6snKoeYvbGyvSapIz*#Tz3ff7?1uEz)H9N-P`&JTa@nhLuB%kX zd-YSEW&yt6)c<^>m#M!+w|+Yb0&m-_;GMd4hkaZ1pWI>x?x-YrIOZ(yPKkCp_`!*G z8hDpPn*#5aXyf2L676L0!bCd(e2~(_J>Z-7iyjB}B1-@BsMz}9W+v#V?2hX#G_$goel>fY_XAH`wR58v3 z8hFK`=#D&~0erfW=Cm#vtr%>hoh5ubDtt&H9jNeN{S*tQfuEe%#uZ+iNUu>yf(j3c zP@^YUT01}@o6|aXUd14(%8reDq}ZZv&r_JB+P*+YebO8l-(RJUH$~&X$0}(X`NgO! z=AP3!C43S0{uSh>B#CDz{CpyPQ{fL2i98-*NgF4Rf0anUGnC}4Lq2cj=8M!hs_=%S zuwN2roYB@{75D~AOotp{P@ZN1VtY>OoQoLU({0U)jo z>+5~%*mW8QpK{Cuyg@hZCO=oWV@%fZJ3lCbgB~{0+Igo6mM%;3)a;wVR zrk`RNc)OCq&N4q*yHmIByEv*8LEsW4ML|YiNQ}Tm`$yS1t@GDZ3|1;}ShL-d*=~hd z)XGs8uxX*29fszCwM!P-X5lmotPRA%oYtEB0690%!N4O;)e0bSgmiSwX4e=yKx`Ji z2g+NV(>iZ%g-r~E)4bk(bIk-rs+k#(LRl_>*hSuVSZ@66k)rYTej{d_M9NoDj8qJ; zB-6H^ksUNI^IA*RwifrTEo@uVY)ckat5t53<#il*rIO^^dDd49cGD&;y!mjZ@}Hel z0Y71+Wk6z@FP^LqzW|c?j6PfpyHM~NQ`iQ+KuLAxUZMlsmM6i1*EaclI`eY(e7U!l zMSxJWjB~m{djXBJAU>o zTgX?iZ7I9kEXNyE$Gc5e5BPZ{mDgi0Sg#gdwBOS3PG>bF7TM35C)vxcx9Fzd7)uYh z?L%TK%Lz>jyqnVM_XikkaX?}O@2Iqzk-@@8Bu4Nfl~yw{c`sjEr8RRM7~0ibfV08T~}5!U%la!J8knRXZrwY9?>3c=Ky=#p* zM}Nrf9C){8q~>G5WQbN6(ML5a13F1m0%@a~j{%J&e83%TT8WDcm|oE|BO0z&$$*v< zl|cHf=3_v|2_KMltN9qv3Bu>wAaAt)Et2HufnB5pa$o<7(R%;KTxCkiw*|4uml{^) zLU>8hcWj_NTD$iZy5I%c_Z8D-f%eQtMGJIUpl#H(kB$qpets}fr7MBBR(G%WttR3e zQ>7*{V6qIFH$*0UJ}@EqxM~wVFyZSL7p0IRBp`8xw(4X_k5nSVxk&Z?MXGl(QoWCn zI@nFyV>I1t7u{@U-4Tqmv;K(GlAGe7lDuKi?C9=KTmX??|BeZU{90haWG~eP!SQlWjRO zyhb;_&ki!$z~9+kR?cr3P#_T{s+|vvw(GMw29Hg|(Tc$nl_iR1r;FERvm9W-S3|OCq&-X;(te=*k{BpymHMUxA_&}hz^TP(2REx41 zn6}WrWH@bsLTbzm1QL`)ppY^XA#dc!HjpVYkaYP(m`J}FLi7tX9fvN=60#);k!`d+ z&DY%KlhWMg)6y{K6Vnv)scDG$@%Zel0tEn?7deqo?op~K*>wA{i(N-eJo1$alB9+X0@Bw>V-HMrZ{RcaU7*+5S6i>hFTwXT}2pkMsd5x@Km?Ff~ir&Qlox+pQp zluRYwWRyXb zD#e_MR6k`Rl{zssBGuO;QhhnRPh+j0?Qr;5D?4I=?)`%9J%jFjgYLb9?)^Jdd*PMV zE^Qz^A*V$xJ!8t3SldL)lTg`X;xp|pBp@@Pq1$`h*woMI2U4Mr{ZmXNPLs&`xi>#y zYNGGUnn;1EW=-@RSrdJ6YoaeJJ3|^!`UUvF?M0j>g>f11IgC`F#7Om7j8vb-NcDM) zRG-L>UU7J)e#+N?0-tT99`HOPO$|~MuFa$AY!xeArnp46z0g$mfUFP2Z=mr^azEL- z^cz*lN+Q=@J)J^;o83<9KK8ed~FjWWTd}pZ#4; zXG-7~{N6L@ep*2H69c;M4g6zaxyNgacd)U$^1d`S^>c?3*6X5QJZhp(R890w)I^`6 zn&^G1iQXHI;vXQ%!5xwlyVGYUQhj>vul_z)uU5Rou3B_}ml>%IyuwJ?BIOl@r4>yZ zRjjnMzDc)z!c^OpiC60>ob)u1Ystmc_G(q}K2tRgM7zjcwLb41A^P6gQy1Nb>u2G- zWB-gLdur34_zqKY4oV}h_v;qDZF#5NzRXGi%WcXMN&)_{(ejJ&zt?RG^t7|^(9MW$ zN-QhAgWpsZPC5&T`&P!1X?dUS&}K7NHlH}JQC8HU&Roz6zmJzo9iLH;!xe?mO`{Ay zujDNQdBvuYbLMOsfkt;M&}~aMjX)yq`SShoA|+c`C%&LnN?V|Oh5Cui*{xa#5*JE? zP}sC-1PYf7>MiTS7ARlWDkVcuC~f$vdNl-zv(jXUmFy&)l}Q~GGTcz-lT0>Ai~Ncb z`5;6g;W^F-3G3Jp(qP>y&`oA#CIHFXsQhpr7`sAS2PRh{3M4l{6u-!UumN#3u=ziq zhkL)C(2Ms+X1(ruCrxh2MolCg#9rK;`&$a-1=4f%J96@T;g=dZq5pl(J>z!r(CQ(J z<-Ldf=3Frne&}G3K;e*nXPsMgOALj+B{k&$y(|HGY3Sa)hpFvH>F1WMRhve~H;r5* zD-$$sz%0;hlbc2$adEO1J6rN4+n_yDRW`&RF;|*m(8xmEw!T(LG(XQ(LLyB~mHmRn zgq)tB8S%pHTT(>KvJV3is?5$TY+>`^1~wn2(AMJwv4|)Tzk=wXaAwmt1Vt+9Iffq? zElxZxWFt8ntIyE-to3*YJD1r#=W*aGl~lCmnF^}^PUXo&xuTw{-&A;mYWskGj>*e#;n1A%CzZZc zOTq4YejA7{vL9OGy0C>9 zJmo@puHw#xgl}Gz=jwNG10O+*$-!EvIpZH`+P13cQrGk3s0ZS&bmP{oiMr;oY*2rz zTj%T6O(Rg|=%iT?ww%2-lnoM5)dj7$_*CNAcT+)ucPJ?qlo9@v3yL~?lY#>AMGBgI zhcIIBlnd&)iUqw>z23Yk&(&`d!`IRrQ&7hHC_NT*Wh^Mf{jD&aO{20+>J8~I-Lz#( zHk~aezOoG(p4TU&ag(na=3y;-Z9Obl9BHitA|;i`-he<1o^mBTSFsXHG{()V@?8C< z7JSE9=6ah(C*rCJjeGqIzziC&EmKTYOG20H-8uFiZ9Xz1LFA%-#BU{g(Rh zdDzwR)HzQ*1sYFYa{M?{%bl$0OBy9;t(!w^ucq&bN!sx3g{wM%r1wMQX|U zcGh!|x8!`gZ{Ojm99iT|HsPBrSm>JPG*+ZCqPIjNkj{#XjHsG0mP*#m=9w^->ebCA znlSb&7V(BhED4w?6llJs0?m6DXucwW=A8>PUyVTXf2;u|-`3fhbwLYERpjjYjk@{m z`YCr$fIn7J{iG{{yNs^`e9A+kKQ$kNmnY(A#o#qcl#?pp)_TEPghwj|oyXaW zY4Wr%13IP_W^j(Z($xmO)JQ$xO-c&s1b;KJ0cpU%2B!VQJ0P7H*nF|;v{3E$>xo!k z1b@>`&U(OaH`u`JETm}k$p*XI)CL}+yAQFtreeT8MLhQHRCva~cauHgO1slOwgCcl zOHa#EK8lmJKq0H93<9}LB2dV-iTIQ!5%Ny*OoSAYnk3>gWFmuRUE@JlpQ*+spR4Ah zvp!qRZ9ZQOl|ExlF`u)Bn9o{M%;&8k<}=q6^SNt?`RqMe7m`ou=e*WfnnQL+zM$v##k6X*Fl$&~y*I;3%w>?2pQkNSKKi)D9SYiWp@ z;hh$BO(yE+TGTc95*@Ov9w?6}HlGw;vj6@1p0Xyv9@tF2fyw_gC2J1;wE8NKVIu~v2v2HpDw-FpVz`v%>62i^O(Sli^O)-F9D zJ<)hP$``+}DX(`ApP3P|CkHY)8VbC}jZOWWejpY4k$;Mb#Ay;)Ut91KrY8Entcetu zYSu*Gku}j5wa|{-Z!VS7> zGnP!tvvp$9LbBrkQnPvoL&;dASKvO*VlwUOGucN;1D3L~#5tP^OuR>ex|#K&|S7c&~C z|Cj3(Wg!8%$XvV|d!wrQx_Q|FqFv;!TJMWhSX4fzi^|vav%vD@j3s+imh;4N!C6=~ zH)VYb?X4$7$LnX|$_OlnDoZE?_%TMyAC>(}-PSeL+4oaEDPzeFz#p!;Uy^D7RsN^jJc{)Bl z3ytdxJLtBvczk8k2%9+jUZ5w*DoW|9fKf6Ds>z{gm@P;J+wI z4!*8j&){o`5BQrVpRW@7kq?J8;D1xW((&bg>9$`ad%%Xh>(+aFo~dyDdW!DTSP%;b zHm7DR*?}_`$hqcSo_?OI-@%9H3%gEv5I$dDJY!+GShv|jF3#n$0zEi2m+TIDJzaQ7rG96C=XqScCJ?k zKJWTb>fbTuX9tL?`m|8AQv zQ#NfHf%3toPlo-Q%Wm%d9Hr@bNC2iOpZ@Ng&1F_*0!qYlIrJ4g&=;}lD)Kka!x7)=~mqok{xRYq5G>e}3IWkC70MnnMGT&DT0K{>LZ8tHH?PWb^_xRqQ*F2F zC(nvA{;ksEv*OF*Q;!fwbrD(+o+XuSQVK|0=%$eDxCP2~CKR7P6(LB3R9C1zGxW4ya^WU$r=tszf>rD{1h>G$`$fl#R_p_ee)VtYg|bqj=vSs_5{R>-t&QV8FAq~APcUN{%W1ml zt;W&=zQ;(Dzz--X_5!1C8e z^#O@FGBTR740M5HATkb4LYpMK+rpgyl8KA`B#y$jIp5Edx^qb;?D? z5k~qMY|mA^9ouT#j8d+$Tj6=Grc1e+F63(3=PI~2UFF%DCR^kpGBP3^azP46E+QkN zu4QH#SSd+H7f%$6IlmMWyy4l4!Ss1zU3ncd^r7T^)S$==Z>jz z-el_>)qADsKez(_R%vowhywv*adHIiOU8=t0ti18}H|wkKDi!C%z&2`skr`Ow-H5Vy!lE0gj7W3g zJ@(CY&qu6#rh#8jQavXa;F1(}m@d0Rb^OZIcYxb!obrRnz7<+uQ}#wrd!C;;&GN%q z=g1FhdF}E;S{KT%XXd|Nb(hMgfZE+JEtE8}C^(samO^m?L8EF#uFEeBr{mdA9z{N+z1ju1B@Xw6Y19prw z2|Ue6J>bblnl7K!Jj`w>%x)pfE@2Wy$Vazg^aG2j3naGsxRL?UMQKI{TNGU&G0TAv zkeH(^Bf^&BN+21CjDwTVE+wIDNx0f#YXfgG(gcu91fy-s%k9S41`@O!&jAm#_}f5Y zC>-_WRqT`k=vV+RHYYm3n~k&_*bJa!0VF@NqrTjasgfU4<_9e#M`XaJ(J3=Zw+JI} z;o-54;n)jI`a;U5Dn_)62m|SrAPoMF<$MY_PiIcl$7sG%gD`lf8Jvt#uCk-PJXh1D zTum2pHC@WpbRk#MecE?;G=Rory6;at(Q6da#(#@9XY$%6%z-G>Sg@yeOeVBNfuDUO@)%V~E{5_?~TKElJ z?^uLOXT*L7BBwa)J`m&Fm9>x^h_L##obIN7<%|^gcI~El@^G4*&N6(b&4V_OLdbi= z>kG;5uqW6`BdyPI^IbvbL=B(iKZFmyRJYYOd=yuyPw&^yc&^C^_vBpjzm)+`BN~)@l~g)O{kU#xw3$!e zx1DA^gY??_o}_+POsy$`YF$& z0AJiV9_>|(-kKPJKQz(`;E!g=GWyKXF)4*h9^)#2aXIq>Vma)uy5`5b&1e_+J|j&7 zu^452Nphp@lF_yqeS=wO1F;mok^tPant%1j1uoP%t+jK_!pn7IO=N)HvDU`Z$K(gr zeH>n+?2Z2jar{9_MyQ)KPjYMbmNgY45+$c;z*LLC*e`QeCUv*2sTiR{b_u|SI$`Xm zj^+^WHmt1}p+m|8Y^W2)e(IQLA=tGQBXr1=1vbFc2oua zoY9W1!2fQvM_1rGjdn~0o}TY(QS+Hn>5D5D);fgfwM6SCTn&z;PV zgDWsOlr{vCLYZ?wGALX?5-40i@+VCLBz?jKBzwXIBzeLGBzM9EBz3|CBy++AByqw8 zByYk6ByGY4JXSL(F#<{IR;`UUti59H@r%!qpG@?fzqMv{-$r)9hbv8V0{_BDUEm5M zb%5s@sSCW!NIl@2jnoDHqmg>Rn~l^3e%nYr;LnWI1sS7;NNIf9=lk5UX zrDVb|Ss1_Nn+>c$ zFf4(=JLoNK8SXNOsl`hkl3ClSwPdEbz~n6&6D9+bxB534*Gu$sUTZ2mp2-mJ#{#+V zR?jUe_H+G|<_8}1=x9sM+kpowDX!6s+KJJ7hx)IZ`Y!NMCba^H!g#2}D5=j*08oFs zsc!=hH1(4}6bAL}icwNO+0Xl5oB9s$;ii59h{B-0Q&2zQ)z4QQi;^yM9F<7GCmSi7 z8Ym)AIR^)#aOjxPMNMSK3vBA@76Q!5gD%?V-gFYW4zM(w3dFB@vi^R2eZjF)2WNf~ zC4Z*+$K)${&X#qYaY}!7?{A0-!Gg9~Xg^II`a27}2fW@$6Tlml6zbip7=14>0`H!| z=o5s|OHHr~yunDzftU$KyJi#%g=Y>?f0^lO1K(|=E)Yerhx@q-x9mEAO~FpKa^eKs za>Ne&Cv_s6QGp3k-T}CNJp(KRM*nw)DnV)3fz&_JvT;-%$DflwdHOaLz~a5Kjxwyv z5kC;8Lq7AgJB7bLpKPzXluEW3vBAT7ai^`jLDa1uaWtV8C>5YOr)>@ z0`}*J4btz6tl__>^+Oe7@@=Yor{D{PACqnhC)tcKxkYHbumJ*^5%7xNp&otAJUhQW z_{rrX-OIm67g-MHv+3K1>RhsqH|j3&#f4sM9L3DU(M&0ovupO zn}6%-namg^Qg0yo#^h(nOFvt5hN^!H1(4}6vmm%D5=lR4p9G1Q{MqDF!d8a z6bAM3W!B2DUt$DeF*5p;{Zx(ZkOaLhCX82Dc;p6cgrnT z1X{SXhb}tn-ZUAyer{#z0r5aj9Qx&&5=wT6>24#7de#+Vge7BN9@AtBGJs0ej1iM2Bwo8Kf?V8+TrzDsk%jbeq zCBdo8)t9^E$hpgD^;bu1e5#zgJS{(WxylJht%kFd3l14uE;wX1bHO3&mkSQrG1qBw zzNDYhYQP^DX%cv+l4AEUT5xQfHnIo-cQMiwa91UjI~PAbZAQDm9~x;n5X(VVw_>!F z8C?#$ZTB?v?1?L0u1v=tZ&< zW1I|RNB`eYcWyVg#(_y{+Pj74H156WVRR*PSr%X+o>|0)C@m$&CctEH$nHWH(?kwp zfJxVoGiDzx)r56h5BMx2tpxt%$*}@*sp^h1d@)e<9+T?=zo?`*;~D)vF&aEBYL$m( zMigQ*$__T#E*WiC7`@AkwhgMKa9ONmHJ{(RsKg`8-5&5bBTWNORMMQ*dFv_$-NXl+ zN_=Z82G2@-z%vuynu@`O#0Pvq;=8zF@S4O2d~M=;Ud7-&i4XYqiSMF{!9OQH;O7$G zg%yKuB|hMPB)$tO2ER&t!2eEs=T{8oA8(xk+*(PJ=kqEC3lkr3pTu`=#o*w?2RtnC zt*#h6Ht_*ZPJHK73?>sFaB1Q@yJB#9;sZV>@tsvMSfBV#)LF>Pklmx77AOB%q&!hS zBIlz6cH25WJzk;7^ht8_up6p+mY6*ia{o-VZwZ9opI9F&prW zMmwYef7@t>R^VS6?XU{G%L&oaXzTC_e2~$OsKBQf?Z^uJbfXKT#BXxm2BaH)}YosplWkwnYzTHS&;EhHa2Y%B?UEnW`G!EQO+gGXq zJith8V51s&>a)vV?CRK+g*1aVnDXpi(&@VC75d5JJ1zJ3GRG&ycO0!ZD$5Ng#Vz5< z$O|e)|D{w}7lB(nw#mroVM>)c0uMA&5BP-{vW)((F^&Ui-Q- z2DrYZ{I=~rvSYtuuH2ek(X5eot zO%e^9dty`|Z3BFWk(L4H>diq@}>^t+ZX>BaO5a z_yi+$fzLG3Qs4$7b%8H3QXBX(B}sZWJok#ZXKlAQJW4V+y-Tw6jH?o@q@}=D z8mSBXdm}9ce#J;#;4h7|6u6`HQx|x!k(L6VWTY*xH| z#!F?5J7i%6<&`tHELm7Vd9!Yjz-~Q*!PSZVR#`FSt4^T2FR@>GChq=1iKQbuLV3yK zV8w{?l z7~N*|9tSo(TRn`;o3jSPCaUDwm^;oWy_vGBn%^h60Yp}U+Hj`tVB{K1d7HmYl@eIk zLwZF&RGv***>K}8+>if=jwv&AUt#E8?@v|lwtpVC$N>oul@i#$p5gV?M6c{Ostif_ zeRD5Y_{sAzHT`mH+NHp2lq3fr|9ZyQA&WQ4v(;zKUr5lOP7=V|%My^2XG`&HFP#{2 zY7C@W&a4l2cXfPm${i37B~7=U!2lm6$3Xn4`S1wy!sksMtEQf8CbJ{dhw7#UmYi{- z|7jutabSTw*5w~=WvyUvv9%kamq5KpK@FQ zL~223peyE{Cy$eP14wtwNqIv;o;zDyK|$I4){^jGg=~R>`fbCOc)oI8Yf=7PO;1$a zr+~4o9xZVKh;#WfloKmI1on*$5$(rJF#SQ<_2ZC-4x#kt6i?NoiA9~ zYBa8^>r-u&?EznIq~*X@D=A!R?Ny8p{L9z_a_`GG1M1H+U0vX1Mw$SkFkZ=Jl+;fQ zR^LYb<)(fTh{B-0UGP3z8aCto8dE=k+N%-?_|Cr|Gb?~UGg2G4mF7>*`_HWyJj9YZ z4kQutAhz%vbXT&rC`%iv zd}+?tY@|euqwGdYRJQ-TNVD@g%hEWI=nJd->y`VxNy@u(liIu^*#_=mQmeew50rZ+ zW63;5`ozQnB$$H7Cn@*YNy^*4UN_MtveW=+h`M1Ike{*G9a(RDSoM9)Om={X&TXx^ z{+bc*$XA>r^1TpqtAZ;RR^AmFHGh-*r@7AgC7o;f>YSeic(Bfx37a`R;LD7(9QaBl z#ifkV`x2v11L}`7mM-uWMp^+xVO+`>CH2{%7U~~oy4t|Yj5G;EVO+`>CH2`-hWb-Y zR|oh~BTWEN7}R$vMoE3P5TC9(E=jtOx-yY~?=e!gYeW%&%2Omj6b>CTx~PeqMFX3< zx`hC<@}P_Mk$oMosjFAemF*PK^%6_NsX%mvhgI)hS5S7UZ%dS=%uk}^O{$;w&={xm zXNLnrR0tNd%|iPP>d<#A@E-6EC57VlDn@rDM&M(f5HlcWIl!KhqO4C)ZnRr6+BKv9 zV@A6`EJ!yFR^KkEZ=3oN3!n{LV5BY(i?P`U8|{>gcFZVh<;otgX(4}rEzjCi$--2@ z!j!jAdT;JuG|4EHXHQRv&z_zr&z_znKWw55)IDQh1J<^a1O6EreA4D=mJyI0Dsc0; z$OVTi6M{7A&^^1Qy1I)sMt0osAl)=;-3acj3OImce1zF-1F-=sW71Z?^A-*peaf-5 zhw7AHgq~Y5K)duL5QY6-bwl4F=HnC)(NeRWrtB{?u^#X(Mp_QMMoDowVYKZNV{nC& z9Mqp>y1Kx(8fgU(g+YC{VwBWpyB^e^Zo1mQw;O3C5QTB!V3gFa^ff{~8If`T-(>11 zfhY{>I|c8v^)XjJh5FZ-`U%v2L`k93!nwyD!;5M z=d{+$tr(z0Is}Nmn$IiSUZ>3AHg$TybBwecc%G8tv}1HrV(h1W7h~xH&o$BtAPVEO zW0cfq(+=FNNVVWbHl3gfh6ba7$?VlguMls!oW zo|^={uF&1*$jtK!3s;rVMh@iY2#D69(}6C^CCBQ(rmk+qh(HUs#?VD)$&w93I{4Kq z=*o`O(bcw$^?-Pw$LjrZO$jACSl^(^!g3J4Mu_oC%}BNgAxc?hQ6@yOG7GDhsYBPA zW7#6KQ8!(spS)1UIEl7_Bo0=eIaI!0Rfadp{m1 zq}ULH1-m7|{N%vO#x4zA?rIAKOSs? zUEncBS`Nf=@S|HXT9z1r7?-esO+WI9fFIkKt~T&kBdq{pIrz~wKUO40Ajaj^6tL+> zKCkeDV#wkFJl@o=1UCKXm>*9|jKHSRd`{*@r!ab?>6*aU$%zCcHu1J!fl=59DCu5_ z?x(13-jZUR3}naE2dK#5=GHhcNoDtX@tnq$y&f=04cWu!N`hGyU?HAa#D^#?B?l6~ zWN^stLKo9SjyQlx*N`*jEY0@|E$=(F^#}S zDk*AYbZlbuoruFfFo(Oq!_1EfATGwGhS6&iBM=uOBcn%{9}_@ajEszu!`a1Z9KPNh zZUaARq;cS=x+cGtFa_SXk<50*=usBLIFR^)Z|#cFL}COIUu3kAMLRxMA@K!wz=tO` zAmInL!3FD3FhU7#bP9oWN`Z9>fptoObt(ee#ysu-_f%3e&4@S(DWIE3&{n zXWPca&JrY-rGYF#@}L=zbTn6^rn|+}2zOaSL?4hPyln zgc*eSFo5}z`-K1a&3<>o&@@8Im!3# zK*uLSht3!Ww-7y&j71d$JWk(H2ZcPLs$c6f$Lh=S4CpZFXS_evv&qChIjq_e!Dn(@XOISd*Ffn8AN?lbHa1riGH-*a;N_swOiWNNd!5 zzKS)`m#`-K`dzK&2G$R4+t>`g=PNHp@2g_r8*A%*yUgM(#h6uPa4k(vDXTBSC$tD( z)lcysNL}O&i>WcWWVvLaKhR8DyxK22wPW%aY?$+2_qL`xKDp{G`0d^fY2V6PV;o4g zBqyQlsm$^~>(rPW^Gal1?|fxJ8U8aymG=)iCgdOs6Xi#9=jc(*j5+1KIYpT`-VFCU z!BkmV*IaOwGdkJ71Dh7}i$(}??keZIz@}jStzBJB z&QtVLI@m|EQa7P+Xx#`tR~4{xW&Ag0vkfF#SQ%s6cIgm^F749$x&L&vR_|pQU|05j zKw5)4>Wr}@Z`52$wIfvEp$s3^2W_Av`ZG+1q+0sZhHUj*S1;GqE9EO+wtY1v(G(V)Ve03d zlneHm4ZhmBV4sOPSg_9;GFY(B^5z9MXhXiz8nXxdw36bw*iY~UCfEfw1@l$xaus-u z36291lrsQt4yUd&sU8ri@N0=ZxrEeD^i?XlM2-;S+g~T|HRtM`;#-T0!s|wRH8xz8 zHY}^TbY9ynDsr1?>jD2$NwIqvJx7n2Y z7l=PnKp@UUb&Sq8**1_gM@B|3N{m2~FYyD3G|CPRwo?kWW5MFP9Df3FJXq*djA$5P z1mbvPWQ3!_2*hW(VtKmvc!5TZ^}deu6Qoz9^uVO6PuqB13)P87SP8~~=-gH&@L`L8 zebDOGuMgVe*R2XVw%C5IZii0(wh!413Z&(uhJH>QXHHB3F*Z*wOUh!*5F7Jk@e?vb zOop|)TUfh^b}u#5Z|%OY|Jse{BUDAdwYyf~*KUM}?(s8ITUy3N&y4gN&y1Y{3dzr3 z)afb_8?se$mIqyn4{8E-{o@)}%qVRRMXeKY?{Zie>!i;1W zEufi>)$P*PZ%dljR_daxl! z3cw?bGzomPk$S*OjWh{-xsj%TuTWAn%IFGX>;b=Jq)FgyMw$ZNK0_U&JB@J?_!}ej zfZOVGC(#u~4^pbc58QVKqn|5zhP$N}NRV0)J)dIzUn*T)>A|mO4PPBwW*{s6~oCX~`G? zQi;LUmSyb+#@GVB$y^-)k}BB^0m)OPj?v9g7D%c@7D%2dSw^HvwiCd+OkE2|s)P%; zr)8}Lj9E)|rDEOi-$YA)%-i zhWL*b#>AsqdPVUEt(;yLFzb3>z#vMc^}2v5Rrm`Sgh<`BHz|?v#u~)?Bg#BUCIT>@_gWz3!bv(ANb{hr*x4AOS#}LUGITk8@`pEsQq$_wbBld zH7r+>UW$DUjcG?NY4j$hd=h1~Z9*#2Jj`N7l#&n~LIled3eT^pb zo0ilLkdBfmUV1!jQi>NCrFz`3iW6EPba9_hq5(!-J#HEH(!~i!iWkWJTA6=fRNv#) ztylk1P15!H$^0{p3Y8%$_-mzHuAfr8K%_Ip%k1Fsy>T90nA-v85oM0=&2*!SIled3 zy+pg}^X<;wD6pxdaKmo2VDz!b2)xcnW5BN(X%x6nZ-mMO02djl3*28xl~G1dFvd~f z1|y9DHyWu6+%!cUqjQWg`JPCck4`Qh9m~fii?jpW*VK;#2~SoNkk~?`or2NnQ5Hxn zA`2vtN|sR!E4fpjhSegh7h3=$K;}BAZ&{>kjj;vX$GjZ@5}w>r1QJ`Nj?wxk3nV;| z1rl2&%P2;wN@ABtBW)9DJ*>8cMSV(-2JWOEIg7G?tf{7rQ z@c+^fy15X#76PkIZY2W=MLHr7uft7lFm4MQFnX4L7j=V{z?oJGO(2HvHy7+mk z?vGV_f39iS7SjVHJ^pq!bSBB)N`_8|$!%T^5W6wx<#E9QraZls%n>TU`v9F~Y9_eV z%mEgyG2jCSsQ%Okj!=E30vmpjyh(F>rRBN7P^!RkmdQnFBd#z%|GzV z1yAWB50-MlU%K7{zczd;{k`_f-R%;$17r!vyKwcLcwwqPOX%G!`JznLC2a&TKf3F3jzK z^N7yIbfb$ozBkkTGM$Il*xYu2bd*f-(&K59QoO(@)#I*?uLyK;15Ba;MqNGbh4s?K z2}g<-$ZbQpAqkA?d)z_v>W|hWo#2X>aa5QT@1ZL4EX#8fh;*iSnH@a7H_oFAa}+P* z?KqDpb9`^68(qxty_xQu|vq%q)=j5G?o(@0~$ zUl?f=c*ST{F)~fStBo`Y{Hl^F*BE^xG6KIpg^^L8{+SUm;_EZbM<@N1FFpt+jdqE20H;pzv>sSzOaSKSw8^mSuw0dZL>6nK^S)dJ$O zZ~ zN}~;=j4DP(6jkm9bsB6p7i`ypWgW@QVIVxjZo!E7v%k)AtIj{c+y`;(kRK#L%Gb*YDOPEP{!J;r(@)tp0SU<8riRW` z`CG)$nKik=D?e0Hf9;whOrqBSolsH@KC+afSD*{5F?WEN(fhJqcaPh(WbS90xqEQC z7Ev<3L$_-YB8pkNU5i5Ic@}TiqLLDs#XEXj-nPz~z=cZU)*)l2PI?cJbpLJdSxjHj%*|u+l4kiXY){T9 zua@@Y^6!@7Ei);B0GZSWMOj@CYPC%lPFT_O)5FBrc&dnRI=|yit3Ro5nwbwmMTj2`lR47mV)ZNK*ncT3R=Lo zDJi@XkXlGCS+Q`>{EAXDNoHzmCdouiE0ZB(@7x@E|8bOLHwak@HDm9);B{PH+8x!fsz-WFfRixg>gpz_2Ed^CaX8dUj zrUkrCNue907LtyPg?r{#l$uF0Q(H4hrkA3IeQ?lY_A2D%%bNAV*+M>6Q~5c|{<`v) zY#Ed4uvG}ZaO;(ZKgS;E-Xrufi`+J4%v8z(24ZT4vx)^gE!P(Vbjeu^$UK&OK8)<* z#Md2Lgh&zAJ&zDg1U35+o~^U~2K|)%Igp@pXEAuTMloPbY?-eKcrK1&!1pPs=UMDR zWk7&)3u6>aa~Au)wx2t#i6&>UA3GK-`DnhZ9!;q$&$OT341KcQ6deT~s-$p7E&ct- zrXF1#1%bzJSMaxbbdH`K`Ws6M@UBK%T7dU5+Q9{QZ=)TOEbJHRu|Mdil#CDh9Q_kY zPd}zQl404Rhb}OdQQ*Z!Y69P*q$*X6E{}}B_Zvx1zR|TS_0ZLk1^96#>M`o&M)S6d3x`^Qy1=I@N#YzNk0dQ~ar-B#^f*)A z1`;oSxB;WLD=T!YoM6thffp$W)=PAJsiR-7;~N~^&oAB{r7p~oFH|qxyI>eIG={S; zQjM~4#m|_HWRLb-)jMX{SOG+OsBvbw5n#N+46FcR09GHx$*P?MOKssU>p@6B3{~ZJ zrkZ0t%y?;}MmreAf@~sCPY6Pq-uuy1(P*kD=~SslGp*}Xxi;zOt5@FNc#8F(b2R^f z2L6)K4lBS9wW)1l@d70UQSe2P4M?5KWfqW^9p**e1*Ip^JQOs@BvHr4tr>Pp=H&NL8FZ=kT!OHACh=W z0_u{}c_1AC!?mX1HD)eweJCtzw6T1kdzH|qb8xV+?D>l=IcTk@US~y}PGi^;;pN|3 zfiEe*|Ex4AEg)vY`A(;rIuF{wOn|WMh!a#_Pz8<_mt*lFwH(-dAqE-?=z8VrYb*z{ zL;@Q)I-C5>OFy?kjEu4T0}ULVZR706YtDjMyR!a)pR2K5tN6<`8kj&t6o~)95ufe1 z>J(mTQ@H|287|Mc!Y3K>s%}+peypEz`U3t$N#QiwDHu)a-;NhX;KEgv){2qQ(MlCY z;JH&6y(>R@;JI?Ux-`%@?bw0=nGj;v((*V>OUcRL@sZdq80@1-C^;FBv{EMn(p7T$ zZ2gzI_Iv%5z5>Kje`9?Ee5g%Z3yAu_=Kmqa6wm%_LdE^k)t(@rc|X6?>+8Djw_<@PVSa85|Q{32()SBF=YT^%?ltsM{(Qa$L8v%91KGas3)9R`0~GniB5t3nblSIvvg@1K8n<`z0y=UJ&ODZtU^ zHik~Ev4PiGMYe&b)!5EZys<_DZ;qlsGLYME(P=(l`RxGX@>!qkx89}fSDRQHh?pd4 zJNqrdA(YJKXRAhXl||G6{#Z%Xp2pYUwtFi0ITiehS!@EoYoreFHX}8G_|@MyHR)7$ zDtlvfyA6(ge$ zL`EQ~s2CYN!xGa1l8TCv(R(a0Eg-3=7#YRHB)1IGZ@Wuawy#Mx|oTa7WzlC71>P5OoO zZ6(E-(5tPe?siqQ@YsqbPYi&MRMJ3W?WO|lC#%YRSR2W^rC7+)=F6$@i5zrX{(X%sRVH(-U?EVDiHb zkK)0Jz{@23QcVRK7_-R7$*2&vFIC&aQuc?2yD|lotMv#4!&Rd0}%{S}awSQfsQW5sPy`T6DSl!aYDLqV7aFMplN z!6W;1&ysE}#?R1}Ko^n4P1j&QA*Amq7Y#&6UN>fUQP#Tq(6Arc7G3|83_moSk{gK9 zZx217N(O#?IU@-TN2f}9pBt~JD2M5%Yz%?VG|~w0SxO2gqtSxVhaw~JBU2a|%{jg@ zmbvsx{pjEaW`fa<8Eu=<4)Cotqn(1$dmC?+?q&jb zi5X}CalPbYuuA#lUULIXG$kX0i=#py!IXRqz7hF=L{{?k3T)Wq+Q1`?GzKKRkfgTF z&sC8Th&vS{BRt7GEJ}-_dzh-N8||7=vLy`%NUnm>t`)^Q%)2)5QX_SNxLCab&|39rrl}1L-QL6MS;Fc+jjINH1z$NX4v+|C;Q)zH?#Kck z`zkUz20UI#1C6sb77Xf1NncAMDNnMAYyp|$;BU*4vOY2bNkd;_6>!+XZvpW`xO&YD zmSvR#>rS-IiMN;&Z6KDVs)0{1C)&X1L~@cR6V!xk>Vb79y5_{E%!w`#%YBWNKrD+o zV01$7bH#k%1mQ~y0M?yYZceN=Czb=F`s4~|FXcw3oUc*)GE+1HyhKUz(&sY5`9qDy z%o11Ni=kBLAZKw%UfiFMC!pRebZj4|a&txJP!NECQ^VoFd5xh(whD30M7)1dWI8Xm zpBL(0wz_=xJdA~XH zuXV7eb(t~XK1v#DY>?gfqJr_h#@eJuz1mvMa$qd7q%oR-LN?{H0?0`!m;saFfp7{WH=Lvb zBlvlm!jh2|ex(uvqyI<6Tx*#)vvF8?^$LZsdH)zc@d-L^9b&UvUi%k}@n%>aHB=U$ zywyoCX2zpLFYobjzUbzH#`{@6hObztADfAdgY}{*XMnA#MIgZ#_oBpfKFYq9EHg{hHW{bMurLBitZb8k zgz|veV!uRJ;TvoP6-eNOFf#9vYh0yq(R{+4MZf5EEt?*$!`H)YsoovqF@k{(;_*Kv;AK35bAJMfs~lsFX!xj ziL*%JU%0yuZkw-7;F$S(N?$SVtY+^2{$FKbV#{3kRA-nTpEwHB*HmM06#Nj&%xd5s zcF{f2w?n!HOji>)WV%inEf_pG^7(jh>nAl=!T6VX2Lg?UndTPo`R0L?2m^AOdD#V3 zN2pxCTup!x7urVf3(aJ52OpMbEqONVS?0=lHv6{B+chCnbf#cZmc*QADLajygDbVI z(>!I@$oX$OyGAb+-W>B{fi|5rX~Un{?YSjaB40n&OV8Krq|^m|OG#nZw?Q^*xb-HR zx-RgGN`jT~4=d;IYHn{lu9l?kzzjj8;4s6hy_}&j36iHW z8)-H0qedD9zVFott_ChSxgv%A;%LF>wMrE~fM*(MHE`~0P&W#^(MYR-|EL%Hl?6r@ zY(RE3@Sl}bF*5pnWCXtYG-OACZ#2?s;FpXv3Vh`02(AYH{EUiJ8D;carAqw3r=MA8 ztZ@8)S`VFEJ#-4{;Y&989Uw&@rF`us29LGr?*J)-l8?cEM?N5hQ1UVO2djz>kWwi5 z7*GizFP%(XbbMWu(rIOMd}Wk;x!ULyzBWqMTy=B`UmYbY^|7*3%Q~K zQWhm21FE9R1tSWhWMxpVlJrqSS|zPqm9(r%HdvLkfWNZ|90%6ts+F0maX(ifkXDgS z);NK{f&Xlz5#aSk zY6F=|@easDhS_QtjQVW3w1KP;kp-R`xquXbZ~>`+$^s*bKp27bdT7s}9=bG6GA&yn z;ITFjO(1P-u0-11BwIKphWv{-(AELGMzE9F7#XB-nO@(s?11l3c7U{#GVugR!!h18 zfyri%BzD~)zBV zs!Rd`sSp?AS++t?)3r~BklVFDiX@a@*P6u{7#XoHgb`R@BHd((q*)jt`i<5cM}RjeDLkO0JD#b1FVN_PFVZd-Xmq|$)M!3a zfsds8rS_meqkFM#0*#!UrE?x=bQkgzXg(!@Hhm4OCi*ih`qjW%^e1GZKOqzS37P0m z$V7iaCi)XH(Vvis{)9~QCuE}6fMO2PQ{JkRH2JZFSA9cCH|yuV@`WgUt2z7bEqu2s z|9<7e$n8nM*{_XqQC(uKy$3zz-CKC6jKz7Gwek{g?G?`$dxic^b8?kU_DUeh39li( zNJAOdPq{MzJYGrMx#vv!`^f5zK3x~gr)L9IUW>Q}5xKc@!w2=IqW?le!j_PDl zN>c-F(L)1`O`8e~K25FcD=&ZR>(Qs`%k5Lr4d4G!ewozxdURF-&KIhLOlp)^n}4Ai z-eXioCN(FscIpe&A6D*|scxF76UmL*laV-4jI=0Ww;nKAPl-tA4jdWv~PPIB*9b78ko8}gnkXnD(d>O0nO9H^Xo z$3>R#!b9udv(7uEmV7qmt9B&iJ}Z#!S=OwVquu^ea}~a+$B6Qb2#MZJzZTLp%0)ZKELF^IQQdX-p%Q?Oa zjbHE2NJ7KW1?9AnMxSzaiR_OhWx)C_vD_+i3L!hT8>MB}_ll(id&TNgdo>z>F=qJMHqYx4)rpHyXxlLyJt9z8qYitKS0(`xa!fR{c&$&-TP~zDuVdtsv8#no2o|7z; zRo-hkud^op)^78x0#Yhsa?55vr6TJ66%yVuS(%MP>em14vj_3A(~yVfK-$A#W8$1W z3r5sg*bR*CF9-arRqRY0vs6(TFa4wmIYWwxk(}Gut+2+}^f1Xa^Ty&WGWml-Z$81R5!) z9R!-M^4taOm(|u!^fTlB4SI-Kk&QhNud0nbezDYr5lG2D;4U)=E=F?R^_!Cn4>khT z=eVN!ELT)t?HN0FX6* z)($a`Jc+3tg7idvI8&KnL_eQD%aX#~a&fhw!mDKnnaR#d{K_)Li}Ks(^)hR_LeY7s`4!zlol zqi6LKL-Z_Ay=RqP?^Z>fflpf|EbBF&gqrb5sA;y&6KcjMp=Nv%YA>G-wficgz%eCN zJ)aS^G(TKL^fo8mreX>iUH&qgS zs!O8JbV>ByymY%=5qGM>@%~g)?@&ec9#vHDQbqMX{XpmB*LD-C4g7B-wSfKB=8_KT z6@?Cprn^;Bu7f^IRX*HQw}50M+d(~3ri0$34+B}gnGX6N2}{yJ*QOn`c7K7&s=AOI z$aGMl<~yhyV4kUX(*q%Pq;OlqPqnf8*|r0I!U+e*tFpjrmTE=Cx#+{a?SC2TTVwyL zSr=-?x=?)v#8aXA@GEMs*iW}k*8%>mb-H!Zn;6wP-8p-@PPbU^@eRuJwfzc4&oFAx z!-(mZUS)dH2@VpyTxPn;#^PtIB>HrfM4zpa=#y0vz2U#wZnLV*RL`@gw_VkgH(pV_ z^=E4uyLKbB13bn^ZQv`6lziL7D++B1O~!zAW2824v5}Gr@QOmB(DY~(%T>Tr^w=^}oy@`E2}^SJI7Ux> zuZ$;C;2I@mPj8+nb9!s*vk;z%3^&FmhesxdCx`c1Q|0a}`WcTv;gQLlATzley@N?r zGLK1kFr;LLY|W#37B`PnRKS{+meW{wI^DaZT6m) zoH$u~qLtnx%(%EM=c3&IOkLczIpgBC%^4TB7{7RJl~8$B_!c3;8ZF5_zf{}B253kGxb9SAvX z0UxR)(Kk>q*eCJ<4~%?+1%oF?KHw3NZ%)DB*^v+U(#SVdFnC$y10EOo<`xWI75RW` zBHwVq;IzmGJUjBuD;T^f@&Vr#`Q{f4-VynLmqfmu3I>-(KH!HU--3d{Mrky$fX^yy%}e#+_u?h*Os6b$x_e87Vu z-%!Ee;K&DjM&z4YFnC_%1GXaHaKYf1$On97>lK+6f}w05mwR>m?q(3e%Mn9$?XS{?UcM+mzo<-&5ft#>eZ)#iZ2@KRko;#4%w)r|$)26-b5 z$gWZ%zi=~yPguz|*;cS^7-+~(&@h;5=a^Ia3kGZmgma)^z}a7jg9QUl^+KFeFnC$i zIaDxsRV22; zcio(V0p~N}94Z)a@)F|Qf&phLAr2P|I4ucrUcrEKkPzn=3^?Hkai@X-XBQzZC>VS? z`nhw#fb)iM%0@>yZi$@nBobTEDq0cehcd+s;>jYpFLacyq!HV_uL8bINdt||rxy%Z zd6{=WQFXMbY66j#g%MS~D)RaDjVfFbYd{<*z3L^0T`5N()=C$=+@NV0;e$TI33fCUqBu?Iw| zwC3e#UUV$^J>JukY{FFQaJ}?ImLx}LdB0tBG=WzrsVqzexRlx6VI{J3yhE^HLbL#p z$}R&A#$8WRfhkK8>&x|SG~21_&OCRSXL1*xWk!FiG_cNi@wlS;*w0q$SDaEE>1!Or zMwtDDki0iSVpoNFA$i>wsqW)U_XxV*sW-vpOa176gWj$ZQmT8z>;9nXzFR+~h|ztL z>29HDiCq!2fcRZK2kfQ(nWk$T^)W5uxXvWWZWW8D58onj|30cNf6~&jGS$5@)x9#+ zo$Q~{eIKRebL#rPTGrb@LMYoggE$p!;1$~0OFkcDN#xv3IhRBqdr5SOZDf{`yp!pT z$U+b*Er!KgQ8Po|LOgceUp7qSY&hM$T8u7&r=j3T~wQi z_-O2a&yXGPezN3i4M!fud%cp6+Qa->mV}zg(u^PGzcG`g^;5D$y9fb{^vY5tLY8PJ zCPFkc6+4j2Qfb6ziRM+ylDve4URJzhNvN4x5^5$(Le12YP`&kcWwNwxN|spRA+lDl zEL9@JOzSfd2sEqtqAR==5!88)ex)8^SKcN+Lz^SnmmW-hwpcT70htOB#G%8iEw2XR zisLY zq4)SOJ=1jK8D~0{eCS?LHp*A4m2=Hnatib;d2604YmMTgQC6xbw%R(C`Sn5nU(MmK z^fR+thvjx<39li5?=o8YhVfiI&CS2E@%zp)_lQ=74V=jI|~U8ouBLd{qgYR0-yee9fv zzo0po*EnWUn(OHUlYNlBt%r864#|#T=-|cY{buvfqWy)7qjxIToRX`*mz}RFoBY5w z5DgCtn~|ux`ja2M^!#z=ifm!MT!wzCN}`{slISO@B>H(Oi9YO~YT7V-|CVURx@>ZM z+~U(c+$Pl)h-mP!Q(3>z9B7HIm*RsoU)v8Qx^*Jd(S;LLiE?yLWnelOLB1PgZW-|6($%Zejr+FePU|>71d8U)v!==Ft0Hh8tpo%Vn~m*DlQL+AEAoR(JJOB$s4W|bSC=< z;V< z7V31?@3-0>(qHA^>-w4BXk4ay{Bv!uKMMRCBaHwHB~@QcW%R+w=tD>SiN?|ao@b;n zAPR%}PQfUuAM2$amu0U7L^|kdnXb#tXbXt?ig9`i?VN>n#zH%1A^izeQiNLhK{;SO zhHl0}H)kRJ1fCy&p(ch`u57Xc6m(2t4(Mk&D(qE-QaIOh$=jPtVS2yauo?gP=2^H`RUFjYI@9E7zMuDNF%`aE2+u@qcipTG1GTb^&QkB zm7hWcUuaIWfR`9)42UN|SF2!zh0G6Wp&n->Tfldk`c*&_2K8-I|DTZ&_!}ivjEvs1 zu_{n`APU@FQ&urDI>ZvW3P=nUqc8L{jBXmn>RcGzOc<+kVRQ?^SZ`sh2I6?|u3Ip| zce%#~tcQ_y?lg?$X&5VWVJy#tu`(CNatnhgkoSIoI3B`So(W@RE{x^L2Iyrv`&6aW z9k5bq?45T7|=ax{qR}V>D$0p8EFNOHC<}+y`OBel*3k+ljE zhy-J@86}|5?~!0k0}WpomfulbR6ro~s0s?nKow(RjS|xbyrHqV7BD3|SN=wpoTnER z5bBv|l}DfudH9RVK^0@-HdPRa4*|wRX95ri4*|x6YXZ|nD+|+mEhO)kkZ|dtR-NOI zQ23-sIz-_Sk#wZOU9AwueH5bJ2atKSz*nZ6Z&KG+;yMdfxPV)A?G8ynZBD-0|Bimj#SM^(?r%&C zlJT9bH8z1yQ&M2_m3WP6MU~urK-0mdsntsgN9N0yhMsc3(!J$J^b2TcTp(7@@8c-S z@vSZi4S%OfdLRBd$xGXtmM_(d>FV!pV|du2Y6G!5EKyXH!i1fYOia%TZIokYc;_8W!gt3Ec#WaB|rv8ThIz}yqU3Ib2kHTY2 zdJ+vu`pG(F<2)-%^mj_aaOU#;bL!q#^)uXA9h@9w_+2H>4df-otnwlIG1`d#gIgOG`GQL&UC+;%- zH>Imv4B423MHH4^NC{K)Or1RyVc1l%WeUQG9v~x(mrvEhkX_6!QzD#9`U{casyu84 zGTpKPx^kjcNF_|=^4qXpa-8Sp>wY1eX{O`Nob7M}Uo#hAevQi4c;Ekd`I;~j!C3~n z_X7E|tW+l?rwa?lf2nj?6+m_YVVf0LIXRwZ<#S3&XUv&0v#_vEVamM{v_+{lX}DPp z2p8c|2tx9a;U>u=1BgzS8RToe-7`l%uPOXjOJxg4qma79EtZO$3V|=uMUZ(_f_6s- z5f+wIOwDp2A!lnoRkb2jHM8pnRPC!1Hdh+UE#O|SYn1XB@pIymTjEugZRM7||OBG`!6#t(ls!gKmS!rjo(k3fJ7%a34M#pWbtVsI> zo~@*?NdpsNCCrE@1E*mv!-_M8P@y!CZZ42+CXj9}kZvZBt`DTVLz0W6Tlh#y)?6s* zed#om<+)InTPT-VSuO_>(qLn){Mt4n<|d?Yc_zT+X@K&5u3Ui2Ex>JBo8_0R7}Sd{ zy<`0jJ!uiuEH|YAO~=mq8TR|&M;L8s0e-a64lWF9vE_cJdW-UtRe8YGAhYBu(0mC5 zn*WC`=1or0K-5=Lhyi9u7=hRJIOBLj^y1`7s+A~mrww|+Mk+}A-LIrR|1kN~(!8EF(q*QoD=yS0)8pJY4#4zOOi%>EA|8}LU)8U@mqW>WLvkA#Mw zd-aU0 zwWxqV-wn?SJF-}FF;}Ym+BQ{bGAaR*!5YUW##DhR@@w4=kO(FmO^lgZ6PPYNvh4lT z3dsi|Bp-v2aQ|GZ>($&5d@MK;Q~_Lv5-X+$Sh}eJ4`0)vA^+@w2660>TdF984H#3gW{@hT4Fypj_7~I=>g8=ow4+K$ zXnMODOx~PF5&qVnIYPsT{7u~2(@N|SLR}IXzD$+GEdrc+)Q9@zB)Ki`*!QNGI z?P)Q$fOr+3WkSKvx+3po{h0UhA~i-ta(f(jt@$y|$%CFGSC*_Q7E6&j zJvqpj-PKO^d{PcVB&aC|X@~!z8mAnDNa(vZ*2&+m=IN@a=DJ({+kJp3sho+?I zbvZ$Z=jp-3EC=4D4B?@!c*Deg&B*05Ylb%{*|G_N?coB;T6m%j_4Zy~aNrkauJnQ- z)2~mQ{nV2>3P%!h%GMI?L^Dy)zGIHGVDp}) z(d*5JanF2yx^|-%=%;MJfV4na96%!5O@2PF{Id1oi{;AgRJryc#yeC`%0X#9LNlLo z_y|%CCV@sH-U4s22^|O0S5iHG=J5KzEek6VX8FinkLjgaNZo1Pw19N0OyOYuPBkA& zJoF33(XA#1qP8XwPi5&`IWZLEr+0WY`Vh|a6sCpp%`{8Tl533cyk1ii-GstPMpiNq zn~~YagjzT=0qmxaVs5a<(4)YcjWh!Mk&?pAuhD|h-@LIhoOxK0s_&rw7E?b4L}5_h zG4;PP#ttxAOm6L_7F$@Xt8bb5-mjG#CQU=`=0fgfLhj~5?q)*n`jE?Oec~s|5OcR+M4Tln1H!CW8NF14E?F7W zqfVbk@1^BOsB(WFSg+Ur@lpSeFYk1sAC!&hACh{#uRD5baaK=J>AcLU`bo{#-{@z! zm`G)$cAxFOuAQ<^WArC)UfzFuh^U#zrxi`}gxTHdWF{z3IT2GWA0*pzI2~3wAxewu^ z6cPdAxI9Yrt_le+Pqc-!`c#EDF4y5eToged-e#`W{VdAG0)9rPx;RDg3Kx)~y{DHw z^q1(5Cse3V7i0%A8ljD>*q; z=NPlyMiUrQlq~c=)m}}{4%cgcp|o?YrH_A7;oU}B`9!tDJcw6*Zue5|#rmmFS{WX! zWP3IU;RrpzlUBxT_wcU`H5#Ou@S%H;k-uK`@EDTu_pC`Z zfvnT`ER*fPEm}G#Va(njv#hXB^~p&GZBeR2BKEQA=@1}wA%fHbF$>9ui|_X|UC7tm zoi*R~(d-?hpK|F5q?t%r0?$^i^eG^1;JrOLaQPvc&Qu54Wm$?ZKoXt`k^l2;md1fh zNh;)L=v4KZ_#|X3qd&7ay;NA7UM#m@kvl`J#s_jxq908Z(t@O%fw&gsr;lqj`Vh`^ z0OpYHn|WLd%SSZ?*>nKu-I;}e)caJ5XS0l>XcLGJy<I)8YGf4nhc{I@ z6Di=`N(vVVqXnY{=fF4$e6x{O13zq}QQ+7A62aAedhqKv=2r)Jv-!0Oh$q3XPR6ee zevO$EtANKTsgh-c*Rn+d;mEjJ@Ljti|5~613DF zupWQQ;wM631QK2_+A0_wYGJGd;&{c#2;W5`upUPGvztugA1sVE5Z|SH0PA72Eez&P zdLEE)L>5Ra!Qr+=#@q=bkhv3ez(-pA<3J*=)G;DxQ3tHYpFW74qqRze(xQNOSQ(G( zp(Xf8tIj46ujPdRyv1VX9f7M8bCz4oRRLj@4XYkTClkiZyAxs1f23gnaXidMClkgDTM=R4 zvz!medraqQ{D{?=^mMN%Tn~Cpf#y{On(vS3Vl~}Q3yU0t$A_uwhh$toBys)v{~3!k z4dX$%NYgMLl#7%wDA0SnNKte@Ev(esO4e08H?8F49{Y{8?>TnvUJ2Y_q&9FD+kcG% z_cu}-NW+vf8ZguJKNtmpvFRmW?e_szKiascAiqplcqml;`lgQ`3GzWQq68Gug9MF~ z(=&9&obTq!_X=g#w5WhU^w9(q@_{PGWFSiTj-4iAq7i;n)F)!Dypt;Dt3}pdi97sO zR}f%K)FuESrlArb4V)KI#F*Gj5d?xmfHA?Ez;vXWsQ`Zri#!HkJRSb=LSq%j~1cA&9g zpkVNKk&n>-DU!aZ@T-yZuL@7Kr8<@-DA~fxgF~MnvO!6OFBcL!dCh&{5V<(p52TGL z7Z=Zp*Bd|_?r&@wWZu_DHek%hra|)Y1!W7OK97&lW2l#lFSMhoc0GnBA@x#(zx9Vy z&@i$<9Erc1iakQ8OG3k!sggdxsYktsPtZ{DM(+Oracq9+SoO6l+&aaQ+ydfNeE0!=aR0c9(j}9u4-~PYXa6GVdiCqO=4umHe-%l%dY&!#CXhCm zyWLo5uymDhr9o$*t%Uawlx@JYciA#LHIjfwDoGx%UpVn<9pGT*mU~CFuhvg}0>kh% zO138{5GM3M`6R15Imq~SM`wJO(x?8UstDq6M}0*h3Sby`xQLHLWm+bF-qX||U-RvrIWnd> z{E_9dg`$>4)FPs5BFRq{%k>kcXeEWjJWC^*UTQp*O7`#A`jFa^yY2FK?>5VHLy2BA z(HCc<3B<7|KfQme(1&nl3-?|1E6z?AXl*MrB5y5~OCHX_@;x;Xo=Jqj8DmH48HmZ= zQKA&iOh^<31)G`6lqH!s&mzA-9=tCvEqpRO?~I&Ep8ejc6?@{FEB*4I0C>8Rs#|f4 z&Wen{pH5-)ex?>i(e-*29BQnWXWaJ{%;q~) z@tlVIsQTm}!@pLtybq>7{2z`&W&U@Oii|%qFgXbQQz|XTH`R~dsPx;X8fA#zQ;mAh zSy4ogdFToYYsIe}{FDU%#AR7Kn+gW3A>jkEoJzhKEGWFhPA`+6)0ALm7ECf*d&*A) z$#38vzW9vrv1m=>so}3tEL*&&fCK0Dp2`^}*LdL}H=b7}+IX2~CbYK)esXtvOGR0t zpYpCd@NgrI0H3L((0xY>M#pWfbYz~Rq5dG#)d3=1b#g|pjEuncU!i^_aIKOmSw_*1 zWXpmd#3t90Kr9DcEsNK5?kDU6J6kBmUFU!|w=H~v4JfOc*I+BN|n zvk7Pef21V&CE26EOhd5RE*L%5QrHGE4HYA!_eVw`(@-%oBEfQ503^E=BcndcOdCjc zD@I0diHyL1t>wC1Fe1TX8hGavMn=p)=F%lie>a!@ZYKR*(ob?j7~M>&ySY?%i!{~U zT&lZ8n(A&Y)!l+rlQ#J-0g#+kJ~F!1Qr!iTvx<>X%wRW`ZiSB0;-f3cSrB-pm1@#= zkcuqHX7_59{wLFtyrSz_!aJ!?KVZS#cOTikKrnFZX3se58zJbpUZY zqzLsaT44hcMYVpt&CF`UH#azk)x+z`NlM|;D$Mkk=g`)_n{t>)fZ zaePoY_mEBH;foXF9G+_mBb#*7+Wx{glye{4taMIgCP6ov+FsGaK{;6TmD)A+?T{~# zgS2x$Q$OX`r>{+aeHt#B(45BFb1t7(d`4ZvRVDYbfwXLJykut2bCh~?vYlI+z^G^( zMPE@@Wd8}oD|x`y$Jv)Vu)6ejI z8#Mhsh63E%d`f>*@wYA(>8NNTl+2|DQaDn-GK7Y6Q@>n;a(t^xLc`ywl71E%8a(#G z-p}Qw{%slz|Hksu0%A8_Q$oR$x+3q#xoVWtWVq`Nwtm(aKU(dsHTz?4RroO@C8hRs zJ%qGW?H*GhEZMCR^TDm9WyC1-T< zOp-I}7#8*|k`o|O`J5og_FPI2SQktafFN9%Xiw3_I-q~GLaqgYLq-|_&R0_KbF^S| zd1Um{nX2!g9_gT~W4aDCT^-;v%xL&4CyZVi8G##=G|4z`r)y(gNJ4Cc=+#fU%I<|LifV-lB~Okq!-H zHmW~g!iWhGMj+F%19OAf7sZqV(I$@w~fh~3V%4H zllj9T&(?reSSFg@uhn|!Pa_MEWMwudc*ErGz+@xJEJ=d~GE36?fcTJ~uq)N+pPTce zz_TZ-NW%kjaG|uSYcjg7#tFV%Y2pd+ziMpYeb29qR042a`ZCuM}|vc7=0F6#-nhx#F0K-{j>F~Z@@a{YpO`G56uPGe+pcyc#}-%|2$ zAg`1o!%%*#N9O5~@yTIG`@OBoN$n7%3pE47jZ&QmLAgke%&m@04*Oh05mNFU6{*>Q z^gTT^U%fkTx5;5>|E|Y&lE0%=dJzsnB)1F7gXc-W;V9HcXvU*2Q1;Rz=DyE;6!8V~ zJI&?BZ&kRwxgnjMz4TMe0i(H9skv3DxmBsTQ=2HAY)nfHxZL-~voq?oEpCSw2&# znN)cyH8O)trBXKIoNHvpfk)XgZvjtHQu$&RgSASmo*gbM7{{}~s9!ayBNjB)N;gfDmG;q@rD*_E}mhzYXWh7u(3Y#*VbO30$ZY> z*G-({cL(ylO|`zyw2uP27gTm6-@pw@s&*!fJ`fp!-^ku5?Pqd|`Y`&HOT&1D4a7#>-UsfHJ*2|f&byi0Jp7CL zN%2p!4p1*_UVCQUp`2-y`&+Z~?~ROs4{`k(*hLQ>mFGFO`}}R#GXe0!|0wH1HvMW{}elaCapQ z$;*o2ug5Zuy*oKGr#3CV+cfD?R=_8l&DFrC8L17#T>kFCR#jV@2j9I?mZUsWmZZpj zsD>KmVR9ShUL+1t@*vl5K=L4WaDdxSNmD(1wRt)Ue1nn3fp1h&eoJto9(|n&j`{S$ zvc*`E+`)3Pv2?tat$OI=#xe?=b0KGe5#X*$s+?x@or{dIm->&G`VMf&)QVnIY7g#RWl7ss}8h;wb zgR&Y4gL;!QEU;dB52!|f&-6&S1_RO~<;B#!((*}OYNGpK1D5?BG>7VGJVf&yqa1SMwd{CkyZd{iKRB*@W?i+I{ef__#@klNmuw?i|QBgp^!i% z7?aH?0fh}05{%iw%ndezaR`JSRY4&csA5d4QDXXlH3exC z{2uk^3iGRpt6z#FjQxeJm?n^QmHW|(*Xm-WAB7vQ^duUR^pkHW8|PVBqT^}^!*a*n zCwYIBdXj$XZ$2>mKT5VAaDea=*2^B}IeuDwqujHejyH3* z!wr1Rd_lplQTZ0m-9M3U?JyCXWuSX6kiSV+VAwl_*DM*|9#z|{9Di=*SpC8W zW6qSBg@ttrQ@SYHqEwqSJVT2?xCoCz5R#7!H%T5DKy^HTPzjf0>0nkUhadW-O)ir5lbqjW;u|Mvo)WpT9K-n+4TdeHfzM?kHLgB03o{D zOl>s1yCyz;^y8=_)Xb`)ebNQx9N-nh@`*TQK*~YBEHltc?1L&6ej6l&%@|o(2|%3a z86&R1mJ8as*te-CBGEe#A&4zgKv~w#tfd zd2@gEu8OkQq?$k!=5n;JvcoHyfin?{JZX;3Qq{FMywl6JRg|uN%G=7oZAKaazFSFQ zS2(K5EBWyM7wgeu^)r*BEA`OH?-qXfC+2|blvL?v^pVI2+~bl;Ht$4gp_8-FDX{QK zv(N#4UrFT`92nHic9Pgrv#p%jmYF@*%(j4kr=$>At6+3>WCZ@u{F85~FsS?2O8ooy zwkl~jC8ru7iKxsn!YN?{;?e!%-~BReq~#P#@(2kapP3EVOO*R%`WY@fz%Mu2(gOTv zMmx9wqdxbd%_ik1@%PglV3uVdk6nN)f*n{9Xq#~mS83D|EfBMEqX#oy7UyCe2v}d;GIKZ6kz#JmVEXkG#$Slox!uHo0{-(`z z6ZkD9$!}=+sc3vekA22?mK5OYjJC7@f5T`87vQKrJzL+Z{D;1$GA-vu;5mN_X%+Cz zN}Ag^@3M)Z!xt71_WwKM1s&^PHW`lZuC4|e81F>?mI($IXcgq{ zv0)PjYdPy$&Q@ab(UzT+C?}%K6&JcGB5B<~6wZD<9jM8BoF%aZ#8pYowj}w;#YBt& zUtx(D!S98Zh!GMo9!bDsjWqglh4_%!b>P8I)${(w(gOT5qb(`G%unD0$2wVJC5_3z z=a$|*+Rs_kP2iBlv-&MsQ&(GuS?zUyNfrFq6t#gUl9{-20#z58s^q@U zpQ(FiM>oIx-pbyO^fPlYGLi&0>V3RTrGMR9%jK%&8ne9`?x&f@N$_xgElRqzo(eOeFQWGqd%2YZKlzH--WdBwvznMcG)1yX}DsUMrz zTdka@pCxgNM$U|9wh+6k-u?Bnyoa($?zcdAoEh!{kM_LYX_Ug4=~b{0jJO7zKCWK& zo@9<~e(cbCd0-3y)iS?cmY^a_7g%%TN=xvvnBerD_jC2&$@(daWQ{^5CbMpsyXTlx z;^pZY8U1Fi+#K3`_)vOBe3_P1Y=vzgm0dOg++kUlM4ua!jj7V%#^%p7hSo`9p`bgn z)kgS5DjaTgua&}L+*Z0gBm>gN|>eSh%-Jj zLh`ouQ^J$=QxpRaRZ=+7rgwDM_oGn+fyZxGkXwQJ16mE)n*;Uju|}|HuPHY^Yp%6` zKkKh(6{8-vkJLgN3;Mf64cX6}4%T1V3A9tdTUD9RP5|roV*}r@VEsB!pmo6)MA;^o z$b|1G@Q0O8`UQ}JH7{GhN9= zrk5I$rE$8fD=!Lt2dsWOS_S@ANg+evclA|7d7BvcUL_4QPVX-mJW;Dg_zqV1nn+r! z@O1qY3#WkJ71`dS@VZF)jzY34`hAd0@Ru#3Hju~$8fVNY7(`Xey{a3O@Is5a^-_gV zDtSo)sYjVq^EC=DiKNRE?qtC=_ffcCB$2#7iKG#Q$3_yldt)S#yT6Yl;AfPyv((P{ zgKHW`hu~Y|K8tE9|T}B!M{@O^#00%AVvHcV>HS$Ihkg1WYP2jOn5XfwaAhYz( z#&rykgbXy!?8{{57@z!^X2+*uCpp0m@3(l#LE%LLAKR3JgoOW74iX9TH|3!B^&y(z z17o6rxEF3poh2viG_^_OtTSd*PQ5-3Aw5%5jQ2wFK4;F^Z&eS!ZJv$+V@ZrNS2x=< zjst(7r0|Z>A$xC-W(|c*Wo}K}sX`AMs4^!feBk~{3a6FPg3*e|2t3|MW572WX%zSm zgIE{?e#S_n!1vBEEo7I4UFnnP9I}8r@m|I z+orw?Brqvk;Ks-Wj82SC@5J()6U)sB%0%9<0Ae{z<#MZ-v(1m?K#F9baq2|2VwM+F z%qOj4mIIH}x|jXJM8ROK5-Zz`2)JaOE@b(o+=gGefks$A*FXXd_kOAF1sbkwP*(!Y z|ND2jbZkoBS&z#&nAc@pzX5+lQiaE&ajRKFb7}rk} z3|f&7IA)~wOp0xVvhz!~l4{_a^-#aO+5;x~gJctP^cjtZK6a?_u;KllynAEG!ymfr zMT_@-?7YU(#Sh=V@8M7W&$}OX!NQ{-limY3O4Xt&Q&A^5r{*-4u4ybTykm14OE=1v zdU4^sdwgT*Ox@$HlUOycP5et9P@dsf)W4{sLY_@_-hIhxr}=U64;=wKPrF67pqk#? zK1z6$e$HteGuc1c&v22F=Ld3gvTx16WFItch&)V>t#1q+ymky#lH|{pZY&u5Oo=l!+sT>j z6qrS=+{6RcEu=5!q=B_^7FuTEPRl?Ghy{6)K9N*zYM_mQPn)WhK;j7LXq(L+8e#qw<@cjG3g8nE@%3r*A;)BCnp=M{4}kzO3=q{(WYgM9P_| zuV8?s%%uoM-mPY(vuu*iXHF>aY|;sZZRx^dps(WCUq7XD0A-Dp|Kx8HO%x3H@%hXR z?>|)RjlQS~*f!D%ATbS#ClhjQoBE7CTnvp4yucK;fV-Q!lDWNffZK9s6?jCQPhGZ; zbh=0VbT5!EtsQc}X4#%dvr7_()t=JfESxHXQiEfd%DgxU9C4L#8HXW z;!%JHCo6043XCh6)9MvYa}dMI)zJaqbxJCCRScNtj8~}olzFwGAB43iQ6U^v?KAt2?p;| zVsIUNd891^e;^8fY=XhxN8*(e46cmC%O)6H9f{i}7<@buFPvboap#yJ@PyK2T7h4y zv4Ou4*?^5*qqYjg|FbHdU`5jbQuAT*It3&C@T2UEfYeUK$mljJkq+<(jkRKA#4LzL z;B`|N8SOQ%@>>{z7tgOql}1LtRjM!o_ui?_$mm3+3M25E1$9P7r&(4zK<1?~&FKA+ z5%^&xRg8?j9T|b2*`;op(Tz$KAAwA0rLhu<|4*m1#gx+Aq|i?RGqFKgD>Kb4rkTbO zguzU9CCrHVmeUrHju3>w^n*&65oHlL!PJE)1hP&88<=$}v=hKAQ=uITW|a!93uciD ztqEq03hgNH$F($twhT;x3hfl|E38;sK*}?)fvHc43`lVXHt>6_P+LGsGq8cFB+&+> zi~<{&`VqERHWi|HI5o;03ShlZ+EysL>S;t6fs{d**LK0^mDUg1K+Yx=Bcp$fj6mw3 zVq`=qh-o0puwrENdRuF4AnmeZWJJ6?Sc>MO{~nKED;!C%B^e3|Ca2;)>cU$8>Gk69dU{*gB^fZjoWqDUv>yLDFY3 z(0m>PjV#i?0}ad-gqQ$Q_kj&e;R_p(ISFjcNGxr;N+9=v&1XK)eBJ}iXI-rNlnZJ4 zeA^8N@*O6sleAvB{!KG(w-(sZRz-(BhREU(7@a{tKCB*%Yn_OInaEP1I?#6(0qae z%`b;Q+kyPjhl1EMbaVMUyM7!6PAW-mqW2XHE;hdAy`P%r+qI)uV8DtKv;D<7+7 z!{QB)H5}A|sj@QSSu8U#>#IyizQ}|$TQwH`o=^D73JaM+lW#g5ReqNR0xNtt9FV5G zDCx?m@{1k1U-4$bFZ*bM!l6du5lV%sh(~AyR6{%hh3;(*Kw!qB2o&bsL?BQIQG{xs zqnii>I#3jWLQf)NjA=}!%6F%z%2!W3;)^IA@wH@TeA9{|)3vT)DJuDj*&z#)!kVXr zC5L=z&8M{FnD10kp)ap^#1~l2lrOQmDPLsuW4_Gl$9$pHkNHxox#EkhF6PUvCguyS zF6K+FCgzK-F6PTFX3`g4O`$Knxc@Nm*39`5tc&>~yhE?r$Mu9aM}G56 z{`Ow)z8aM53qi@$z}0$sQSyuR)IxVqGV?^@mF!chWS>nX`$Q_)=TON$ ze@gbL+f8%C-eOKe{%YyuFvFO-l1%DO(DS)+Gh{pPc=<`^HI3mT_pV+xTfFpu;UiCN zEHuIaXO-^~EGmL2@_w`G+#z9RhrNP2#E_H34(ViSe{EA+#_l%?YRWIq9Wv$bKB&&L zE=mv5x+uwMU6kasE=qD*7bV%(g})Id&)>UAZ!U?nW%lzWkwT#&N+L7ONxdX8gG^OP zWC~bDC6PtMNwFmQRFy=ZsFLW@R1$rXN}^9uN%RT&JMB3i)=$~(0@+f{m3Oe$G!`vg z=w(0TWW$g7g>NT$1v{hQ{JwD5tHA~A$!k%brK|TTUt!PFnn3OZnU@fl$YHTA5AQPm zAM3Q@zgO-2_Xc(`1Nr!fYq}ecX|c((sI=IGm=>E5DK^f*g_wpd z#58OnP8YWPMC-PP%8hp)IdmU9bRR!-p91JKDY5|FCjz?91$3Vd=sqLReNv$Nyzo_H zw&3y|Ew-#b)$42_0lA2lCng*9F;==TMtsboz`H3V?_o`6FDL(()hu%p!yY~|d8#xZ z`>M=R6{0WIlGE2|N%Rv|5`Bf1L|>jI(br~4^fO)({hXIXKihm?^Z{LxTw=tzk*4FD zN!jvzCn<^P+=zHOH$qJ3Mu_R$2r-=-A*OR9#B^?in9hw5{oH(7uV;R2Z__33)WULu zvV=#H;9HEA{^k?Mer~Fd0mhSVvY|OtpB=Gt&g^;EF2t#l(_%VRLQJPhi0M=bk*T8l zm3su-jHf~2#4%0)C1*N+!kNyW5Yzb+Vmg09Oy^ICe*V&2$Rww^5Mr7OA*Q(yVwwvf zrnwMenhPPOxe#KS3n8Yt5Mr7OA*Q(y;&i!?Q{?-4Yza!b&j|cJDbRghp!?K7_t}B& z69nDo2)a)bbe}2cK3ULvP|RIVy>Mf=Z&F9d6t_aApUOV&9cT_`kH*{HN}v2kq%o8#rX7CU93HCBF&d6@{}a znii;7?iUrexO<wQt-JDybUjr8&RBrw&Z^L8E%->9J!b`yi3~;YKMXM`$r~b#CJkD693zJds)7nCufF zNZgPwQ1t>>obwzD#7f5fE@d=JHKE>pcq{uHW zkqrVAlJ`zX-nfva*S!MWJDjgBQ!s_D%M;WI0th{H(!r z26jbo|AG7sVQ(q(YT5I3sZ)<-WE3O|wd2!${#rk4haj<@OQY+h1&SCn z(p(*zD36hYeY6kOR6?Rf)>V?C0evsJCI*SC@4A@JAn71hDI8+NSs~#gjtB|+7|B#z zs(TFGBulmuKyoAYG9i#5BJ6D@cAek5E&ucC6QPO zZ07yTKXAS;;&`tX{|EF_QVjfEalFy7vE`W@DYtOYMong=-eSU#{tV*JPxuzthw*o|GdMiMP>8$`E zrdRz!^fPugF4`4i%Uj4~5UC68f-!z;flYLPBpB|YPj;!Ap4nZQuavNVB@xI0*-dnu-j;fa?1jMJ|#@Zg+C4})& zHu*q&l9R+n?@gff@&mak1;5?-!igv;8((iSyGbR$C@BjfmCURXFUiKr3q$wOk|o{% z8fu(b)(GQat=gSR`uJqunNlMV_tt~MjgiTIIX2k`W(DHYaO2D#HG(ic$}c1V@k#1ollLajdijCe%xeTEqU20#WP!#UC1+YA zUXp&{BZcmxMgB?Z0X4$-1xl}KM7HIjMj$rz;5_+rvti3FWdfr zBO;{)f-g&?ydU9%h(~-FxQmg@FJ!@jIq5+_EK50T_BI2pmo?;OULrUVC1+Y9ypkCu zXIdg&l6K)Eh3=yzr(`R<-6v-nhO2b?<_EG&X!vJEf6_^Key|7 zGai9Lhc0h`$px7Rd1vto_0mPqi?ooYml{I~+fpt-6iI1JOs8q)rQtmwZ~5|CW1#Uy9GlKS<4T6@WuUcnSIO@n1RDEBp6dr1Z!qHI|Hs{V2S!;n zeS0a=kwf2_U@%=}jV_Ac90Gh9(hEst~H7M^Qlqi6DZ4 z7!(yj;oI!FCW(`rXLkcW@B4oFgX=zXW_~kg&e?l6$!z49681e<)KOFs0RVADO9^~>kaU2k%eOKFMw*VnoyE45U#mMd_1#BHZmK18JViPxsF4r<>h8IXx_xe=?w{LeYoPqq6W8xG zl|eVyW%Arz%OCGb*~vC3 zJK08eGGR;Gq__hbtL~<;NS!D3sQi5`SFBvh{lUb_D5>>r zbnA_yZoaYV_8V);4QQh`z&P6WAlY#a((QKbaN?iXlarfZ(+y9srE+wGq}kG~1V3q) zbRof}+mK+>O-Qim79`ko0}^by0|_?U`sHDr;1kB5)tHg^ca7T0 zqHj2f6Z%n*q$T?#Taq-R_a$jYHb?F|iD|?j?M@02i)=t*h&;yRK9o2i2Dud_X^BN1 zmWd(yv6&R2H!*2O?`6`A-q6G}`XQSX;^~2#?1OrGcawimPw#KCFulXcrt}^sSJb;Ag>uPVav5 zFung(xDzjf^4UX%a!Rqm*ME#gaZoIU(7{ zD5bgb%p`j1TT-v}WLE;&2K4k(5@y*IIo+63?uvWuDR;$f*1O_$ttt1-t(S7&+-ALR z@?EeTV)suT^qtFX)b}j6Q8rRuGPsSh8}el2HcCt7M7xdBIe89n8+G5@M%^{HQTNPk z)ctT9bvN8b-HU6yDeDe&y~vVF`NlT!^MX3|wu+Tc1!@8HsX#3wB}hKk%M|T$kk21} zlJu!Yxs-3A6RQ-1yp*Ee@V+Uo!R_k_w0 zC9Tl)ixz!6-9Js&XVY!e2h(lT+u%0pBk4Bk^XN9}!{|2ZeRLc3PP&bH56k=?KZEjG zJZXpif4fV@@tV_*UudQB?qqTK1J{K2Psz1|u+)C#A#+6&r8~=|JZ+@%f@3${@+6Xq z%fIm)E{_{|RqcLaPx-iUy#z@4xN#X%K5ksbl#d&iG3DdNWz>(Glq*crxs%)UKc1bF zR_M}9xk8sQotpAFI((-8?~={D-!aU1p4xQ%*W z+(x}AZlm52w^470{6OLV#CAxr_xP-ov>mR;XUgqx8B=bD%b0RIT*j2!;WDP&4wo_I zcDRgsJL>Q%w0q*K(G;H{6PGr)Bs?q0b(ch2%3s9Ge?NJT&6C&5?ptfBtJiBP#E+RO zK-Vv!kHnR~|l4mQ~*+4Q7DVv!%MC8_CcOp4KMOp4Ksl#a>LT_16iHOn$wCS9G&q?5Z$ zp0&G(wsg<+_B*+h-niZ@<&bOXQ^I@eQr6H#wUe+ttzh8E**prj(m6R5@sP=RNhr9tC!`tzQ>hIS%vFy znPNpQ+k}(}JEcsR;`6+cP%6L2Q;|~6^*Gg)dH?8>-+SGs>fYqs?0~7fA4*&{;F9o) zMXt9d+ETt&lb=({1JK<^xsU&&e7Qr*GnLy@-;dqK!Mry7$E#ocTV%KAJ-+(&>V@uh zs`&)EehDP6uP5=*9+ycjXC#{1$z^z=sij=XTUcFz`XMFhjwilA4orMKFNz79BiAC8 z+RMo=(TGbdy^BuQgyLEz7THQJ<-ND5JX|j$xSpRk{bMsKBIP#fd^*x~J9NK9l%14d zvFrG&WX?$DOpY~4Ga`RT^E?w?}J!(y-6mMcnm}l}o8Nqw5cb z7F#AGsWYSN7v#lKe77cYQd&Akce#`*UH8(ly1oRPtul%8e($A-q)69IQm0Jydw~aNeYQPcxt0ZFg<|DA#!tV=X4#pVM~D zwVIz!ObwaJk?=Wl*OWPIVr|N77G~$C6HA-qlIBxWVv^>#50(2s*b~#UHfymHJ5OvFukOvT*pvMVNB&n_zu|> zWsad(UuHIE!t5~DccNxm%<`Tjcglq6`RT+Y&2yhP_l#~r(u{6{ELX~RogcZBvR!9U zE~S{%&!T9_iqbi{$)%L&I(OrBaS1kCSrX^{S(GA@BE3C}x}DzV)RA(3(<>FbUaiE> z;|OGPy?ra;fFwMficQ}Al$ANxkrR_FHDQ-sQLY)CIBCX~S~OhyuY+AC&r#h{Ms@QP z<$m_jaj8Bf$tQ`b@n4C%ehDU*{S!@H}8ZBpJLr2Up`T*rCdInXlf;wy%J5WL}oPBK26sMSg?yCs_1 ziR{EAlU&MbvN(o2UR^XMQ;`%eawJ88#NZBc**DSDMJ^>(!U{U1vVs&Zl|^2IImzbihTt4$pPZ`>7Opc_6u-J>*gzNcZxXk~1Pr`uF?gTgIf*(pq-r>BQ4A)OA`O zT_$5WBCaLKWzuOeuKKi;M@sxt`6){YV#>qnuqJE_n?t!<4`w_HPK7h!D{wJf3N`LE zzt)qw+hHYSP*blx4CB0DU`4xj7 zd5Wlf6^OUx>tfNjtxxhu9l~Yk#PP0s`JaCF@-BAq6MiQC@JDzZ{s#YsqBjD)32>T4 z-=^cBKB*&-*fYcIFfX*_ok{+AaIvKxo9;vCC)UA**o!j%2rLb4d6$xZC0uW*$EGWP z-Avab_NU2j%d^#z&q-V}7!2FMP&gWngLC0RxEL;lZ^QTD2T<4VDZkFAN^aN!%yKC@Lu!jc-<#&>xi?|f#aFh(|A74s{x)JYekHgT>Us;|S3&tV!oL~p zV2Q6nToY)E>rVXhuovtHheBOnD|EZS5ZD*mbjG1K8P2ij+jIw@uj>)}1m^X+Yt>-zfeW59lJG@Jxy!q=dVmpJ)Bt}SjM@o&L* zEqbCeoOxT^YUZQhHjAF<$hvHC`z`V1=vyFs8a@jfLR){mjSogg_qRIx_$0p`b;5SA z4;%}_q3pZXTgiMR+zR)>gRm(52!O>c{jurEZ@YDVvFm&%$gBOOuktQ|=?@x@r}>G6 z_TNe$yWkfBB~JT)kN;(O6<&kdzae>>!1k~!>B-=WqQ-TllT zfyXWWHb40>k=7TxuKNu7x^AiSmL4~-{vPj<<JT@O}&1t41a8GsP(4dKNHS@i{TRZ9@O#C*rUA6mj%Bi%HK(yT3`Hg z;a?EWfODX?dI9(shh<;|sPju*gQ;&A9BrxhQTkjGmV*P}AUFb!h7+OAuk|Y9SKZ>j zk98b_r{P(65z6_{`Kq*Wx=pp=boe@41h+yRzYqK8ZH-Q~c7~Ci47GnL{62&`;9j_h z_!wx5uYiAb_#WH{)2i4eIO(d&3cy`p;p%1^H9A*BmK$> zbHZX!=ex+dF2f(;Pw)o(6FNGZ{$zq$uLkxpa2k9KZiKN=$M?oQ2o8awr~Sn)c|9Pomv ze;m|$;@5!ru2AB%e?$+X7xKK}P?*Kb_(=TM!yRxJ+y`UeQ5Xw9hlP5Y{88{X=uls; zP}gD@2d_Zo?ikR^=njHI;3zl_&h2gdbo?CVzlT@gKo9<(;ur6w|Eu^dgv;Ox7zqQ( z8{{GX{nFDqQb!c^NFBC*=ymv0x2!|=dkgbB;ZYb1KZU2DqmS9QM`1}=8dis~@Fctp z9eqvwNT};+i(hy6Jlqbed5Evn&*)c$^JiB7fw zMlT<1FvyG>!(bQzzZ_!bwO$e8OTvn;V#R1`(YmhXTVo2{@KYh6gu&fyt+Q|`wM>u{m%gd=$rJ> z7C)N!$?yie3Dc?{PW%G61V%tz&x6!$J73|fCl`Im3k$(R@F>*!n~C2A55g??XNNX@ zZ}kqLtNlM_eK+Bop-#7pKmOIoFZ-_J*ZJr#c{&r<1NMPhzY}%!fc;=7du@vv2=0S^BaA(Il%Yd?Y5fD}=TQ2khW+RGb;L>hF~*nC$zq9@ zI9Z>>#StG%yy&R%9rqe9>$**S!6VH&!=TRh9De=bi*P6$0ms5H_;2gKMjlB)KO3x~icxC!cdej)xZm=V9musLk&V|-?dznp))ud)tZ-}^jX z>QLV^P{&7M*YVNB?}tZWEY$u->Gvn_OLzvx!3*#b{1a-u{G*+2lRqp6E5jCW6^xx= z{J(`-PyA|)H+~(Uj%yHR=5_qqF~&Y*tl<=x4V|Ur6+Iol1OI4v9DV}7fM3IN@H==J zmK$gCZ-tUy=L?~(L$XeI5`GQ+sM{ZgL+RrWjITpmf4t3ij=Z|wSFkUGd!em<9sl}0 z#@`|Se)Z!w`gIHb1@FLgoU@FU^X4f|ufGR6gJ2x|*`ogo_TOcGqSGyz9%h5{CmDMr z{Clz)yPcE$6N!HtFa6izzZq_U7hx264#LCmD|i;h!(^qnFGc5}m!0l$Ay^E$jmiEg zk9{f>UE6xaFDL8G0}H}v7!Q-Ba$idP9P%xI8S&2w)2bhV-g3AWZh#K;RhWFTe^U30 z^mhz=+0yU4*o(rFus7@n?^Ul9I;wm}FX|r#$H7T(8kF;_t`I;AaojyXoJ1MCi;gD=1-a5}sRMb}oZxB7k1OSV+^ zWijj}U@2G@mWNNmR?xN{U-f;}3nPD8`_+nl>JDS!Im`Y%FxTu~ZWszj!EtaBoCa4x z-H&^X{~X=3uyDAk?_pRH2EwP{vryO54*M1OI~+dG=*GdfUo&G{d=KJYfP>*na4i)5 zz*mi4%zVRu1>WfoRQeKs8691p)FW|HPZ&ChahCk|8vikQj=@+c{SGGYL`#2(hpnopW_b`cN~5S1BvVKFU8yP%6c9oudXB1Qjf2Cozb!Bm8FkQ!p5)} zOtuy7OX+jVH%!0%ml_5^?Z1Kio8fMFukmNm(fMMqABCU6(-!|JtYap86Rv@q;TEX# z?ZFY@_6aTKD-3S!oBbyJO=f8Z1EE;{@-A~ z058Go@OO9%{tX?hN9ULOLFAie{iX05Y4I9Q4M=d_$&;94)t{i`*HX!yaunso6w=Yvh#R&5C*^sunMdWRk-5<_9Z+H z-iDFz1WY_iD}n z9@?*Vx~DqBv(V;WkNFTU{U5;ZL6{pxz^%}hFRk&mxO^(FZN1_bfd6zTb?bUXCl0^w zV9qs8w`*>g5BkHxuoQe6)`9h41K0=#!#;2T90Z5L;V=wNgEQbf_&Qt!BjFCX8}5Y% z;UTE|am&Z`>v&)Lq2t?9m$&`V@xJ!!e#hTy|L-+^-dd-7Nvq*H_~|-hk6UjT^`4>I zIoUsZ@qY;ag0Lto0Uw9KupR6OyTYEZHw=eMVFX+O*TVOp)cZYk$HQB&6zeDtpM;fR zO;`tpz=3cC91WAT!hI?Ih@mf^!>^&I`u?})lXV?oJ-+Ub?w9CDpM2dPttWlHNPT{s z3vbV-*3&w^o?opekAo+ut30d@<=ppU{309$<($9I_yZUX<(&V`c~8eV%>m_H*JIoO zwuEx7$1$D=XG6UnxlepS{;yyR`Q^S*d4p4jy2m&0Z;gK^*d6wOFTjy-415_*gEQeA z7y*~VHE;vm2tR;Q?+mcj(8u^TML=aS#3YlfGo3uI%vtdOq8_ z@~}Q%_e&ot(J6vnF!c6(X+6<-3>{z3ht?Avsms^%tn~(Qj>f}@@J+ZC?tnU8{N7=H zdz91d()E2q?LPqfAUFn2f>U9xjj6|9$No2bV3YC70t>(jQ0LE%Jr8^ehHf=FFTr_m z#35sU4Q_zSPklU9oHK_UH=$M zyy)t>>)>A(Hh@iGE2!(yd6(h$4qOl4hx_5QT~4>u9XQOxzHPDWpRf6|Z8Ldu!~C!u z41~XLH-7pYMN-H8s>h#v0V;nm<6iJm8tbv?H$peYgT8x{lKmrbO^DkGcfmbyKTH~*?LH})Gk4;YSy3uDZ9HJs1=H^TCKFnq7^ zp9B}cNcaKN`3BzSd~cD*Z=chV@a1hjw{x<8L?>de({0U&UM83o=77mU-Iu@ab-H;I z-i9_Ew{x<88hz|^kD9|^_#Es9cR|@Fc|K1j<-Xj)IoJt5foCn}!{&F@;vaFu)cY=6 z1FxPic1NsXHrO%FjQ{-Bu*G@9W6+_#dZA;}Q|^vo_>Y0h;M;IJjE0}X(@^WZjC}^2 z4Hv+La4FRBI_@ZbI`274yy)t6tiXR2TnpcaTcEB-_wfn#r4rQsPV7;iI^9!S;LD$z z@pah!oEi6k?JfK47D@I`ee|2c!Egi&gG*rqEK41&$fxs<#qKXLJoEVcWxA$pQu z^3B3N555K$LtRfS_U~cQ`&3^HdWT`*Pt1A(pk7Z#?0zsm^tbp&o-}%UU<~~EGh^@j zh2dQICR_%W!}p-B=PUFa=w*aiU_MwFO1@DRJ#TfMLQmIY%UjgqFFL`@w}su|3$PCy z3`am&hxAX^lU7~XA5VJyRb68lkB1ZCG^q7t9TK-3zqRmvxWy7L_MOb{h9AR2@C4NP zWu4D+pXmh$tNTx3+j+q+1I!L{L7i{^DW}_X1bzYIq4sa_rSbpzYeTv3%;Nr2=De|2 zgAVo8hWL(f2#kUs!Q*MvUqk#l7!QAgsvO5D>=)sW@F$DE;~S@2;sKZ$`ol-yFerVI z=LemCJ^Qi|?uW6K{j&L;v-m%9!PNU0d;$)Kx}KF6jsG6#|GgR4fZAXDj^lq4o`c@{ zpSJinzGU*ZhQr~JAB_F-kA|^V472=X*dIF7m)4t&-dwl^?tuqjTJx2-Z0acm%fd>q zD(nOutV7nL%6If(-NWH17zVZf&1)uqfu9Zgz;^LwZ1ew{I0ydT#!tk5G7N`5!R+L< z<@Xe?>;3q;S;sLL4=ev->@{Er^fvx3I@*6dIy+!AJP1$2^Y9vshdF;Wb%nrq*x@&0 zkNwLq;%~#P(EpAZKMA|QsgAmigs(5*Tj|X>8WwuMjGu;X=Vbrr`oH$F{ucjB>n})M zMc^awDOedk4gKhQZK(U3tX1wy*{^jsobIvgR}lN8{Y6*ySN3x!{s-0mdfG49@0slP z0uTGI>)FJ*KZp99+5ETSzY89M+W%GhcmZC8VZS@wu5xbu@0xKbcoE9EeK4arw_0Cx zW}^2hTme@@|38er=mx4jwMu97C6Dx7=M$Yr(JKvW!e?PPtU(^xC&{ykeDXLfk}37| ziOxFoHozV50K5dR!Av*JdUHZKm%7i=2dPu)lKY0#^%eD{&ur?<1$BSCt@j)1)baDN zds?q8zB}vf1;@bUQ0IFO`);@op0fC#!~Xc6roV0Ji?{vBL)=5qABvu??*;US!ZC0% zoC@c_aJUF+y|n7c`m~hr4ghLC63EvI{GIa~ubz=QB8 z^i_WXc^1Oe@Hg~s!8HY^Xdi|n9V%jI-$<@ z1AZ}C|LM2b&y2UiJ-N+T$6vR^-^}&Tcn9%LXw%D+#l+R2o*<~})B17LDf+?Wndl*3 z0rCaH%CHu!2lvAmcoJ^+pf7sW@fW?ujCDVQvA2hvU{BZ&PJlZ89o4U`>_ZIH{vn(T z?Pu$+^m8HW)_PIsjV9k^c&~ajbC`3X^`xE|mU=#-p0vhG+}WJ}JkPP5w{Y^>`q7cP z(;6>v(oe}N=Uvt9>!h@mWPck z{`vEmeJKRXLMK%6j+d~nhf#S={5Gilzr*hm`~hBt@$fFx@mbOJgLz?5SR7V^I{ps! zC-Rv(op37D{$j6P!1%X-a|@dBYWN=1@oUlj>mlQR7v@B#19bYE`CvF2z3EW$+4T34 zXDjgsq4qzV#(G58Q#~#7n{{-AJzzihBAgFH)%rc@A7@?X;IB~nrPpigyWUT4_rcSC zZ)CqV!$a^W`~(IRHhl^&Y8VOk!&}h#Fy{j5^?94`$3m&+^Q2#qxGHb}wCU;kWSv7; z&m`#Ux*siO>aPaF0?hbT_&T)ZFHhd5q_5E1z6YrMq4$_Cn*71+tJaSqpI*;=&dr-} z8T=S(y$Ez{dQyizcR8s`;&xf;If(rm*cdj4(dfrQoBmt|Z?O;-E$ zZ|7@I-y=(yeg6>Z{p*gOr}{)Elyhe5&sXUE2(Q7skD5B<9EqN;XBTxIf#;y))&7Bx z8C@sr2K^s5cJ1F2`!cu#rYn`Yzu424G4pw0`*LO+0w+M7PvYaBF#Z8)jQ=K$@nu=p zYs4>tt1NoDeu-~@&d9Q69aEw9Z|-G2(Fvw+(od~-0li=0Em*9ysUwhn`caQ3y;tyi z7w%ATI)6s$%L9wR;!ykB`YikB>AcK9_f5DNZnNZjyu9f{mkNfVFw2u>TndK5F!=p>QIc3O|OrpJGpI{wb<&(wFxv{oaFpA3Oq&!yn+)il$%JVERgCoEheV z1z>SF9qxsCz24^EL0(UKE%0jvM?)RI)l$z`{O3W*XY>CAos;lu_znCM{#DuZD_s@C z{IC#w6g~l~zzFyi)cx}|{}J+g((8_24>%L*_*hFlv+;iiN>x#V{OA`byBZK>n%oaDLepAz>_eo`D#=*b<~FSU}M+} zhQcfKQ`V#RV=;Yx8%DxUU_A7zXYx*m5e>}v7R=nxjQhf(P0U#5i&pygs<;0>>&<0- z3*Z*muKs^sZ}T8CAJ)k50@Ul*`Z=F5{`X6N7JBpHLii?p3+{)qUsAuW_seHZeV5=3 zcpGZ}hw2*tA}|2H4!1%%Pu|AQBu@K_-CO;V?5CW&C=cgY=j%v(f4GVG58-wn^&0ay zX$HH&?Uu*Sx7aVkKVinErteu{W2o!@u9@-s0cL1!#!tiAus+oBEn67>4#9?f;0Cw} zegLCkm6pa|>)k-_PnZ|~Ca^hd3w3-?bo;}pa29+O&WDjOOPkc|`GEP)U_dM5_h9S) z?7toVk9>@;fqpI6$U{Dx{|4f8e;#F>kHZSE1#AmD!tU^S=xA@&Jq&7nZ*@XNPYGmc zYxG}&Bj6aQ+#PGMzYq7qWAIa$w_WP{HJf$33OB*+@DS|Z!Nf0u!5z&w6jthVZ~EuR za~^&VJ?Y!xZWFKjwG_K;o!Y-1>v|S;f$ch*^>%{cFrd4!2SP`P8E1pO>iep-$Eda5AXx}NT0*3lG-POw_1l5^PfR?)B3a6R;-XN%iQyw2Z&dN08%Fhf_f9_`;4 zdpp=04uPX#G}Q4W*yrEjEtugslP?P_3nP0Ndn~*OGd*wY*eE?Gx!;)X$~!?uo7mi5^5>(Q_Juo3iC-=@pfi+nmOczLb%wxPgUq-fECL^f#i0|fhgyFH`tQOu@CzvU;$JlRZo<1T)Pw$D z>g|SqxCj3c_>Y2P;agDGFS;M9dL)nbe-6LVa6Fs^wSNZkR)@8q6Fv(Y!a&w1eb;)K z@XHEwz`U>kd=%>VnxSSN>cAdw1&oH5p^oo~J=4`0Z(ouQb%;3nP9C zoNdw5^-FwnbV5cLy%(YOZ|7w`(Fvq)(hsfoDSBVQZ{T@&5&i&wf_Yvtb^6oqo6u9f z;jC*mjDQlS>$CM$_RG`x7{U4{z*k_4F{b`@up@jPmKtZ~gWy`I>s#|L>D8i+&@|SQ zb(E<;WUS$fP}iRuKb=qFJgt8Y@vp%*;8GX?SHMVk42H8l$*1cZPTjL%1eE>J{?}!n z#+!9N2K!7j<0x2TvKj068fo;e>!bcVeA~8t?zW3q~55kW_%QC|FP`TTDWM6Sx*FvhH+5G z?<6i3o`mP2*3t17@ymgJL)b*;ooedR@j5O5zor&{(H$l0!B66}exqq7?`XILMnmoY z<8VQ2VE!W9G-hDKN(? z#-3@u;e$}e=fgh$mV*_c_80xkUh0c~IC=}NfI8m+^4x;y=9=}?gQH*=)bVd& zUp~*oom^me_cg=%uNw}8w)iID#;-Z-4F|!r>aVB1e@p*K>PxGBFAx2B8vVMe9}$do zKUZUqQ2Wu9{n!O%e>~-nK`-O0W`D90pB8mZzDs)!ED7X#o zgnB)m^w#^RC+9}?MfSnA@49YXpFU@v{AJyGA9WuNTKe%R_BeO}esA&r8GC6SzvW>? z%i~%6|0eD(OppHqFsmir+qegb_k;cxy^+|*z%V$?;xG1P%)bLyTKwO@?ybMgZ=*%O z!5ilOIRtiCY{q?|K2PE|0{^jaCk!KQz9l{yzeDg7_&0ojxU5Pi(xNARxtPxb8^RG# z>Jpu8*!B8iu%CdR!;9kors+?mrG`4bDf2C1Cl~?;z~wM%iP5#CQ}iuk?+b%g znDIEM`xSxT26zY_ho|6|@DjWNwVtsdJ^JUR#_K|2I|{-Lmf*E{C>$ zX}!Yq)oJPPMC!~)zRXbaOFq%j{h31i#kWm;KfF*=!js6Pqy$dD3JTAE4H2 z{D85vfSuq_sQt%c_s?u*o`p@IUshwU4|V)o#7Dv!nT$s*=!98B4@y3*Cpy`^)Unm; zZ>d-EN?bds3;)J&pe24f{_|nQEM^5&V09>cYlY4W7Cq6qjh@z#dbEzMzHiBM0bYi; zE%}49nfgz{ICvReh1a0ouZOZ5OKIqY``~F92QR@Npw|1JjxUhIRP-<`0Uv{v;IA+~ zm+|l5XLu53%dJ;t#?E_;|BCoCFiz>|tXXmz%WKpb38n5X)cZUPRqN1t6VaOrXTWt2 zYELshj6ImVI-ibOPTy9+b#MdR3U|Pd;2yXS9)vOQ7`)efI^Nehb$o_A#*zut97T&AAj`y`L9lw&g_P`kUxuxFK*tfv#a3|aiMNjAR6esx)qod=$#Qs=5 zGZP4NKBPU(xGM9V@|*DpX!Do(Q}|^>*XI8Le%|`G^*42Ng3rP6P_Lp_L1URz#BeU0 z52N8;=pSJGoR1j#KWcayno-ifq3F+n^WZ}G7L0`T3YpmEFiFW=iC!4~_p0|EIve4K zFcbN+!JIH3EC_XdHob4ry$FAVI^O1&)_Od}J;FM4-^W ze>ggCz?JYbcp9FCFBCQ%?gMqdH{o~4OMh?kev96B@ImtDf%#zz*cM8i{Tb_ee?{+4 z_!rE8|GnyaTX%i(1;Hjz*3p4+4=DREm+?HP*Y_Cq@-Prqh4(w3Z5^^NdL6PadL4_Y zF9NQCdVQkTm3h&7k?}A%6-u8LGS+&wya(_<20yjLe~tYnyan%C{B3?kcz)3JiTyF= zOTjW0f16)pi@(@y&tJM8@o$4(JJbJ!ZHaEHw=t@Y?Qse2ds_rXK(82l7E)R#?f3H}jqB~;-Kn_pV%(Q&_$_jh;~ zX5{ryRw%EJbbVS+{Bsbe{iRRd=GF06sZYoMiCz24>$y_Ao+}4i!Zxry>s4y%+Hx0!LWlZGLI3 zN5>r??@{;_yltsR$DhIPoW)=IV#}lZwF@2Xf5M{g_pnhY2us2?umkK0FT%@E=dXyp zk(d77=6w#mp70eo555LZz|W!7d4;jAcPREza4ehz?^WO1x(|>q1|EU3j?;|KL)nL1 z+_&;Ty}niWzXzk>7I?q&+14TZqSqn&qSsM~^^}06E$b7#v#dw-t}*@%W+GntRKTKV z%Ud1)+VB}mygZK#WqueOZSlAHEwuQHeGl^oV2lU#^w##eW1GYl*k{rL`Vi z_j>Z`*FExjO0UNje*ztO-E`5i9`TcW;%Cbzujjtyb>c;M6<&w=dA;Wk3&V$Dari83 z2=)4YQ2Ku{&WwLH=m$ON-LE=Zk*^Ev4qt%%pk9ye%TxH(gNZ~^ZgBE|OBY<^C z9if(bO5pc2bi(d%4SWxZ&TNZbZ~P{~sc;6I4Ohw-ohXZ*_~|<0h?ltImiVjKufyNq zZJ7RX)A2$u5biIncfyRHW4;&c2ZzB?a4eh%H^E%!R=4PH$L}M!7v?Qx@=M&amiUwS ze+AFNzo1@~%|FiKFLl(R9;u_3rJiT->jb;QQE)$ufui%WMQ;Lr)8Q<*8%o?NOT75$ zytjy#INSd7^3EYoHylGp1sYosR)Mu(5DbCOLk;n3!9?zK=Au8;{sHNYr8q1DD?sgE z`vGIA3+uxmsQp`FZwEu*OBR2bf1dgNaFE4+FZTWLFg#}QzlQxfyaDrN(v>sDo&n1H zrgB;QKgw+4GG;Mc3_pfCOCRk0;ZQij;{PZ1bXkp3CYTLseX-|Zz5pz2@vn)!HmnEb z{ZZ{0zX_vw-;%sfsUYt=3V@|x1PtW;G=4cvr4?WgsO#@ap5AZ(90hIuAzu3LllNUw zU%aYM>pSrCcD?d`B+>P>-WSjx4uh%3+jxl^Py8e}73%fa{9d>CAE2*?VJtiiwfN zvgG>|`n<=fACbiM=Rz1Q`$M@Qaw*@v-BU+=OlehE6;eAJWt z60g^%^FM^X?oUhX+J7wljDoVxIQ;c_(eWeIdDf2yIj`QH!?fx*^YWZ){jsY40Cb#b z)U&N$>q@=49({k2`-wdM_2X2>%lls)yst1Te1i9bIpIai`@=rt@h|b>zsTaR^~KMo zFa8D5^W;B*`n~mE$omJf@cZ3Rejp;h=Zj~Y!}9x%$BC2QpO0pI6H43?OT75m^6LHC zOg%f`9w@(u)c*U)dmMfa<@Y*{022`eb-dW?@B^Y?c;Qj)X~x0m^nowJX>dN=0e8VY zQ1rKw$DzJF>3NFRb=)Oydh%t0vR--LlWo0z_}7G<{P&P=Kl~Ve0?)vk@GmHJN?#<8 zLw)IZPyXrny+uZt1!jlxdyBOR=>(~^i8pBZQXCgj7ED0;YdayCns<)}<4y=k_4HYNrmU^d^G)m3rTZpB9-s(>w z-wYTIea$z7IGdhce-G9%6pn*oQ2U4a=sy$x{qSRW0{WFR8A4%p_Pv5-|3z;;@eAQ{ z_%7T6b^SX2O%=bvL%igZILRk*I-jksX!7{Fe%;5D=wy1#WX=X3gt=is7**EzNnYu5 zXG_0r`MxCYS@;9|3H}XrKRoGutMq>JpeOkyPV!5f&Tp$z`sZtZY<+G(zdORNuqTw? zTgrW7E%%QoD8HZkkntY4AI3oWec1)ZA5}0bz6^hbf51CX9%l>L_hpv-x2>-T>j;2l zVIXuu-G5JdC6rz@4|HM}jWxssgkFV*wt^c0-JDhlL_h0AJ>l;nK!eBJ_ z)f4a(JOi)5Yw#xQQQjy&#CdrbR)*DKM;Hpjp`1fqUo3TIdC-;nhCGi+-+j%e{nx7eeJtzpHoyG7 z`2@P3!P8LZ7yD)AuflkXzr3FycWI-PA5MVrUe+JZIn?zOMW-Zu5;lU~`d7lgDy#|R z{R1{VPjNbbM$Vm_LpdL|_06G3$^|}?8o7k@LP-jE$qcP&!yp$a=ujpn_oBl zLtq~-p>uEnv`K2zs?r`$VgRj9i;c}?+YrTQ^ zg~Fjw^wSzY3Y~Fq2Gn{sKU;pWds=tm{Zl3M6u*i(SHt!2L%0p@flA(yR{ag=ZGw_d z>uWzx^^D^FC-*_Q57?eBe0|;?qMo-O<@wm#=WV^dwC0~j9r|^_z3S`u&v@N>8lHvc zV1_46#Kor!Z&o!7tY+w6n=uT5(eP{?GhfANSf{SxRA@#?|9V@`x&Kl8zB2SX z0{YWuTfZYzzmqy?uA-EtnJPF-J5>hBK+!;w(?%enlX z^Z7e<>vQQ>-9!e$w4PH>`G-|B9+Th<_-tik?*Kj3H?pSj-v#|@nQ;L8x9Z+m+b1GWD(AN}?IEucPGr`FHII-9`e zupJ!7{Z1Z7J)K5jI<)B(CcYSa0+xrhp{MxbD!z(`c*!Sml277vK3iQ4$>Zz#bssyS zGaH7(1#lr;28+^f$t!(cXX&>sUkG|V;UG8^PJ+50p7i=Cy)hp2B)`N-eu>lhZFNfj zeC>~|&yR5*D-Ekd>GNg!U4!R!d7jtx+VtO{FRS22_yODxebrl|^mcmCll&4V`6W*0 zx7C@S{uG4Lzb7sI`^?M!#ZkYv`nq1JL+@i9>I#E#@DZMmo`mvzB+nCCUv%{IpXgpe z_g8oedU}48I`s2nAg>eTbw)H}IrmlQtK5hGP<{5b{vcleHieyFHy8tB;TQ00cmXD^ zm)9rxEYJV4p3LfXw72VNiGCZ{0d|IZ{h`>0!ZC1y#XlB%#u`RR`un)$^?e=u>cVGX zL#XQ+gnbAc1{YZT!>~_+v*28dztd?lJ_DP;7SPNl{cAhF_v>89>-@ymuj+iuIgs<6 zRGGQ5)t9M?_B3O$Kg4`tSj^%t_6p2bgjFs69kB0$(R@E6>+`gp9>n#7!{IEbS7q}b ziT`Lg0ZxTBJx_5u{}~_GcZ~d}U>v*zwZ4Pd!b7J=Ho680+4Nqw>p`u?foC9Xd4 zw)Bi zI!o`9-Yj%px9CYe?Ptp`>x@Cq)AOS4?*`&@{7FlF+kEuT$oG5sVO`i7%J=nC88_$k zihN(M>pjWyUZr|w#t9F@FX2_F<85)D;U5oQe%3_B!<*2PUgu|wM;|xg0S_Oq}aV&`mg(4yS}kBhaqqv z)c&E^N5IkWZHxaM>=_#vr2tq7YJKs`&HPvx($IMJguP)nTmqxuS!mPGrt}Y5>JdMw zCmz3B@NX#f+3MMCsmGIk7UC+y<}j`Kw4Sf^#!F7 zX8pGKq?MX0@o!JQj?mlqx5#%ImLk3a)cIl?8%t;t!=~)p_t1fk^g-{N*82^;Lgy&e<*Qv^L^x$0fg@fTpI0pI$>qs*W#6AOt z!^;-`i5CCYu`h+oE%7$LRrqg*w)`bpn!4MzGAz~FuoB$b#*E{j872KYhWO1e8vX!( zgLh#j@<_fUC37YA@9}e5>Ug-Fjx^)$#CeK8(AM~W4Q+a3(TjqoE%j)<%+!|)`l=sC z9_f32RqrLnx(h!OcMJXv!`VO4k$OeP+j>R+SEZlVaz32cb^gaJ^&e9E7fl}NyI$`s z=EYC1L;IIyKPtfLuqJE__qI2Yr(yO^TFHz(>9w)wMRqjtyI@-NN1-21og+IKU9EqE zJb!t}Cwgz8o2Z9wIPqHlFm~Og_2_JcJ76E`ja0g#<88g7|Gv`ygt4wC19>HXe(Ek` zsedK@avnmcSNfv$pI~16^g6VEG0s&aIG{jCFinOMEc#y`Z;xebMVf z9l>2pUadC|-NhdCMeh}KMQ;qc)ri;nL$K>EEJY_0u7ytO)w-hNZM~xZw$k6jSl4q+ z<^P;@eGj$&IO>%B4wU{{{BPqgetI2R{}=i;qnnwDfaANH@yl>BjDowN*6T=oSJ)ql zzJs`bTfdl>^=7~?8}x$`uh(~g`NQxe3=1)y;c&!rI?{}rG2aq)hY}yh*w^@R#7X>Z z#(k*6rgs6qA7MO{c!&C$g5FFxA6D&Q9E0FFsKOmz;(rzv!0&#?_dz$-Lp{xyZx6dd ziP!ai{JgOog|YBccp6@YI{rHL-(Z2B#-lI{fKR<(?7?se)OrO~{G5A?e}nk1q0YCR zI*!4wV833*GYp>Yrz6c+{NsrG(Gnln*Z2p)k?6ewz17c5J=I`fAEPV!BQ5%S$aBa; zKAV0u@&uqS`9jds>sgFlcda2htzcW|kh-DhijKGSi+&TOKS1)Rf+ECD-iz=yRPK&u z^j*$JRBz)E1GWDQ{DYOB?3>O1E8^l{MvMM<>{5sBlh*%*eHk{u%!I*rq4s|Qd$5=O z9R?bO&Ts;p_@Y)e<1Nhp1pkDO;$I$CgLR-yZ|)$YJ7Tb57(4+_!p~tGY&<0O`l^Q- zdrcSudl09i7BPPueg!k&p9>a*0nnz`Zm7vn@g?ej+u><=9qM|v5WgKBfOlYe;&Q@E z9`p(@@2MWq(e>EkJE1G zjNh7SxE5xZZpMXR9k?4lIK#~Ae`l+DbUy8GtM^0d-lpom-}UPJm;OunF0nsr*caKK zHuPWi<9Eijr;H8H;WNy0ZV}z1;sS|K)r+$(I}Y z!|G7ie}r{hgt}kdiIaZGd6zitpN_ssKSlStMPD9Y(r@wm+v2bHL;P%yzaHEddX6%c z#*8xz9Iq=kWAPtITqyJ-UPtx8J{XRKGcEq#jz)p|f|Gg@>zZKZoiIf{AN`AvNA8Pq z9}MJi{UVh6qTB};a6gpCo$QmYr`lL!nGeJ1YmBA8LHMu3lK;4&u)E~ipPwu;VU$nl1I9t5-mp(-k zcS_}z`>gaaLOl=4eN^rnw&%SLGj(P&4kuqE+z6$f^Z(^~Z26n85C3*QuCm_z;iA&$5`T}k7p3P3}s(rA6nDriExW$ zf6|jj;v+ox>%3j??+%}bZ!6;+aG#g@8SqQgqmB-idYsswfem0Ii~n2bzXv~nQm?F2xYkGi7s%TOz6dA7 z4%~2-z*d<|rtMJj3ZGN{5gz<6lczLsqI=My?~h+0SQG|8sYB?*F6&syc%P5{>n;9A zuwQ}K;4c>cOFsH*-9L#hNPmjJ0877Y@fj`t`LTPeul*h;zKKQO=Kqw%KNkBpFdkOQ zY<9R8JPc*ug?}plC=dQRPc8hrS@doG^)3Dvuvg{@JP5|Yz^tb7FevK~N_2M)j23NrI(2>ooWMd&&(LWi9-&(i>o{;!|x;sQK1G<^v4tUzn1cc?*fsFCf z@ngJ%G#7BSW+X?c23hULNZ`b!^Mk^hFj zsUWzZ;c|EoKF57^u;uQ6+O`vogdH>otG_oqSGI}fp82AKwt8RZhrLY!j=~O zap=s0a&AP|rXzjO`z8M`Sn-_nFR!6I?&O>Xb6&^Mhj6I%MCS?g%E6lOarBqN!RSUn z(UpDF`gPE02Iamax}tLwJvkT8S?Uc#XEK}x*Fl}%=D*0|FFKo;-wxw^^gnFz7oAJY zUxWGN2R&*5Hvc;oebMnpuNbUj(YN`RwfKw9Gt4)J{eARrZ}AtMAQo?^Z#YzP~}5UBHcir4<4(;L11a10y|7eHU*wZG^rLT?FN4cEcl(ARkFFFO0s zI}Crc=-d2HS^P!kI`hB7Onl#w6+R4geV*dAzvz@i?+I7~)`Q)kukqTymeTFPxEJgT z<@+po-jwG#c^;DIQLW#b*I}PPd7WYV{!si!;lG^M4K{!An}z=Z{3TA;BYw+OJrbw= z51I2PW3w_%B5X1)w;13SR6FbuAQ>!AF9C491MfH%DU)?ZW!Z9a5|g?b^IN4#}_d( z(uaA)wWk@2e~_2{rSKE~Sn9ZJsi!jj)!@^xHq=%1qR!!PESzAef2fcC6Um#YsL3pS zIgU=nlEyCoPlm2f;!mUFYy4*7Bwq40Ca=!77ykn=1|EUhKMtMi@DF&)qJPatf9YE; z`j-bzgwnTWkD1D{lrlU8b-gydu4&Y(Nu0z>{$iH=jqz^=gJEl^>y^L1zZ1Q^@PI{s z8+LE~rH}K-U-$`A!3cO1{tRbwUtMguzlzQZ^tP2YdZH^jd6do?i{2`9*24|(BN&9f z(^g-`eFwdvY!-FM^hdl1&oDSu@%HJ24zuT?#M5iBm z1L0UG|F4wfS%B_7D7vzbT3>V~p*I!22IW2_c|`XY^yQq)x%tOzSZCwvAr zfI+Y|YzI5S&afBk3kSd#;V3v3PJk2PY&aLrgA3r>a1~qw_5O7#Z!FKlesB(qfCr(D zUygkhTn#tCP0$was9+Q`z^pI_)cI_F^8ez*5;q`FNB-wL!-?}0Uje^h*nvE@d^Pc_ z1M9*1P*+(7y{fP#tOK=wB_I83JZTi3fsNrn7zUR@ov+MO#!?Oj!YZ&ftPAVICa@(8 zfm(kz_EYdYylC+sUdbq&sBBoarePDP^(W&$3(kcRQ2xH}ddBa=t#B7S4nKuoz%%eF z{2gYZzHIP8m=BhKo{7Q2xGL55_~_a5x%{g|p%7a21S#`=Ru&wxxg4XZbsy z#jBVOmxsFF-=KRQ#zQ~+^TMK*cw3%2)paH_w)rotn!0}t@&?0Ba1a~?XTxx~04|2_ z!Hw`kxD%d$C!zd3&$EoL!{6Ya@Gi_l{SU!HQ2wrGAY=J^p4AxFfX!e_*cP^j{ox=u z6b^$kVK}t)^C0t|S2O)S3vK%)_5>-t1r{5-8^CUwk*i{VEw7XA)(zLVI$fM3FM@B*~O->PL4Gt@TB z2FpPw>soQoAJ?XdbQ2&?cpMlrXnC~g_NIqS! zLw(i3zaFd)gP=owEu^k@;X1ehDtE^UAN`~8xAjBo%f94de;Kng`@EO<$ zc7QhhCg`+*W8nli8R~d%?@usxy}mE74{Km%M!^|yE<6RlhF9Qqn5CgsHe=EMnLIb4{GH5XHo`(@J;ds_P$L%zfC7!Dw%HolrAz6#j3qvrecT5R!p|-H8=b@SXD>VeW1y}t z|5kH;3d2WXQ>gt%VRvpbaiMSu)c)h}p9m+xC@Aq^mU!`#eBx)zR~DTL@JU$7qQ4OP zQn(6kg*)IbsMlwU-)!+0ySMtWCMQ|f@zpsx2X?2A;rp6aVf+%vENd>;0K1E8*VH})8K1jbwZmr&n3@IzIv)|dLL z(&u_`q~#kn|AS^9O2bO<1*r8OB){zQmyExIzrZ%^cSk7aU>f5}yG`DjQ0kU*5yAOb z55I^>{k+sfPx8IXxB$9U_nWxc(4Rg?KD|C$y`md{uCCvf$5Z_}pXdjw{%qtM?+@W| zco|mPW7Zj=*6C?|I$u-r4ud1%bm&Ph7{9i#3+xV`heM#~hcUii`u))xr1EKf?dPeU zW$4Ai(;oUQe!1AE+^`_j^~9lj0bYXf@K2aEjq!hbh%bz804xEYwB+lA{Z;rHTw(Da zgZ&ldzr=(8bmnKNc-=qkC-vC$SE08?^7A;-k0;q@olov(a)0plaavB@m*jEl>3-3N z`Ub$4;7F+J+lRfc+j4&$MBHHb5*z__zSpt80hhqd7XJwB%i&tM z!Q#J_`gg#6@MEa!iDO;g!)%94AF9A$coPQjAAY4fV*D~fKj;r%hC1KFN?-h2So|gb zBI2AL@{5iwzv#6_uM-@i^nWFfC%v}#^@Af|oR{&kF5O>Qr|#eH^r^~G)BhT<7L0)J z!ti6pFAi!wnU7?ChX;Qrx^rL{{-Ps!ezfG1_+i9*@}Gy_0(c3gM?W*n3jLterPnL* zKcVBv|0Dbk!Ox)R$U1)`ug-Vmxaof^JPBWoHTKtGy3frxE6jh&jElk_p(p(wCrsQ0 zScw0a&y)Uk^kY5bTZ`^TO1~I+D#C>@nDq>T?Y=Peb%fpEn^3Q>t$$Dyo0&eCsLXBqTIvaT>V4ZaK4!uQ}NxEt#Aj>9f`ix_*-+v20%GW_51 zps(xGak3s;e(6&n{gQP}BCoA4=a~N<{s4c1e?ncK){n){liqb7^-kgcl?Q!YpN^CD z*z(KcXeIrNgqNW_Zgz6Mn{#geA9L>kURAlgZ*K^_hF+wEDn%fI0#YQ>krqLUB0}h$ zNbg1vL9j%Tjub&b6e&s(L4yqQa6CWMe&vXZU)B5cD`VtK z+~0cEKI`Hy*X<3i-#3`+B4fUiC+{mM(aCiydGday=XrvpW@12^qIN8X?-zWU=Yd0ulng#I`@ z1<%4OP}ldWw~Rb}p4JlI4!!DM_%GGpq# zo7desxS97q>;5CYgYd77&u};rT6OjQo=4vS_JF-%KPdjPk2i?ltKR_fhbVvTr*&TS zWiE1mpURxRf$QNm7z_78U0?j(pUxAVRo~iYDdsNMoxE@8 z{Z~XM*VzcJ(=o>T#W7xAUiX!p%97jJmBuPN8#~< zZk^|lqxK(5d;**fKZRez+DHHI{PF1K8u=%P>-xD5JKslOS@;C33hTgz@LAXrwu4<@ zZ`c!p-Qh481xLZra1xvb--ol{GPnwU1NHg+p7?%v5I$vGPpye}gMHwJ zV{Tpt;Wc;*=8tp7ABG!GxV7%*LHm6{ecj(_;^*M+@J}fH>;80J684i0W`wz*?6(B# zK>f5{*Ad@J_*aK@U^Cbf`s??&q1Sa98Fe}m?*eF%#5D?eWH)Cow2Pac>b7Bu{e z5U&cuU=2fGhj<&<4t6y367R|Jm*6Bg8_tDlJ3cFatf7D3@Lf%OH#`tv{wIchC2@cK zMYoo`b#N2>9&U%y=V{iL;Z>;T6{x??KZ&ou^+b1$ykDX2GtF^#edmG&psw$)-*sdE zDe?8Up6D`@mmTUpA1D3E;#!OTyZ)E!=nB>0*9!JPmb! z+CMM(;$PG7A4&W*_&U`71%Gz+2b^}??9oq2^krdrSO+$Q&%&0l4eSW}!@+Pk90e!9 z_u(SA6fTEf!SCT-cnY3{7vM#B3#Mk@xnN#c02YGfU{%-a4?(z--j#VI(P`4 zg12CD`pOTBz-q7-Yy&&O9#HQ`_NiZQa-Oul-#NGM_Pjp3z@9J?4ux;RK>e3cNAlO= zBliQVzq6`8$&>neKNs-529sTI`wNCypwyN7(BH=Od%yV|F1h+2!BucQ{0?r1ad6}B zE>G`I{JNlfAI^gQ`ZvAo{9C|Q@FmzEE~O8vAAjrpLVtQc5|@6w>RraiU%kvl=HG^W z^?<|SN5;H$emf(774c;Fh1)3j%e?_CNp8qLy=io)?yK?vIPvX@% z-UW{O!|C6Kx&QpX^OKSH1gxz5zG58%H^77N7}Wcbeg<=V41CwnTRN$){iRRY|7Lvm z!qZUCBcm}7?f1RmSCF~(gW`Xd^=~j4b*y^QpZEouA4{E|jJkSWA@oxUhQcjS>%)lG zhT-r7xCDLySHVF229h@vj)bqlH{c8?^}l6p)suex_4o3NrQRuc8fLkA_t)2>#4Exu zSPzE7Pp{oQe?G^5fgyi6-6Jr9ywUJ&X!-REP=ABE$*7kR>OLwE*Vj)c;%~qoVN3e3 z^pWV|JoJ{YH5Xk^^4C#E&r$1_(s!xr?mUFTvQX>8h}VF1V0S~G^oH|I4s*dgFa#EY zrC=O94O888J{e(t7z*3NZt!P#4*mvH{q1~ww;Yqfv@is&h3nvEcmPgHa`%5gqyznq zhT<1Ro}T9sbb7zP6W99-^|`uHa5LNmwSPt8Pr_R8X+!Tdzhdlrw6XuUnDaaEJvbfC zf}cX&?VO{}=G>bf5kijrmF6^7@k3i}Wei&6`I4GWuE$SHbO2?`JRZ zAK?*r0_yrYKRBs-Jv{_NU@xfkA)mYZMsJQ+;ol|rjQ{_>@U-#&w_l9mGWZSLuM4rZ z^xt!QJB)>UpylV4NBTcF#ZPD8uW}@b`=Kpd28$(i@ft7=<_wa=f9DXDf7L^u1$}my z8$J#z!piX9tNDCZei1`ombkzE;pm&f7EtEYfc^D^x<7yOTN(Ls9@bEIGnDfp=b;?u zBLYq~&d+g`ALXIfb*rGS25Z39Q19R0{Q5@zhr|~^d5-f2)?<^oiU(nMa<}dQi=}Yu z=b`TJ1C<}M!w8Zo+}+s zeVME5Gn)N=3x6~ApN2Y;f51bp>rO?#+VHpZa}50)`dJ4zz|Bz4b8Ucn?Yo2gi|`Ws z1L}HK{y{@8&!_g+U+Zq-Tfmr?rB7w(JEn3w>JIzB8Bp(6o`3vuYIme=8pk*on%1pb z!%6T1sQu*m!bJ=pt*?{LMTWqU@O2msC&Qm%s`M@|1j=&;e`kFS-hfFmIDI_4k1Km%&v7fQ@9h=Ej*ar{7Bd{+0O8>f_aQvIW zR?wQaS3T|T)xTGN{n+O~I0C)`bv>EO7aV_y^VlEC^Oj{F@yschIcdK)@RH$$E8 zm8bRM^S|0(3g+VP{B*za{L-`Re=N_DodC67&Pz>oUfQ#cf_tIdk9u>yv(C z|B>j#e;RA;|H*y&%_d*^wfy9~9%YUv;LmU;??-!}Jh$;E>z|-JZ+-#iaWmBOuiqK#GtL!HWdE)<|{0iVJ&)F1z-JjNZ)wlLNo_aE8x$bqnN$6(5NY2Y( z)0}UpJOMhP(|bU+cBb%U{2)KIe6zU$!b`7Om~1zZKiU+eF;UQ7CF1>3=nP}h@vOyl@Fd|sOhKY&YMsvORt3w(j! zR|iAwABpb>7zIbcG4LIz^S$!4UVLOff%YqTE9pnBukP$W5)OvL;42bm4pX51yg zt~+^u(fc_=ALn5_yaNA%nYcc3!~C!?d>9sok3d~d`v;**554^T^-FB@2_7O{K~<5^_P7=g^$1c?ts1v><#dO0s?)Np0%iKi&A#s28Q_#t^Y1b>FwuRuN*DFw^H3a~P)4(q}OP@ZSr zf^};c0XxDTun!y#UxBZ}aZsKwK9%(h_yJr5m%=aLYPb<@fj_`q@Cf_~o`gTc-{2K^ z6<&kM^SimFhUs7im<#5GA+R7U1s{cv!Se7aSPRyLPs0|lHH?7z{Q5#%WF?G;!3A9W zKfZHZ>;ER6l)My1ep+g#*rd*NYt9QxGP zDE2)WPJ@dIDj~~gL*EqroA7?~r{gmV&V{=QpXBNNh>xyc1OM8v zF02o=z6w?(B~=S916f9FdVjpISV`88^s+b!rCR=x+Qe0#P7w)FAYPX_Fl~;dqs= zvzp@596kqMF#O{d0JWALr>`_Mvx?p4YLwuBETsrJPc(Yw^+f6+&I4BkT%$L9Ji( zsEfQ%&he*;D4x(tw_bsMHT)XxfP3MO@CZB!&%r<8b$AOVFYD@LhIwFdSPDK0E5Mqt z5o`t9!1k~!91KUn32-u;4(Gt7@GJN|+yQsPz3>#g2yek4=A06yhPhxNSPoW&&puhM&U~a1Go6cf)<~FuVva!$05+ zn34Hqf!SeBSQ3_jW#MD64y+Fw!Eo3fc81+x5BLfk1K)!Btjc+wRo)$$3*|c1dO6;g z>vs?w4x`|BI0ssOSE(m{+t43|y1w{no#h{0$vI4bli&Dq zEnEj<;Q^@oUx5E2xEM-(FMsW)`}L|HPG3!7GuYDTw=ePG@Krd@(2uR+IvfwD!B}_( zmaC_C;?{EfH*{BE{;KYHC_D(;)o}5*pmzO>{J)`(Jn4pJP84U@?5;Jjip%b^dSY z|A04CzSife=^}YxepnD{z4TF$ekwzm>-<_SqakzC`PIsS za}H~uJm*{b8cN<=_*-44Pj0+kzQN}^xD_6RpD-t>Bm0&*Tk$^%_5P0Fa~{fh72hrN zDc7O+NZ;Bof0&CDf+b)*=#D0SxAYYZz4(N4yagN_puU5l7oRAOkAaH<)K50_;q)t_ zrB7q%#iuaGOTzjA>MI(0@oCKQrm!t+5Bo#CE3bU57oTDHje;K<{+2%4(2LKf9OwTI z?28LfzuwTFA};^@7XQqYPd=x~`{u*f9!LZ0Xi9 zaKQ6!U8|L2k{2A)Lbpo%ejNYs`YzT|`43Y5iJ9&v$qR1a)`8YPK%U;8^db9`K6HJV z(;PLY{>*EHF}E>IoYUJdrm0(RhEmtxxkk0sow>E{Zxr)w^o%?H9Q5jEK?|o}0(F1S zc%1j0oc~5`osZS8^!FNlz6&S9Eihxa^Vz}mG5tBW*57xdKLmeL`C8u}pFwaW9BugL zqK`)KIT!)|AW!BqpV!46sO#q?uPl5V)`XV6D91}c$*~`xY3c;J zUu$lY@tXnXK{?M-=T+t?bJhJeCr_W3LBxl`anRrVPUyQq@f+#kr}Lz~~^j znmU2b!pcJS`Ap`_`MlTb)vLcCJ|CupT3>{C zODLbupwM=`NXl-^@frc1?BVaLe}@1KiSXx zZ_(ZB{`7vn_wcjy`^h^5PeVO#op0sIezacV+Sltm=H>hqfyH1btPbzDUM76J>Q#6^ z^|E=?)BBPBBv1Bl_2abi(&AF20sYuC{Q zsQZ(+-rpGZsr845-<0F*KLynKA;k5(a<|n=xBi)U(FnIL0ZYMXI9b;1oGu=^RpR$( z;tSwfc+k*WI?F$lK8C|6e5J1BT{G%Qen;}X^dr%|3ircH@K5*`^ify(N}P=QN%D{Q zsb7TdOSl1wkL)vfdtJ$`-{|Og2F~v6)-fo9p48Jq<6Pe3QxCAaY=CzmjL3kXVHuQOV zxeklNFjx;ZgwMjZuvMhqnOlorL3IB5?Lxl~9)uU+Wmu28R(*f#b;nQdLgLboSG|7t z_^X$>$h>8r=kWOhK55KX=g0JRJJ<$e;Xe2yJO)p~>rn4Mj{3Tvvgo5=Gz_B8w6Gks z`j`EQU!eJusJjH}x_UoD=x0AH_L5Vyg)x1|=<65>7ennAdY}2?KM?=n9`%w_FSDxK zjdd^BA4Wm1eP!&YcjneoFD*VYC$DmfqTUHbW)xR1PGI0}x3pF^vE?H6c%cKkz)x_UoX&|QPKVDLb#bZd#f z!SN+MV)hYdy&yjlc98M4zc(I#>V}fw~{bUx(igc+lwI z(n)>oFMY~>rLR2nQw-{Pzhul?`yD3#EIbb{!7K1O^x>Ps@Xw4+_t$~=awz?l$LC4d z30nQje#I}){Ji*=HtOp6&qntVTmq90amrwr73PG+p}&6n@jV2O!=K@KcmqoPf*$px zUw{3*{Jx@Ix}k1oQ-?Wz1Gm6SQ14&#pDFz@LoYh<7oFuVedVCPv+xS^Q74^IFC1MI z+y)Oo$&2>Lf6-69t|PuOC)u~mRnJrMms8J6A2HlHw1@p+k`db3t>yR%bZb5I>p8v^ z#(L;4QTGZg@Up8r1V+KEQFrfekkXI#(97{H=wdze`#63Cp779TeZ}?p7>pj});~kN zi+bqV!`bj4T=FUzqjjcRuj2UEa3kC*c`8kOn~!t{o`(@|33bxHrV?5HMPEtKCxdBW zMwka`m-58D>L)Sk*Yn6cO`i`XkaMh<=H`$EySAe|2@L^aL>iyIt zUKciiO`xu?^A{6X2ET#(wUD)?|DNO9VJzGOEkCb3(*MC(ep-^m{SXJQz<-4Nkd%0c z(#IS6w~5b!bKpEfKO;bW@}#b@R4@Y!fl+W5tQ+L??V&H3TUUf^^B+u8`8hrGKcL?M zA4Xro@Gpn1Jgfwp!Isc^UXGPti@bj}!VgUX^p8Z}7Y>9n*B;FAO_`@E=x_c|BVW$j zVd|cNavtTpHD$i9z?H_llTt_WXM5-?psxfMqhDtD>povV*9NwS!|yY{yOF<<_-42r zo`g%1yAzZ&g<}+)45z_)utiFz8v$Q|r_!CcbtRP_ zjLy=(guWjvfWENdulw9dKbMU2Vd;M~^s)324==+%q2BfR0QK7UZ}RiePXSoO=+DYe zZRm3o_t#(R9w9&6@VE3841J2!Zi4yY85otu#bco_ty@0rUk&}X93 z{UmX{pOo~Q9%h1@v${Op?MQ9EQR$D9?+j z$v*4ChEVpo)!47U_2oGpdR~&BhJIx3y8pZ!F9VD+B z)McMz;k!^?Pm6e6wdM5`tzKVpUbX)-_;|g3hoBn{r4Q{VuXn8#!Nu zXD2BCzP{Xtq*|<)JaO;dLOxo7l5HK75i%fo5Kj$750L%?}5ht{p}|^_4Pa?|GLM1GLWAIT7Kdq z^U-;_zWDsF=2e)v`a6H!ubjsxn1ehYXcX(&@G6woTO6-D{rb}W;`=1|b)eVl{Uvn$ zp!DVCFZT_e^_xo>Dcxo>n-_l+pl(NOMl^5+amvpc6Pa2wnOgN^sIdCKpM zhyDQiL+~&3H=y3ntLUWfwCFOzEHD=gffZmSSOwOG^33e)*SQT>+js;xh37m>j~w#E`3<*{mOZhdU9UmxjEtF>GNdOdrDW8k-N8;pg!;a(UAPe6Gd&l%R2;UDlXcpaw5 z=jNUorh^$^E|>@AhXr71sGskn&+oVBX z{VkZ6F(0n@`Q&>cl_{1b*=*Ad^~hn;g)m;>s1R(_D7f1G$FSQS1CTfpa`u4m=f zG4$fohU4wvKsW>rhq|7X-_6kLIy2DCf*-)O_nE)Q$d`Rg;kd4=_a}XPi+&UQ$?&)I zyA8egoZ|Rdcnc=wK9CIR{;d4VMm_P#i(dg)*zlLQzAx2A-xxLtFu#(af7-QT^|TWa`Ob&sR} z72XIif4`v@-%}hv3w0fNpUkV?C-pr1^-GD5&d*HT-+Fb*IEVglFdPB3|JqO&8D7@$ z(I*_kq1Jzoeh1tQPr`HXS9l5j1#iK0k2>dIm=)%RCE#PQCaen^z$UOA>;Z?tDEKNI z2dBY#@C&#Ku7TgeUGOkG598tQ@DG@Rc?QD}SQr+AC15336Fv)Hgngi#pBl#bk@L6- z{T8?f>ht+J@i*ZVxC(v+H$a_l)p@GCR=TyNU-a1B^(U!&6(%j`bXj3OSQ?gvkHgBa z5o`jR!B(&r>;wD5q3|vEE}RIb!_VL{xEy{3x5Hg^LPeY*Nen2b3OeD z<#nL18ackcqC2t~%JcAISs#MGLCY_^g7Xo-yu9unFjc$lk_tD6rN zfO@|=uQ$4H4ZY-~lP zCQL&8U}*It`?KoFe(u#j8U3V&nP3)}14@0bdU~o8$e#u0z|o zKY@8qf-~Sx&`UoD-KU0rJ@IXD58P+ylT~#UQ^9PoG7N`R>*&thdij%%o8b3wJCx@U z{>56JBemrzm$x16grU`4{4rP=#=xqz-0_#-q}p!13c6L|cd47Mnu}$s>3Bd$-ZdjX zBY6d3ry6(nNk>28lgF4>5qv5@*_Zf=kL*+Sb-?JaK0du*UpNpZPQv}vJwUzo9Y+4k za3q`wKZFaQu5abPY3M&Ne3uaa9BzX<;U2gj9)`LffBjb&`MORVx-;+>m?X?qP7YH+ z7ft-`uivjmKe}#t>NGd%S^B38z4V=y;~8K{SO%7ZKJ{hg=Q8xIh_`|5;H%K5zASxD zLqC@Ic=#6l7%qZKpig~S`BM$O_ulq3lBh_`m*xZ8G7+K#POr>FL+b(xc~ap zmz953@|50(PY_H23&UctB-H6vekMa-fVjW@rO`Kl@?6KJteeADQ1>VCp&TCxN5ePa zyHMxLa~!?$CXzn|E`m$oDtN#CI$xdx>2E!)i=i)hpO^RbUx;6Xm!UlGF)8owx*vc2 zf5cDc|Do!nq>elfvN-EfP=D_5*DtftfBm}7p&jfDyF%?>>1h`k1c$&`Q0wavAE5NF zuznSehi}6fQ2RBg?;;Ce+lFo(1vkUsMmqD~^NKcbYtijC^r8zzKL;*=i{T2m3Vs8> zg|Sd~C2?yXTED)ri^RgcFb-;cXt;}1gVo_kcmSq*MrXRUrMr$kRTHNRe)jJ9)A0Wg zE`Xa<-TwFug2R-5gQj=)m*=5be$tg{UmwvT(Us-zowbt&ua$epfKN`-0dcUHN;`k^i`I9~Jb)M9dJY7$G zMJIj9JObUX2ikNf$5tKVq) zles;petj>xxdg$qFe7wF6Tden{vMRig{_!NFDUa$%yd8PAa5t!1NTF%|AY8n@CMA< zMk`qp4}xi7Mnk`V^Zg0@9Ik}A%5kpabMQC#psq_fuljW%_g(osA=jb5{oSNbt(SQE zR_;hp`p#EWvg1T1an8xj8!#=y-`UYB)wUF!XN?YB8TFTyS`3hMqXeK+)dJ@R*R z{0KY^FF{=|6ZfBoU=bJ!wLYA96DV_$`05Tzow&6em*-{{L08?-TRO`> ziaxf%GxQ^MC9j=PPx6+iP!Dw`d`}J@vWYB-_Iy?qv2Ti2Gm&}6aN%0hCdkkRm8u7 z>*05XKAL_f!0B)n)cu@dU%$%!dO1aMsP#pNmw;tq1w-G1_+S_fuSaTUx9+3#^?JMb zbFee~(9itFM*dvl=b=3RG6%jPunMdRr$MW}^e6s-=BLD8`l!RYAq%0`bG}7nl;? zVE71p432|Ved$m91I_wMOaAM!_jC-5@XCAZf3t&IG|#IM4X_!WYoumOA)E`(Nn=}-Iv&Ci0r^wENKYuFZ! zgnA0)iPuy*>09gNe9QAQpJ5()?p=uYgtE^V>R8vy*PPD{a2u3*Uite1)Cbd__P=V( zOP&K+g6q8u3^T5OiLc>!9LzS@O*S9QJ5*2FttGz(c~jtGsIw$~ul3~ljFLZ^{wBid z@W>Ed*{yZ{cq3n)M_2%#P}l&rg#F+!xC>hKr9bfxG`|Y|(#L$(i{KI{edsB?$nhRf z`ku#H>#gfyAaffE%MR>l;;}i zd0D!BiE@{^PArI$Lz&_}~qxED$vGwG-G%et~# ze}w*1m7jq;ulyg-WyU`rwCb(n_U^5UXB;XFBDq-B@lR&>2kO^X^>?rN1MwLXpr7nR=B4M8 zI0^TY%y|>@mO1*2`Mrs55_}&ngrC8sP}e)ac{&5n!;4VsrLW|iC+SP(s`bB8SM;Im zqlvMfboe%bm$?p7sks+3`dfu=J=_X+z};{k)cs1`1Jsu~a^AFl0J_0&7#s!1!1tie z|BSe<7iH{kozj;^R~}Y^RgL^X%yXTZtGteNe^Kbh!twB3I0;ULGvFMk{W22B4D-VZ zTFBbcm*jY97z!VQmY-K1>Hk3f*tuvD7itTK!+(VQ!0p#Jn{`fflCSj{IW9W!Z%@1< z>;iTEXyRkxTX44Mle!Ff&gN#;Cty^NJH85P|J*8niif@d`o?fR`h|x7=k&P(u7-Qz z0eA@N{;d3shJF`ufBoe-vX{yGQ+(Oqd6?eVpTGIn@zMK}^Y#MAyFfXQa^7}RKS?s@ zPyuSckt#pLL%$OJYFGw+S;JrVd6>SA!4vQ%)cyIJ|BI1di@xf@Mz8}6P3|g=guWDR zodf2Eh2efE&s|KC(&^Jc-T%|dU!K3{Z~idyUxvx3lhUZSnSG^3pA%g%Lw^d5-c}BmaG)&d0<*g^S@Aa0A>7w?f@tp#C~v*S$oYEAT4JoYrM#hq>Ur z`UM&Jvd>-A-3t%HIQR>^2&He`kL7oOJgwJtWS{xyw*V{(i^IyW8hi>qDE~(cKi!Aa zt&P4OjDX!>52*X|*Y6o4U+Q#Hb^5UG58r?j;3TN~@z-y-kuP=L=J-sw3a)|QKwaO; z|JcyWKBjX#E$>&EU{+WXmW7YQ%CH*L{rKyj$Hs$G;hF*M5 zbNn1kPJgMP%t_a?@~;^6WUi(0tqAp8n-hN?wuYTxFR1(R*T0UDFVC-+I(^Z-0i)r& za4MV)bw8Hha6>;y`F_Y+K6lA;?6(ltpSyH@%Wtt!zhtm;m=5K0vi19b=&PdtmCu!y zUUc&O?ZfCLPxmjno~j?o)B0ptT_hFE2J67KaD0f)bn8MKFA9sp$Dz*ZO#Bu28k}b6 zXA%Dl%JZMs8T#beT*ch*QeL-iozHPFf57}9tQF$o&7fN)eixrI_`Qao_=?X?{IcOw z#PE9*pNg;&d>TrhvGm&zU+G(XA2{3;gH8s_H~~9ar7UEZ^P^^6b0XbYjV4I1iqQ*yA1rth#$nKz4DpJ zdJ6mqhT*U4yh}Z)Bj-ielRDD3_%6as&GF+#;SVhpU(a=7IU4?&n_p(-?lbkK9Hd6^K`cPu{0r z2_s*8TXMV|l;^hVd5G=>bn@JJo!5=yz2Qjsp#1t6e!A`gbPM4kxDjrFKS14|zkaKX zeA)LHj*o-$pzcf8xAG?&^<*EXIDQdcGW4Q5&2f1Sy3R|U({+{xrh~d)fBmi*{YagZ zs!mCxo~6%Y=*1_LbK3-*Oy!f)XQsOyQ}vmCelmK*t2oq|PlCAXG%QI3~_O$`0-#Q%aJ z#hgz$SRF2htKoO>dw2^jDdF;SKH?Y#qf5E+Kbm@a|B~O0<1%;M|8S0vgqEN9$UJnOt}j0St@(8Kn6K_f z&dUP!c?MpB@_I^N(q+cLcqp$gIj`D(DL!7W--GCmKT#{GUOeY_9n!1?eK_zheS zH^S|3C)@+~L%kpE|0Ozq{k{Cw-lw0QgTMPcL*00I8Ty!ekTHL2Zl^3iH2^FB2>Cxj zfBj@1=gG775zKzF!aT4vd=#p7ed6~A{xTQsmn*>hbm*lYD__?MAy4-mM!W`;*Qwr* z>{ILOlV|BGar{ZB^J^RVI!|;~y<2?lOjFc#5&{dua!{TVyoB{yxDJN!{vn@3gNi%9 zoKW6J^Kic^1Vf>`kLu5TUY}F-{H*?4(r0Jb9oFUZvh*kQS8;#T{xXlsBaHB z(DD->nUBuX^~L9ZHLq8htH1Nt{mOZq${dcuf1cO;u*+x(<@FXuUH$sf{^C0m-}%t% z^}Z9`9w>cz`OE!6=O4ny-~0{ezk}Luzu~9zL}%5HV6O5!=nj)!uellyl&exEJQ`+6x@ z7Ru*dT`w8;XYs4+p|6O(GJFpG3sC!?RDE}+&)%>v91Nr2WH=4ZfFHq+;pcD#^s2Ax zjU(?(_zv{9{{7aQc%SvGIS#|u-??o-zZGtSJK!#;_bca3>dATe6#WvY&y!W}LHVzu z?i%QCz3^0 zi@dxlU+cBb%l~ISw`Y9RIhBUx;gj$wsO!mdm)|byj=TeRL9M?){35&rGndm!xBi{u ze?WQ8vCq(tdd%gmh3nvEcmUR|aQAv`A9wL#a0FZlb=E=Rhv6|8R9-9Hx(RikgB@UJ zI1mnoW8gBVT~`s`50AoA@I3q*TKQ`@z7Bq8=rdMy4iCWsum~&$%fQFs6R--b0qenr zFdVjktzj3~9rl9#;LC6%91X|ANpLEh0Y89?;1c*b{1R?}-@&c$2e=;|gh${pcn)5G z@$eG734Kqv`3Aw{Fc@Zm*%%L%$g{8od>*!i z3!qoN=(`wt?Jv5%hJGCJ_uzE+zM(%%{4|v3Qm3xsIuC(;q25m+;zi*j0qV=5uL`w) zGvaMvC)mZ%UnYJ9UWI?dTTq^FoTaL(m=o&y9f%KuW8hncemn7`PdcX*FfG*n^4#3I zPq`z7>N-}3S})JVeXp83G6jAN7ejeI?P}KR;CE1-gL{beF?b4Ifwy4BFjp@N%mMSl zQm_K71M9=auo>(Kd&8IEC^!bb3EzkF;Y#=w{06RrJK-Vt3%m#~!z(a3^UnzL!{V?i zl=Jhraem}HzKwnYoC$sE>sjJ0VOtmlN5MCtO7~fHffxPm2NH1|CHx=Zl#Z7 z@GQI`{Z)5)$zU3o4;F+);UlmbtO4u5hA;wlgk528I1;`F$H8bg4=#Y8!ewwH+yb}3 z-S9O01zv<#U@GR94hF-_uqZ46OT$oD1J;4{VM7=JJHoE8JB)&_LF;_3;`q{s28B6s!xMHuRqpUj@I085(G3x3=_K41bAl0Nf-*NLhUylzq7Cyeh=FJ5dO#D2|w#e|BoB}&%?*^ z*Yol6TZP|RxDkE_e}Gm!J>Of*GcEn6hrz~vQW5u8AB2B;m=$J+x}O(`N5DRCu%RDG z{0%q(PBiq5>Ax9_g#DrJCv~`sbb{SsB((H(IbI(&hD~5gsPm5y@B559G7!E7$3b~c z?>5$lVH}Kymj7ky+<=AAKMWr+^8L*#L;j=iF<2gY?N6SAto!dkeka&f`3+&M=Vkdx z{9flN&rx1O-EW}y#j@7t=L~VZzg|sTBxO^_v{0U7osV?{>I?pE5dlt5U;@5?>K0i^! z^?bJxU((DS`3WXzp_Ohe`V-`xhgqL>$8*3uP`mh>m!JFsP@c0L&szGuK)?N)>q>6@ z5XTF{hvEI^N8)p@_1lo&9(MNAPx`;t{JPY!^7Ry(GS}8H0(Lg$EpzEhol$Tc)czBQ zzYpg@IX_xog7Yuefn2ZpJXqInGx9sZt`Eqs6+SP*wovzP>2yEK$PaXX{`$|s|3mmO zl-H3pKbhCP=8vO}m9OV@i0kYuyZ|p7*WdT}?}dlqaj5-UwseuV;4~=DE#ATUPpI=> ze9lEGKkpa@`$4Uri2i%HANtUzgO3^c1zS1Co=~2bIphUh$*n~%&jt0;i|-KpGx@)U!Cuz7hjo&_*(O*(#B0_3XF)G%O3~i?;`R{q*ZQWXQ@?3JAH3!`SxEQX0 ztKc{ATet`M)YnzwR^LkQJJH@n;^7tO>!6ix9ZtLrYzIGpSK%Wa@1Aez9_r-u#b9Wc zyXUXQexnX{Z{nd>a(y9n-t@?6pVtYz-j`CX#&z4XbrPJ&@J zX!%PY2dQ%u{tVATxgPcYYvNxQHiXS#Ul`xbW!`|dVA}33o(YEbbn8eM1%rFJ_{{;1 zTLwD*HrVkxbgRVgf%f;H>rZC?S7Er?-)6PH#LnGMk{8?6tpnYE9_CyK7Kfp*9IOn( zU@NHm=|Wu2r_4+CCFe`_rSm1v>pJ+6>*34;x-KdY(Ve(;GW_K_EXle&42RFc=U^}R zDjWlKmEnd??{8iYm$?MSL%Hs!as7Yh=j*`QukL3y{cSS(mwioCy1j;8;(CAC*iT+q z92VjAJPBUrbzN|fo}^pL^>Cf~f5Y_n=`7Lzg6>x+`N_$%@^zlnlRRBdd_^a|vj0H$ zD|vcu&FT9^*b#Pzec>1VU58Rv?wc}a-M^fdD&$v#ji9A3!|}3E@@sqK>pZC^dAgqX zicb2Hc?7y&$@7|{ygudiIFQ%j5cm$);vb4d7Nb~+P^Y7Jx|HkdaaYW%l$D9zp&me^ISj2WV{ZF!9h^_ z%l%b+n&9(1jPTP}u0y%6$#wAv_ci^w#kybXepW`hj@Q5vFS&IYSP52#wcyjR3G4zx z`ndc!V^03|U)*?qt&4ws*c7&adj2zs$3dCbQS~`bo|CEbAL{EOd0_}F1hrnC?_7)H zbzuiXA5Od}Yz5mG`q^CfbKxgYo^PxBv97oO)_IrLfn2Y09aQoS3%wtS59jzu z_?n@Y_#}=`gEI~NTH@cp_3&X{A99_~=6aX?j^*|5wcjoH?SzNmWvKUS>5riQ*(1L^ zeTTuPVKb=f^&ma~M#1rhejf3Upv*Yr*&3++2_5! zZ|V8up)bqN(&s0y04!$g&(ir@Zv^j)Z^7A6o}aAyljkX~M7J9L1hu}_P#1X`Hi0dm z*2{Bur*eD=95O;H-CB-c6u)6E-Up6@6JSn!%6a%5Q+~zB*Ih`wF~?iMu71|jefKAC zG@J}G4tJGvK)nxZKH7g1`McqMcmc-4EATIP1EyncQeV$wsOo<%>%;IJd}c#A4^QG- z8-^R_NBaeupHJ#SsbkG=GJO<%+3j=?Tm>sex%gN(=@qx$3*9R5d(?gA*P~7Y*a}Mh z_o+MAs4x9#oz?HN^cxXi-$_R$?)hJTlDYkjUoidaImQt`3D3Y=FctfoGtzhGYZG*< z#P3q?XQN)C()}cJTo>im+u$MSRZs8h0(sY9diov@W$xP7nxFPh&b~9l9I!Ba7(N2! z`OM{5UxQY?c-4O{`j+QRw<5nAl=HBY^^fo<^rz63gXj`3#JI!ogI`b+)NuR5nN7y~!J&F~mJ1Le879gY6Q zKcDDf7pVJ}_#BQef?q+edU_w*$U6uxz)quG<(HuLwa$n3zeRpJ_MZvng9TtwSOS)U zWuVm8^N3ab|HZm2b&tXeP|ianzHh_#p!U~(f#wgU?psjmjxp-(As#%&by5#DgCpQL zxD0*?ePdl-Iv5R?L%qLyt#gEWIgL8I#_39KE%mafzGJ0-l_olW=b+Mk;(M?C%Kn+?<}}@ym(;Ig)Yto!^L!Zn8TbeEs+XF+vcqC<7F-Cm zk9EG&p%=f$SoeZi*>@o|hmTn=g=?VRulAEU`kTLuKE8ue$J$Rd`e*>3fvsU%*b#Px zJzy^=^V9o@=Xm6JcVrM83SWWI(8|xp@rtmmp|^a0LVpUL4={fr`nB)~^zxH^N&fl( z^C!RICK3-1yyezq-gev#k3c;?(P#2gpZiVcFa^e_Jndf`T~9yt7trNrAMyC-j@Hg@ z9f?l%A^VoTWZ&YWT`J?>7<%c0-f<4eVK%rBYJWN2hU1+)^z}LZ3~T{)eh=b(;b5ri zTRN-Xv-m_$a89#fER2Wg-*d+^!7MNYPKCOk;CEf5BrFa0K=Hps9i1=w4(KD{RQMs> z26g@=;v@0b`;)r&x<4;}uleYH{=e22<2;szRbd#c2^+)aunp`9`@-Qc3hMK7uRhlG zHlI2l!zFMz{0erSAv@pcMARqdrx(h2SV*@&0qT$W#8ps1=tXV!)EX~_yX(!rM{lWA=Q5n{Z@g$K_5PH z9%i!s2rhxzU;71`KZ5!bpwzMEcZ)v8&2XJW!;tsgx-cvXL*X>2`+sA)i@Xg#hr{vJ z`u!YF&+%NaI&1_7K%E~;d=ESXrGKq2Gt))N!e%hvEUk3wK^*@Ku7)SzMVQ0Lk0jm? z4uvs>e%WkScR4%>d(Uz4K``3~Zmqxfp zfTdw&sPjuw=TTS%hC!{b$i8F_(&s?uI|YI4I{?&hOvEgE&9+U=#Q(MD`Ej_~|6H z{AwBbR-I*rUg9e_z7`%g^h3FBp5{7f3a7&*a3zf6I=BkcC3hPe0^^}CrHjYIuvBhc z7dD1V;7S++gVQ>FvGk6w!Z;X`!NuFcC>R4b!&tZ*9)Rbeb4mOz^Vq~(0^Psf-!Rp` z?kBOb`$_7rqW&5f^Iz&m*PlY2_u+?dKKv9~{ptN{|E63Y?O`|AQ?478&^AE5oVP#d z^9J;B{mxE90d;+abDmUr-&XQ}hG*dg@l^@O1Jql-(tn`+N}k?#ZuTF`>nm$&*LW^i z7#4%2pvrHeIo17C&b#&(->Uf4hWh=WFYy6T`tb650eu_T0cyYD#P8Lw8~MEq z|G|d8&J&&0PaWpm0=9x}V0+jF9)$9KsQ2e@{xN(`z>DIm488o6+-LRO4gV-8`;hBO z-p_s9N9Fxm`&;*wa@=34!G^FkTnJz0el7PwulIjlf3Wh?pBJ>giRxn+>o4IN_zheS z55nW{6#NrjgI@JkptI`z!14W1*LzU@JMiBXU_IUUQKL^imt>shbTBi_4fDc6uqf2~ z)%E2(T%eDC&I9>g_3qXG2K55fG9`?`gUGfJ3YM`>8jDG%}xs%VO^1h(#eBze`zvM9W1M>fg`Tq=m zh4JtTwCd}A*YW$#PIv$wf?B^lK)uXW{`{vFbEpSn(zyz9zLzs!nX|5+g}gGbEG!ST zzGQ%U%UAjjv|q{7`)*HPA$*?c0K39|a1b00<tDjh>RYY@dB0o5p9{*Lb7(*7J}`#scp8j@bGe_&eJTWh{dFbB*Hb^Y^(-yg)2k!Se_>X$M=zl`X;_EiC0WmpwH1#7}* z;ImNo?^Q1!c?DoeSPpvCzu$VH_gPQR$(la>CFU?=+9j}GAIC28Q`cVoI`)b5mG0HMU&qo< zRj*g7UqokCoqP2!-L8Mv9vw<`?cghYd(|Z(;h&-_-C=OA1V6`qecX`&9sBg{+Pl}k zE!rd;>(jAE#O(xritN$PSGsGjt_k1!bsUuNy;H&>;aKku5&a^3r8{D~#33ExWJsp1^l_3i5`-M)9vo*jE7 z9PH4sUH{Gr$q~IecT9-8W!GMvddtrd?b`L}I6ziCy7uZQKPSvfe(cdZp_t3X=|2Yk zpMfA>5?{rnVgGR;$d}aj>|NtQzJ7O&C-Y6YYdpCx=C1J+zB9?feEKUW;X~tsTddUzVN%o)BBdBzMDRS@An-weflfNm+&Hpy<0rk zH#=J`El>DQW0`#658o}G*%v&s<6ZPwd^ztL&*}@gYdo8;*j?kvDae=SUsj1f2Kn+PF8<}GAYZ7v?~AMzcIQyrJKueG9B#*B ziO(TkI)S(2QMJSFM7~Yt4%8+dUf;##uQNVFT&_2nNKfMN=(FB#FyZU8UXz!2Tvrzl zM_+|_d{Y-+#eW}pLG*9A_$Ks2i7$EH#pP{i67d+~{PBTrG4WUx|4#gg*GO33{@P1C z^o6iHYx#R9=ZHtQcJZ0)D@DS8XT3AG2p4b2^-zL%R7V%@e#gz{dxm&C@d^0!A|9>g zGg17zIiG|0EF>P>-Nm=0bO&}3kL%;&GM_8N^} zj~VRrPrI_d5uzXB;^ooLB_1B-;Efjy^TqeHniN_JYknk39`|F(OkB8k}`ot$$!uh=2 zf86n~+hO@*AYWeMOMZ26`TH#uiAP^@@t3pS&hq&hC&cgEf6mstv#xvRyU+I`aUb`S zO2m5-4c%D97jA#uBK`xPD9YqUq#|E#Gk~c z9&z6nPQQ%np(XJuX~OO(`_NB!@mb;YCGi1;V|2Wp2R04eb^n}|JT{Kz{gcp zd!N3blmaQxPe8yal!pR!@@Uc)fzv!nOVc(sfff}f(`1sS&11++N{e_6kN!SD@QVty zK+%YZ<)H@kf^hYA072x!B|O9;mur-lf|Q_q2#Rvo00m^iScO{`r`WbQo>7( z>ucciJoq3yc>J6K%lRho|Am3^eSF6EJy6FBwR!gepAGy>(DVIvrvD7^*}$Izem-#X zi$4P20elYV|4z8rE$V+#uN;R{{9ydV7|!=w8BfOW<-ixjaK1mu^gi&J;NK6t6?g*p zR^Wp%d^_;r82>A%9rG^$9|9kX!>v!Kzbpm*UGUind@Asu-UDv+p2g?kCza0z(C-4D znNKNhd@Rl@4$z!>KUz zqk%Ucs(2IlF9QB8;9tSSuOWPI_Wa0E%E#=v2Ymh;cp3PcJ?{Z-_RROgc)afcw|HXq zTy?bakK6My;3t8<*>fIv+@1%apSV3=9>dL^hhn(d^Y$2S_B;$cZqEha3CL;oJQl;v zp1r5leoRiY=LGP$Jr{t-?RokyqjqTaoB$rTXYU!M-vBwyp1Xm^?RgM*+@60Dqc?lL z7V{Ff=K|>C_PjgBXDjSv7x-+3oZlq8lzmzp8v`HHr^T_~$8ekX@fdFN{!R=xzw>?- z)$@-b=f;zKk?;)2`5^GyfHwkv0{Gj&dw@R=yzOLPkTt+x1%5B^g~0y;{LjD-LcX3s z2Fd<4IjMZMgZ>!8OX;nGaE_m?&*``dpg(m2dfs1W{;lQ8e_*mU;C*(+$0`-Cg*+Dk zPu3{@KH}0g!nxnXe?|4VgYZ&v-ZlY#kMX~2u?X@s{bB zW6Fou)pF2+k;gUqTg7*S|IxrlUQ~SfR3)z>ockSoL;FoA?9~Hrjo~feM#(XaXM z4d7Guy3$+xxBTvdPXc@_9wuTqKMx^x_-Exa75cy1{zsL^^n5AdrToIyM}r%b&)t)?<7>d*?@@dh`n2`YSg+#tJoCSS zzNugF5zzk*xR+PFVwUnRJ4ogCw<>-Fa9gL0y{qyMg3swh&w4HZH#=ELc&Txn3;x;3 zb<*Ju=(87i*%ZZX{dfuRCg7%r?MAQaW3_q7} zwyW`V%D*sK8(c(qDLwyh(2rfN^rq(>zz45T-1IyGya3$R%MTczE0x~#|19v~s}=8> zqVl|9^xst6^nWPX5$ho_J*xi?6VBuE?^pVZV)}0dz3Kns;Nt-|{htARaHjG%eOjHL zn4!4UMYfI@++T64yR6P1+fQ+;i_V6ez--XG`VFyP)HN^f>D4|rj=;$|oHz>|k7z5)qp0Qlf>ihmLKEx^ZW z6@LNt|L=s0f0e0z%={ph($|6I(jh7VDCv#V#p#|IxwuV?vj7`WNnA3#3{J_*qR z?WdVMG2HCXo2C86?XdM=#p8BZK)+^(vky`Ik8II#?N9yjVRl#mAF~s)!$I_Gc6B=O zVZ9X&(tgbjmjfR=P;sMQ3)}~8{4>B4z!M^k=M4b&V)$0VOWE7D32@t2OzhI}njKEG z@AMt%cb1pEPUVL<-oFZ7ydTBaOUZK|(R<#(MfW$K6h6iGF`ol{{66U(p{Gsi*nP>V zhcb6AUeq7W-evv^{WPhO+_&H=s)_}CMQ(>AgkPXZtPyW&SqQTz|U{ZB3y!06ut zo>=4yZhDwS8wH}DKdJmPl#O=`@Sz)3{?$|U!fC)0oyyX9LiF>X^0^Oujs`xy&==$g;2#A(`Vpl!J_~@i{#^BTEa=YyKDJ!@{RZ%j z!2N0+*K1Ss!dBz|mf~kYZ`*;FjVQhuFBP9q&g#Utu`(K&Nti0(jHTDF2nfTYHuN_aM&( z;6vbZ4ETJ`_-xbhK7t9j4*2Mkirc*02D}V*V)#9TC&1JF_@(iIotRy{2E3_4gNIQc}Z}b zI3IrjeF5@Y96RK2=O*1AMu6o0L<4+3PfyI!{60;d`#JR9|rC>C_OEsg0S5@J`^~^D6&akbl7Ve@FR$4fIz5AFEdS-vYk@_z>*M^3->MzjB~2ln;ac z7r=*O{`lX7mwAVIKa14^FM)pKB9-6nlky`KMeSAm*PiG(hF6<%P{ZL zLEiw}>sI=cfS+UZ<*En1ug^z^;Vr6%4CZkVc=APG==$WZ=Y2_VoH!phfqwXh+V9=C zb}w*mgYx$=FaH6&>=j=qzp}43c-Htp|7U{!Eu&wf^3MhT1CCU^jnye`<2@eu$mg}+ zE8)OZz`Yxl{;Bnu(l2HZ<1 zeG|rYGVsJZsvonrM#JlUk%r7$j?V%w{7U&)os$PXggW2i&!xae`<4FtQ_l+pNn~UN^l6_e7pks#0jb&`yKp0jL$+}s7?Q~kA{ApQhCk=uakiL@UNpF zPp#oMD*vNsf1i)Fz`d0!=P3$%TY#6fD*ewf-md^JK%XYhw}B_WrF`tYB=-X!M4fy9 z_&;Oxh}(03zY07ts^i)U{C(g)`lW3FIgX^liS1+zb&J)9mB2?b%E#=d4tVlz#jQ?T z3%uz>wI7qS3%HNE#rSMCdi2`^dAruci4 zwBsS*t#A4QoTIS!ZQu#)Pu>aqZo^+t`V{b=10VmK+U>2tUjaUd`r$a>(~eO+B>q$N ze>?EQftTH?_*ql*LJjcj7nJ{`Ns6Bf+?%TO{{cQ*fY;6S1+j5mWBeB>ZuW3HaQ{<_ zh5ixf?IGZa#}%)TtDg63;7zA0pApFUN8p37L$h=5SRHQx_HXNpS-=y^mH&Q|wS!{> zM`zB*V$hF$M#uX-T>CWerX96{+yuNIc%eb@1sK=Wz=vP-g_6nxIlc~j{F};u5%gRD z?jKkubSM)`I?sxzNu!DyMSuypw@9y{-B= ze~L0%0zCVk;*G#lgmc{(f9|~-^kc|JPl5h&;04(KV~}&Z@rQqH0zLxVhd-MA>;gXI zt2|fZ!fxQ>sBg_4UIU)MKCSuJ-+>PvtNOVS`ahT)hx16;ef1&u%>B@gAc&!HBx>@-w zpQabufe#KU{mr1?EI3Y_k1GjJIOq2DI`A2*_XRS)z8!cO^n4Q<+z&jN)qZb)o?ijp zgnc}FuHYZQM>>_>?DmKgR6k><=(wJNKC1|yHTfWK`l#yPhb0h&VKZ_gM6XU*?{3j{?3!_I~Grc_zd~l80&yUdW zyT<1y%I7ZN`<|rwNj|LfR{!(!=j?aGShw0f*oQ%1s8IUJ;BzwYG5Ga9kpB$hAFJQa z0p5!Bi`hvAc+-9==gr`=#c=4&))AKi_pmOu`utkL%jh}k-Ld_s>p|a&b)xD24&a3s zed$qq_q_X!9`~&-fjqwk{>n_H2kiaU=%KecxbP?7BiP5wPtpt1PF6ii<^oX#_q9f4AQkcLHz1I>qMYbH)ejHQRUH4!nT4YH{gS;A6LGzent^ z^6xS}_b6`l@Y6^3Oa)^^kbU7sj2+-aA@woH!qqpl_}B1*O|b zax@tp(`OIxF|0ezo1*lW10RaTpBsVupHcl^3;zF3INMcxpYsvWXR%IP1Nz?p|JyOX zAk`Qz|9(XF86fAsn4g0F`lQmE{r>~_ShdPy`kXya`Ltr6;V|OM$0@*@;CFe2z0-k@ ztf>{`VH9L7hJRZ5Ukv&##x9**k4)?`a{X_S)a+v7YkzcFc

    dXgbYJ&#gLzgIj^Xv0~mtmb^ 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 K+TQ-g`~M$a`H%Vl literal 0 HcmV?d00001 diff --git a/examples/seekable_compression.c b/contrib/seekable_format/examples/seekable_compression.c similarity index 99% rename from examples/seekable_compression.c rename to contrib/seekable_format/examples/seekable_compression.c index f4bceb10c..8d3fe8576 100644 --- a/examples/seekable_compression.c +++ b/contrib/seekable_format/examples/seekable_compression.c @@ -12,6 +12,8 @@ #define ZSTD_STATIC_LINKING_ONLY #include // presumes zstd library is installed +#include "zstd_seekable.h" + static void* malloc_orDie(size_t size) { void* const buff = malloc(size); diff --git a/examples/seekable_decompression.c b/contrib/seekable_format/examples/seekable_decompression.c similarity index 99% rename from examples/seekable_decompression.c rename to contrib/seekable_format/examples/seekable_decompression.c index 641e0429f..b134b87b6 100644 --- a/examples/seekable_decompression.c +++ b/contrib/seekable_format/examples/seekable_decompression.c @@ -15,6 +15,8 @@ #include // presumes zstd library is installed #include +#include "zstd_seekable.h" + static void* malloc_orDie(size_t size) { diff --git a/contrib/seekable_format/zstd_seekable.h b/contrib/seekable_format/zstd_seekable.h new file mode 100644 index 000000000..959b1bd0f --- /dev/null +++ b/contrib/seekable_format/zstd_seekable.h @@ -0,0 +1,140 @@ +#ifndef SEEKABLE_H +#define SEEKABLE_H + +#if defined (__cplusplus) +extern "C" { +#endif + +static const unsigned ZSTD_seekTableFooterSize = 9; + +#define ZSTD_SEEKABLE_MAGICNUMBER 0x8F92EAB1 + +#define ZSTD_SEEKABLE_MAXCHUNKS 0x8000000U + +/* 0xFE03F607 is the largest number x such that ZSTD_compressBound(x) fits in a 32-bit integer */ +#define ZSTD_SEEKABLE_MAX_CHUNK_DECOMPRESSED_SIZE 0xFE03F607 + +/*-**************************************************************************** +* Seekable Format +* +* The seekable format splits the compressed data into a series of "chunks", +* each compressed individually so that decompression of a section in the +* middle of an archive only requires zstd to decompress at most a chunk's +* worth of extra data, instead of the entire archive. +******************************************************************************/ + +typedef struct ZSTD_seekable_CStream_s ZSTD_seekable_CStream; +typedef struct ZSTD_seekable_DStream_s ZSTD_seekable_DStream; + +/*-**************************************************************************** +* Seekable compression - HowTo +* A ZSTD_seekable_CStream object is required to tracking streaming operation. +* Use ZSTD_seekable_createCStream() and ZSTD_seekable_freeCStream() to create/ +* release resources. +* +* Streaming objects are reusable to avoid allocation and deallocation, +* to start a new compression operation call ZSTD_seekable_initCStream() on the +* compressor. +* +* Data streamed to the seekable compressor will automatically be split into +* chunks of size `maxChunkSize` (provided in ZSTD_seekable_initCStream()), +* or if none is provided, will be cut off whenver ZSTD_endChunk() is called +* or when the default maximum chunk size is reached (approximately 4GB). +* +* Use ZSTD_seekable_initCStream() to initialize a ZSTD_seekable_CStream object +* for a new compression operation. +* `maxChunkSize` indicates the size at which to automatically start a new +* seekable frame. `maxChunkSize == 0` implies the default maximum size. +* `checksumFlag` indicates whether or not the seek table should include chunk +* checksums on the uncompressed data for verification. +* @return : a size hint for input to provide for compression, or an error code +* checkable with ZSTD_isError() +* +* Use ZSTD_seekable_compressStream() repetitively to consume input stream. +* The function will automatically update both `pos` fields. +* Note that it may not consume the entire input, in which case `pos < size`, +* and it's up to the caller to present again remaining data. +* @return : a size hint, preferred nb of bytes to use as input for next +* function call or an error code, which can be tested using +* ZSTD_isError(). +* Note 1 : it's just a hint, to help latency a little, any other +* value will work fine. +* Note 2 : size hint is guaranteed to be <= ZSTD_CStreamInSize() +* +* At any time, call ZSTD_seekable_endChunk() to end the current chunk and +* start a new one. +* +* ZSTD_endStream() will end the current chunk, and then write the seek table +* so that decompressors can efficiently find compressed chunks. +* ZSTD_endStream() may return a number > 0 if it was unable to flush all the +* necessary data to `output`. In this case, it should be called again until +* all remaining data is flushed out and 0 is returned. +******************************************************************************/ + +/*===== Seekable compressor management =====*/ +ZSTDLIB_API ZSTD_seekable_CStream* ZSTD_seekable_createCStream(void); +ZSTDLIB_API size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs); + +/*===== Seekable compression functions =====*/ +ZSTDLIB_API size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, int checksumFlag, unsigned maxChunkSize); +ZSTDLIB_API size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input); +ZSTDLIB_API size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); +ZSTDLIB_API size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); + +/*-**************************************************************************** +* Seekable decompression - HowTo +* A ZSTD_seekable_DStream object is required to tracking streaming operation. +* Use ZSTD_seekable_createDStream() and ZSTD_seekable_freeDStream() to create/ +* release resources. +* +* Streaming objects are reusable to avoid allocation and deallocation, +* to start a new compression operation call ZSTD_seekable_initDStream() on the +* compressor. +* +* Use ZSTD_seekable_loadSeekTable() to load the seek table from a file. +* `src` should point to a block of data read from the end of the file, +* i.e. `src + srcSize` should always be the end of the file. +* @return : 0 if the table was loaded successfully, or if `srcSize` was too +* small, a size hint for how much data to provide. +* An error code may also be returned, checkable with ZSTD_isError() +* +* Use ZSTD_initDStream to prepare for a new decompression operation using the +* seektable loaded with ZSTD_seekable_loadSeekTable(). +* Data in the range [rangeStart, rangeEnd) will be decompressed. +* +* Call ZSTD_seekable_decompressStream() repetitively to consume input stream. +* @return : There are a number of possible return codes for this function +* - 0, the decompression operation has completed. +* - An error code checkable with ZSTD_isError +* + If this error code is ZSTD_error_needSeek, the user should seek +* to the file position provided by ZSTD_seekable_getSeekOffset() +* and indicate this to the stream with +* ZSTD_seekable_updateOffset(), before resuming decompression +* + Otherwise, this is a regular decompression error and the input +* file is likely corrupted or the API was incorrectly used. +* - A size hint, the preferred nb of bytes to provide as input to the +* next function call to improve latency. +* +* ZSTD_seekable_getSeekOffset() and ZSTD_seekable_updateOffset() are helper +* functions to indicate where the user should seek their file stream to, when +* a different position is required to continue decompression. +* Note that ZSTD_seekable_updateOffset will error if given an offset other +* than the one requested from ZSTD_seekable_getSeekOffset(). +******************************************************************************/ + +/*===== Seekable decompressor management =====*/ +ZSTDLIB_API ZSTD_seekable_DStream* ZSTD_seekable_createDStream(void); +ZSTDLIB_API size_t ZSTD_seekable_freeDStream(ZSTD_seekable_DStream* zds); + +/*===== Seekable decompression functions =====*/ +ZSTDLIB_API size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, size_t srcSize); +ZSTDLIB_API size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, unsigned long long rangeStart, unsigned long long rangeEnd); +ZSTDLIB_API size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input); +ZSTDLIB_API unsigned long long ZSTD_seekable_getSeekOffset(ZSTD_seekable_DStream* zds); +ZSTDLIB_API size_t ZSTD_seekable_updateOffset(ZSTD_seekable_DStream* zds, unsigned long long offset); + +#if defined (__cplusplus) +} +#endif + +#endif diff --git a/doc/zstd_seekable_compression_format.md b/contrib/seekable_format/zstd_seekable_compression_format.md similarity index 100% rename from doc/zstd_seekable_compression_format.md rename to contrib/seekable_format/zstd_seekable_compression_format.md diff --git a/lib/compress/zstdseek_compress.c b/contrib/seekable_format/zstdseek_compress.c similarity index 99% rename from lib/compress/zstdseek_compress.c rename to contrib/seekable_format/zstdseek_compress.c index f9e108afc..2504287cd 100644 --- a/lib/compress/zstdseek_compress.c +++ b/contrib/seekable_format/zstdseek_compress.c @@ -10,10 +10,11 @@ #include /* malloc, free */ #define XXH_STATIC_LINKING_ONLY +#define XXH_NAMESPACE ZSTD_ #include "xxhash.h" #include "zstd_internal.h" /* includes zstd.h */ -#include "seekable.h" +#include "zstd_seekable.h" typedef struct { U32 cSize; diff --git a/lib/decompress/zstdseek_decompress.c b/contrib/seekable_format/zstdseek_decompress.c similarity index 99% rename from lib/decompress/zstdseek_decompress.c rename to contrib/seekable_format/zstdseek_decompress.c index 92901befa..3cb851ea5 100644 --- a/lib/decompress/zstdseek_decompress.c +++ b/contrib/seekable_format/zstdseek_decompress.c @@ -10,10 +10,11 @@ #include /* malloc, free */ #define XXH_STATIC_LINKING_ONLY +#define XXH_NAMESPACE ZSTD_ #include "xxhash.h" #include "zstd_internal.h" /* includes zstd.h */ -#include "seekable.h" +#include "zstd_seekable.h" typedef struct { U64 cOffset; diff --git a/doc/README.md b/doc/README.md index 0aee04280..6f761b33e 100644 --- a/doc/README.md +++ b/doc/README.md @@ -17,6 +17,3 @@ __`zstd_manual.html`__ : Documentation on the functions found in `zstd.h`. See [http://zstd.net/zstd_manual.html](http://zstd.net/zstd_manual.html) for the manual released with the latest official `zstd` release. -__`zstd_seekable_compression_format.md`__ : This document defines the Zstandard -format for seekable compression. - diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index c7017d12c..a5b66f7d3 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -28,9 +28,6 @@
  1. Buffer-less streaming decompression (synchronous mode)
  2. Block functions
  3. -
  4. Seekable Format
  5. -
  6. Seekable compression - HowTo
  7. -
  8. Seekable decompression - HowTo
  9. Buffer-less streaming compression (synchronous mode)

Introduction

@@ -671,111 +668,5 @@ size_t ZSTD_compressBlock  (ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, cons
 size_t ZSTD_decompressBlock(ZSTD_DCtx* dctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
 size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize);  /**< insert block into `dctx` history. Useful for uncompressed blocks */
 

-

Seekable Format

-  The seekable format splits the compressed data into a series of "chunks",
-  each compressed individually so that decompression of a section in the
-  middle of an archive only requires zstd to decompress at most a chunk's
-  worth of extra data, instead of the entire archive.
-
- -

Seekable compression - HowTo

  A ZSTD_seekable_CStream object is required to tracking streaming operation.
-  Use ZSTD_seekable_createCStream() and ZSTD_seekable_freeCStream() to create/
-  release resources.
-
-  Streaming objects are reusable to avoid allocation and deallocation,
-  to start a new compression operation call ZSTD_seekable_initCStream() on the
-  compressor.
-
-  Data streamed to the seekable compressor will automatically be split into
-  chunks of size `maxChunkSize` (provided in ZSTD_seekable_initCStream()),
-  or if none is provided, will be cut off whenver ZSTD_endChunk() is called
-  or when the default maximum chunk size is reached (approximately 4GB).
-
-  Use ZSTD_seekable_initCStream() to initialize a ZSTD_seekable_CStream object
-  for a new compression operation.
-  `maxChunkSize` indicates the size at which to automatically start a new
-  seekable frame.  `maxChunkSize == 0` implies the default maximum size.
-  `checksumFlag` indicates whether or not the seek table should include chunk
-  checksums on the uncompressed data for verification.
-  @return : a size hint for input to provide for compression, or an error code
-            checkable with ZSTD_isError()
-
-  Use ZSTD_seekable_compressStream() repetitively to consume input stream.
-  The function will automatically update both `pos` fields.
-  Note that it may not consume the entire input, in which case `pos < size`,
-  and it's up to the caller to present again remaining data.
-  @return : a size hint, preferred nb of bytes to use as input for next
-            function call or an error code, which can be tested using
-            ZSTD_isError().
-            Note 1 : it's just a hint, to help latency a little, any other
-                     value will work fine.
-            Note 2 : size hint is guaranteed to be <= ZSTD_CStreamInSize()
-
-  At any time, call ZSTD_seekable_endChunk() to end the current chunk and
-  start a new one.
-
-  ZSTD_endStream() will end the current chunk, and then write the seek table
-  so that decompressors can efficiently find compressed chunks.
-  ZSTD_endStream() may return a number > 0 if it was unable to flush all the
-  necessary data to `output`.  In this case, it should be called again until
-  all remaining data is flushed out and 0 is returned.
-
- -

Seekable compressor management

ZSTD_seekable_CStream* ZSTD_seekable_createCStream(void);
-size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs);
-

-

Seekable compression functions

size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, int checksumFlag, unsigned maxChunkSize);
-size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input);
-size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output);
-size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output);
-

-

Seekable decompression - HowTo

  A ZSTD_seekable_DStream object is required to tracking streaming operation.
-  Use ZSTD_seekable_createDStream() and ZSTD_seekable_freeDStream() to create/
-  release resources.
-
-  Streaming objects are reusable to avoid allocation and deallocation,
-  to start a new compression operation call ZSTD_seekable_initDStream() on the
-  compressor.
-
-  Use ZSTD_seekable_loadSeekTable() to load the seek table from a file.
-  `src` should point to a block of data read from the end of the file,
-  i.e. `src + srcSize` should always be the end of the file.
-  @return : 0 if the table was loaded successfully, or if `srcSize` was too
-            small, a size hint for how much data to provide.
-            An error code may also be returned, checkable with ZSTD_isError()
-
-  Use ZSTD_initDStream to prepare for a new decompression operation using the
-  seektable loaded with ZSTD_seekable_loadSeekTable().
-  Data in the range [rangeStart, rangeEnd) will be decompressed.
-
-  Call ZSTD_seekable_decompressStream() repetitively to consume input stream.
-  @return : There are a number of possible return codes for this function
-           - 0, the decompression operation has completed.
-           - An error code checkable with ZSTD_isError
-             + If this error code is ZSTD_error_needSeek, the user should seek
-               to the file position provided by ZSTD_seekable_getSeekOffset()
-               and indicate this to the stream with
-               ZSTD_seekable_updateOffset(), before resuming decompression
-             + Otherwise, this is a regular decompression error and the input
-               file is likely corrupted or the API was incorrectly used.
-           - A size hint, the preferred nb of bytes to provide as input to the
-             next function call to improve latency.
-
-  ZSTD_seekable_getSeekOffset() and ZSTD_seekable_updateOffset() are helper
-  functions to indicate where the user should seek their file stream to, when
-  a different position is required to continue decompression.
-  Note that ZSTD_seekable_updateOffset will error if given an offset other
-  than the one requested from ZSTD_seekable_getSeekOffset().
-
- -

Seekable decompressor management

ZSTD_seekable_DStream* ZSTD_seekable_createDStream(void);
-size_t ZSTD_seekable_freeDStream(ZSTD_seekable_DStream* zds);
-

-

Seekable decompression functions

size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, size_t srcSize);
-size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, unsigned long long rangeStart, unsigned long long rangeEnd);
-size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input);
-unsigned long long ZSTD_seekable_getSeekOffset(ZSTD_seekable_DStream* zds);
-size_t ZSTD_seekable_updateOffset(ZSTD_seekable_DStream* zds, unsigned long long offset);
-

diff --git a/examples/.gitignore b/examples/.gitignore index c6b2a9f42..0711813d3 100644 --- a/examples/.gitignore +++ b/examples/.gitignore @@ -6,8 +6,6 @@ dictionary_decompression streaming_compression streaming_decompression multiple_streaming_compression -seekable_compression -seekable_decompression #test artefact tmp* diff --git a/examples/Makefile b/examples/Makefile index 2f649eaef..b84983f08 100644 --- a/examples/Makefile +++ b/examples/Makefile @@ -18,8 +18,7 @@ default: all all: simple_compression simple_decompression \ dictionary_compression dictionary_decompression \ streaming_compression streaming_decompression \ - multiple_streaming_compression \ - seekable_compression seekable_decompression + multiple_streaming_compression simple_compression : simple_compression.c $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ @@ -42,12 +41,6 @@ multiple_streaming_compression : multiple_streaming_compression.c streaming_decompression : streaming_decompression.c $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ -seekable_compression : seekable_compression.c - $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ - -seekable_decompression : seekable_decompression.c - $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ - clean: @rm -f core *.o tmp* result* *.zst \ simple_compression simple_decompression \ diff --git a/lib/common/seekable.h b/lib/common/seekable.h deleted file mode 100644 index ab11a4388..000000000 --- a/lib/common/seekable.h +++ /dev/null @@ -1,23 +0,0 @@ -#ifndef SEEKABLE_H -#define SEEKABLE_H - -#if defined (__cplusplus) -extern "C" { -#endif - -#include "zstd_internal.h" - -static const unsigned ZSTD_seekTableFooterSize = 9; - -#define ZSTD_SEEKABLE_MAGICNUMBER 0x8F92EAB1 - -#define ZSTD_SEEKABLE_MAXCHUNKS 0x8000000U - -/* 0xFE03F607 is the largest number x such that ZSTD_compressBound(x) fits in a 32-bit integer */ -#define ZSTD_SEEKABLE_MAX_CHUNK_DECOMPRESSED_SIZE 0xFE03F607 - -#if defined (__cplusplus) -} -#endif - -#endif diff --git a/lib/zstd.h b/lib/zstd.h index 748a98270..6066db45e 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -776,125 +776,6 @@ ZSTDLIB_API size_t ZSTD_decompressBlock(ZSTD_DCtx* dctx, void* dst, size_t dstCa ZSTDLIB_API size_t ZSTD_insertBlock(ZSTD_DCtx* dctx, const void* blockStart, size_t blockSize); /**< insert block into `dctx` history. Useful for uncompressed blocks */ -/*-**************************************************************************** -* Seekable Format -* -* The seekable format splits the compressed data into a series of "chunks", -* each compressed individually so that decompression of a section in the -* middle of an archive only requires zstd to decompress at most a chunk's -* worth of extra data, instead of the entire archive. -******************************************************************************/ - -typedef struct ZSTD_seekable_CStream_s ZSTD_seekable_CStream; -typedef struct ZSTD_seekable_DStream_s ZSTD_seekable_DStream; - -/*-**************************************************************************** -* Seekable compression - HowTo -* A ZSTD_seekable_CStream object is required to tracking streaming operation. -* Use ZSTD_seekable_createCStream() and ZSTD_seekable_freeCStream() to create/ -* release resources. -* -* Streaming objects are reusable to avoid allocation and deallocation, -* to start a new compression operation call ZSTD_seekable_initCStream() on the -* compressor. -* -* Data streamed to the seekable compressor will automatically be split into -* chunks of size `maxChunkSize` (provided in ZSTD_seekable_initCStream()), -* or if none is provided, will be cut off whenver ZSTD_endChunk() is called -* or when the default maximum chunk size is reached (approximately 4GB). -* -* Use ZSTD_seekable_initCStream() to initialize a ZSTD_seekable_CStream object -* for a new compression operation. -* `maxChunkSize` indicates the size at which to automatically start a new -* seekable frame. `maxChunkSize == 0` implies the default maximum size. -* `checksumFlag` indicates whether or not the seek table should include chunk -* checksums on the uncompressed data for verification. -* @return : a size hint for input to provide for compression, or an error code -* checkable with ZSTD_isError() -* -* Use ZSTD_seekable_compressStream() repetitively to consume input stream. -* The function will automatically update both `pos` fields. -* Note that it may not consume the entire input, in which case `pos < size`, -* and it's up to the caller to present again remaining data. -* @return : a size hint, preferred nb of bytes to use as input for next -* function call or an error code, which can be tested using -* ZSTD_isError(). -* Note 1 : it's just a hint, to help latency a little, any other -* value will work fine. -* Note 2 : size hint is guaranteed to be <= ZSTD_CStreamInSize() -* -* At any time, call ZSTD_seekable_endChunk() to end the current chunk and -* start a new one. -* -* ZSTD_endStream() will end the current chunk, and then write the seek table -* so that decompressors can efficiently find compressed chunks. -* ZSTD_endStream() may return a number > 0 if it was unable to flush all the -* necessary data to `output`. In this case, it should be called again until -* all remaining data is flushed out and 0 is returned. -******************************************************************************/ - -/*===== Seekable compressor management =====*/ -ZSTDLIB_API ZSTD_seekable_CStream* ZSTD_seekable_createCStream(void); -ZSTDLIB_API size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs); - -/*===== Seekable compression functions =====*/ -ZSTDLIB_API size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, int checksumFlag, unsigned maxChunkSize); -ZSTDLIB_API size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input); -ZSTDLIB_API size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); -ZSTDLIB_API size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); - -/*-**************************************************************************** -* Seekable decompression - HowTo -* A ZSTD_seekable_DStream object is required to tracking streaming operation. -* Use ZSTD_seekable_createDStream() and ZSTD_seekable_freeDStream() to create/ -* release resources. -* -* Streaming objects are reusable to avoid allocation and deallocation, -* to start a new compression operation call ZSTD_seekable_initDStream() on the -* compressor. -* -* Use ZSTD_seekable_loadSeekTable() to load the seek table from a file. -* `src` should point to a block of data read from the end of the file, -* i.e. `src + srcSize` should always be the end of the file. -* @return : 0 if the table was loaded successfully, or if `srcSize` was too -* small, a size hint for how much data to provide. -* An error code may also be returned, checkable with ZSTD_isError() -* -* Use ZSTD_initDStream to prepare for a new decompression operation using the -* seektable loaded with ZSTD_seekable_loadSeekTable(). -* Data in the range [rangeStart, rangeEnd) will be decompressed. -* -* Call ZSTD_seekable_decompressStream() repetitively to consume input stream. -* @return : There are a number of possible return codes for this function -* - 0, the decompression operation has completed. -* - An error code checkable with ZSTD_isError -* + If this error code is ZSTD_error_needSeek, the user should seek -* to the file position provided by ZSTD_seekable_getSeekOffset() -* and indicate this to the stream with -* ZSTD_seekable_updateOffset(), before resuming decompression -* + Otherwise, this is a regular decompression error and the input -* file is likely corrupted or the API was incorrectly used. -* - A size hint, the preferred nb of bytes to provide as input to the -* next function call to improve latency. -* -* ZSTD_seekable_getSeekOffset() and ZSTD_seekable_updateOffset() are helper -* functions to indicate where the user should seek their file stream to, when -* a different position is required to continue decompression. -* Note that ZSTD_seekable_updateOffset will error if given an offset other -* than the one requested from ZSTD_seekable_getSeekOffset(). -******************************************************************************/ - -/*===== Seekable decompressor management =====*/ -ZSTDLIB_API ZSTD_seekable_DStream* ZSTD_seekable_createDStream(void); -ZSTDLIB_API size_t ZSTD_seekable_freeDStream(ZSTD_seekable_DStream* zds); - -/*===== Seekable decompression functions =====*/ -ZSTDLIB_API size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, size_t srcSize); -ZSTDLIB_API size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, unsigned long long rangeStart, unsigned long long rangeEnd); -ZSTDLIB_API size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input); -ZSTDLIB_API unsigned long long ZSTD_seekable_getSeekOffset(ZSTD_seekable_DStream* zds); -ZSTDLIB_API size_t ZSTD_seekable_updateOffset(ZSTD_seekable_DStream* zds, unsigned long long offset); - #endif /* ZSTD_H_ZSTD_STATIC_LINKING_ONLY */ #if defined (__cplusplus) From 0e30059ba1cba30730174bc44c7a69e5bd7c3c5d Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 11 Apr 2017 14:41:02 -0700 Subject: [PATCH 171/305] cli : FIO_createDictBuffer() replaces FIO_loadFile() makes it more explicit that it allocates a buffer and that it's meant to be used for dictionary. Also : simplified function a bit, now only works for dictionaries up to DICTSIZE_MAX --- programs/fileio.c | 29 +++++++++++------------------ 1 file changed, 11 insertions(+), 18 deletions(-) diff --git a/programs/fileio.c b/programs/fileio.c index b384c3c2b..a3bc7e913 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -76,7 +76,7 @@ #define CACHELINE 64 -#define MAX_DICT_SIZE (8 MB) /* protection against large input (attack scenario) */ +#define DICTSIZE_MAX (32 MB) /* protection against large input (attack scenario) */ #define FNSPACE 30 @@ -278,13 +278,13 @@ static FILE* FIO_openDstFile(const char* dstFileName) } -/*! FIO_loadFile() : -* creates a buffer, pointed by `*bufferPtr`, -* loads `filename` content into it, -* up to MAX_DICT_SIZE bytes. -* @return : loaded size -*/ -static size_t FIO_loadFile(void** bufferPtr, const char* fileName) +/*! FIO_createDictBuffer() : + * creates a buffer, pointed by `*bufferPtr`, + * loads `filename` content into it, up to DICTSIZE_MAX bytes. + * @return : loaded size + * if fileName==NULL, returns 0 and a NULL pointer + */ +static size_t FIO_createDictBuffer(void** bufferPtr, const char* fileName) { FILE* fileHandle; U64 fileSize; @@ -296,14 +296,7 @@ static size_t FIO_loadFile(void** bufferPtr, const char* fileName) fileHandle = fopen(fileName, "rb"); if (fileHandle==0) EXM_THROW(31, "zstd: %s: %s", fileName, strerror(errno)); fileSize = UTIL_getFileSize(fileName); - if (fileSize > MAX_DICT_SIZE) { - int seekResult; - if (fileSize > 1 GB) EXM_THROW(32, "Dictionary file %s is too large", fileName); /* avoid extreme cases */ - DISPLAYLEVEL(2,"Dictionary %s is too large : using last %u bytes only \n", fileName, (U32)MAX_DICT_SIZE); - seekResult = fseek(fileHandle, (long int)(fileSize-MAX_DICT_SIZE), SEEK_SET); /* use end of file */ - if (seekResult != 0) EXM_THROW(33, "zstd: %s: %s", fileName, strerror(errno)); - fileSize = MAX_DICT_SIZE; - } + if (fileSize > DICTSIZE_MAX) EXM_THROW(32, "Dictionary file %s is too large (> %u MB)", fileName, DICTSIZE_MAX >> 20); /* avoid extreme cases */ *bufferPtr = malloc((size_t)fileSize); if (*bufferPtr==NULL) EXM_THROW(34, "zstd: %s", strerror(errno)); { size_t const readSize = fread(*bufferPtr, 1, (size_t)fileSize, fileHandle); @@ -356,7 +349,7 @@ static cRess_t FIO_createCResources(const char* dictFileName, int cLevel, /* dictionary */ { void* dictBuffer; - size_t const dictBuffSize = FIO_loadFile(&dictBuffer, dictFileName); + size_t const dictBuffSize = FIO_createDictBuffer(&dictBuffer, dictFileName); if (dictFileName && (dictBuffer==NULL)) EXM_THROW(32, "zstd: allocation error : can't create dictBuffer"); { ZSTD_parameters params = ZSTD_getParams(cLevel, srcSize, dictBuffSize); params.fParams.contentSizeFlag = srcRegFile; @@ -776,7 +769,7 @@ static dRess_t FIO_createDResources(const char* dictFileName) /* dictionary */ { void* dictBuffer; - size_t const dictBufferSize = FIO_loadFile(&dictBuffer, dictFileName); + size_t const dictBufferSize = FIO_createDictBuffer(&dictBuffer, dictFileName); size_t const initError = ZSTD_initDStream_usingDict(ress.dctx, dictBuffer, dictBufferSize); if (ZSTD_isError(initError)) EXM_THROW(61, "ZSTD_initDStream_usingDict error : %s", ZSTD_getErrorName(initError)); free(dictBuffer); From 5c42d0edc88e75fdc6ee9cf3e400e3dedd670162 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 11 Apr 2017 16:57:32 -0700 Subject: [PATCH 172/305] cli : better status display for zstdmt in 1-thread mode --- programs/fileio.c | 22 +++++++++++----------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/programs/fileio.c b/programs/fileio.c index a3bc7e913..f1f5a5577 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -349,7 +349,7 @@ static cRess_t FIO_createCResources(const char* dictFileName, int cLevel, /* dictionary */ { void* dictBuffer; - size_t const dictBuffSize = FIO_createDictBuffer(&dictBuffer, dictFileName); + size_t const dictBuffSize = FIO_createDictBuffer(&dictBuffer, dictFileName); /* works with dictFileName==NULL */ if (dictFileName && (dictBuffer==NULL)) EXM_THROW(32, "zstd: allocation error : can't create dictBuffer"); { ZSTD_parameters params = ZSTD_getParams(cLevel, srcSize, dictBuffSize); params.fParams.contentSizeFlag = srcRegFile; @@ -361,7 +361,7 @@ static cRess_t FIO_createCResources(const char* dictFileName, int cLevel, if (comprParams->searchLog) params.cParams.searchLog = comprParams->searchLog; if (comprParams->searchLength) params.cParams.searchLength = comprParams->searchLength; if (comprParams->targetLength) params.cParams.targetLength = comprParams->targetLength; - if (comprParams->strategy) params.cParams.strategy = (ZSTD_strategy)(comprParams->strategy - 1); + if (comprParams->strategy) params.cParams.strategy = (ZSTD_strategy)(comprParams->strategy - 1); /* 0 means : do not change */ #ifdef ZSTD_MULTITHREAD { size_t const errorCode = ZSTDMT_initCStream_advanced(ress.cctx, dictBuffer, dictBuffSize, params, srcSize); if (ZSTD_isError(errorCode)) EXM_THROW(33, "Error initializing CStream : %s", ZSTD_getErrorName(errorCode)); @@ -567,8 +567,8 @@ static int FIO_compressFilename_internal(cRess_t ress, readsize += inSize; { ZSTD_inBuffer inBuff = { ress.srcBuffer, inSize, 0 }; - while (inBuff.pos != inBuff.size) { /* note : is there any possibility of endless loop ? for example, if outBuff is not large enough ? */ - ZSTD_outBuffer outBuff= { ress.dstBuffer, ress.dstBufferSize, 0 }; + while (inBuff.pos != inBuff.size) { + ZSTD_outBuffer outBuff = { ress.dstBuffer, ress.dstBufferSize, 0 }; #ifdef ZSTD_MULTITHREAD size_t const result = ZSTDMT_compressStream(ress.cctx, &outBuff, &inBuff); #else @@ -582,13 +582,13 @@ static int FIO_compressFilename_internal(cRess_t ress, if (sizeCheck!=outBuff.pos) EXM_THROW(25, "Write error : cannot write compressed block into %s", dstFileName); compressedfilesize += outBuff.pos; } } } -#ifdef ZSTD_MULTITHREAD - if (!fileSize) DISPLAYUPDATE(2, "\rRead : %u MB", (U32)(readsize>>20)) - else DISPLAYUPDATE(2, "\rRead : %u / %u MB", (U32)(readsize>>20), (U32)(fileSize>>20)); -#else - if (!fileSize) DISPLAYUPDATE(2, "\rRead : %u MB ==> %.2f%%", (U32)(readsize>>20), (double)compressedfilesize/readsize*100) - else DISPLAYUPDATE(2, "\rRead : %u / %u MB ==> %.2f%%", (U32)(readsize>>20), (U32)(fileSize>>20), (double)compressedfilesize/readsize*100); -#endif + if (g_nbThreads > 1) { + if (!fileSize) DISPLAYUPDATE(2, "\rRead : %u MB", (U32)(readsize>>20)) + else DISPLAYUPDATE(2, "\rRead : %u / %u MB", (U32)(readsize>>20), (U32)(fileSize>>20)); + } else { + if (!fileSize) DISPLAYUPDATE(2, "\rRead : %u MB ==> %.2f%%", (U32)(readsize>>20), (double)compressedfilesize/readsize*100) + else DISPLAYUPDATE(2, "\rRead : %u / %u MB ==> %.2f%%", (U32)(readsize>>20), (U32)(fileSize>>20), (double)compressedfilesize/readsize*100); + } } /* End of Frame */ From eb70d219fdbd0834f83bf2c66551a4b3fa145de0 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 11 Apr 2017 17:15:13 -0700 Subject: [PATCH 173/305] Add test of file > 4GB to playTests --- tests/playTests.sh | 22 ++++++++++++++++++++++ 1 file changed, 22 insertions(+) diff --git a/tests/playTests.sh b/tests/playTests.sh index 266bbd91b..6ef5914a9 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -11,6 +11,7 @@ roundTripTest() { local_p="$2" else local_c="$2" + local_p="" fi rm -f tmp1 tmp2 @@ -20,6 +21,23 @@ roundTripTest() { $DIFF -q tmp1 tmp2 } +fileRoundTripTest() { + if [ -n "$3" ]; then + local_c="$3" + local_p="$2" + else + local_c="$2" + local_p="" + fi + + rm -f tmp.zstd tmp.md5.1 tmp.md5.2 + $ECHO "fileRoundTripTest: ./datagen $1 $local_p > tmp1 && $ZSTD -v$local_c -c | $ZSTD -d | $MD5SUM > tmp.md5.2" + ./datagen $1 $local_p > tmp + cat tmp | $MD5SUM > tmp.md5.1 + $ZSTD --ultra -v$local_c -c tmp | $ZSTD -d | $MD5SUM > tmp.md5.2 + $DIFF -q tmp.md5.1 tmp.md5.2 +} + isTerminal=false if [ -t 0 ] && [ -t 1 ] then @@ -441,6 +459,8 @@ roundTripTest -g519K 6 # greedy, hash chain roundTripTest -g517K 16 # btlazy2 roundTripTest -g516K 19 # btopt +fileRoundTripTest -g500K + rm tmp* if [ "$1" != "--test-large-data" ]; then @@ -476,4 +496,6 @@ roundTripTest -g50000000 -P94 19 roundTripTest -g99000000 -P99 20 roundTripTest -g6000000000 -P99 1 +fileRoundTripTest -g4193M -P99 1 + rm tmp* From d37e1df2ab26b9cbb22e6b602ea5da9b3662af89 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 11 Apr 2017 17:33:26 -0700 Subject: [PATCH 174/305] Fix message --- tests/playTests.sh | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/playTests.sh b/tests/playTests.sh index 6ef5914a9..3675bb168 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -31,7 +31,7 @@ fileRoundTripTest() { fi rm -f tmp.zstd tmp.md5.1 tmp.md5.2 - $ECHO "fileRoundTripTest: ./datagen $1 $local_p > tmp1 && $ZSTD -v$local_c -c | $ZSTD -d | $MD5SUM > tmp.md5.2" + $ECHO "fileRoundTripTest: ./datagen $1 $local_p > tmp && $ZSTD -v$local_c -c | $ZSTD -d" ./datagen $1 $local_p > tmp cat tmp | $MD5SUM > tmp.md5.1 $ZSTD --ultra -v$local_c -c tmp | $ZSTD -d | $MD5SUM > tmp.md5.2 From 20d5e03893004cf236297808a0ee51a2a94d506b Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 11 Apr 2017 18:34:02 -0700 Subject: [PATCH 175/305] content size is controlled at bufferless level so it's active for all entry points Also : added relevant test (wrong content size) in fuzzer --- lib/common/error_private.c | 2 +- lib/compress/zstd_compress.c | 13 ++++++++++--- lib/compress/zstdmt_compress.c | 11 ++++++----- tests/fuzzer.c | 30 ++++++++++++++++++++++-------- 4 files changed, 39 insertions(+), 17 deletions(-) diff --git a/lib/common/error_private.c b/lib/common/error_private.c index 44ae20104..b3287245f 100644 --- a/lib/common/error_private.c +++ b/lib/common/error_private.c @@ -29,7 +29,7 @@ const char* ERR_getErrorString(ERR_enum code) case PREFIX(memory_allocation): return "Allocation error : not enough memory"; case PREFIX(stage_wrong): return "Operation not authorized at current processing stage"; case PREFIX(dstSize_tooSmall): return "Destination buffer is too small"; - case PREFIX(srcSize_wrong): return "Src size incorrect"; + case PREFIX(srcSize_wrong): return "Src size is incorrect"; case PREFIX(corruption_detected): return "Corrupted block detected"; case PREFIX(checksum_wrong): return "Restored data doesn't match checksum"; case PREFIX(tableLog_tooLarge): return "tableLog requires too much memory : unsupported"; diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 0ce2054fc..110698c2f 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -81,6 +81,7 @@ struct ZSTD_CCtx_s { size_t workSpaceSize; size_t blockSize; U64 frameContentSize; + U64 consumedSrcSize; XXH64_state_t xxhState; ZSTD_customMem customMem; @@ -241,6 +242,7 @@ static size_t ZSTD_continueCCtx(ZSTD_CCtx* cctx, ZSTD_parameters params, U64 fra U32 const end = (U32)(cctx->nextSrc - cctx->base); cctx->params = params; cctx->frameContentSize = frameContentSize; + cctx->consumedSrcSize = 0; cctx->lowLimit = end; cctx->dictLimit = end; cctx->nextToUpdate = end+1; @@ -313,6 +315,7 @@ static size_t ZSTD_resetCCtx_advanced (ZSTD_CCtx* zc, zc->params = params; zc->blockSize = blockSize; zc->frameContentSize = frameContentSize; + zc->consumedSrcSize = 0; { int i; for (i=0; irep[i] = repStartValue[i]; } if ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) { @@ -2493,6 +2496,7 @@ static size_t ZSTD_compressContinue_internal (ZSTD_CCtx* cctx, ZSTD_compress_generic (cctx, dst, dstCapacity, src, srcSize, lastFrameChunk) : ZSTD_compressBlock_internal (cctx, dst, dstCapacity, src, srcSize); if (ZSTD_isError(cSize)) return cSize; + cctx->consumedSrcSize += srcSize; return cSize + fhSize; } else return fhSize; @@ -2503,7 +2507,7 @@ size_t ZSTD_compressContinue (ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize) { - return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1, 0); + return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1 /* frame mode */, 0 /* last chunk */); } @@ -2516,7 +2520,7 @@ size_t ZSTD_compressBlock(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const { size_t const blockSizeMax = ZSTD_getBlockSizeMax(cctx); if (srcSize > blockSizeMax) return ERROR(srcSize_wrong); - return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 0, 0); + return ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 0 /* frame mode */, 0 /* last chunk */); } /*! ZSTD_loadDictionaryContent() : @@ -2767,10 +2771,13 @@ size_t ZSTD_compressEnd (ZSTD_CCtx* cctx, const void* src, size_t srcSize) { size_t endResult; - size_t const cSize = ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1, 1); + size_t const cSize = ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1 /* frame mode */, 1 /* last chunk */); if (ZSTD_isError(cSize)) return cSize; endResult = ZSTD_writeEpilogue(cctx, (char*)dst + cSize, dstCapacity-cSize); if (ZSTD_isError(endResult)) return endResult; + if (cctx->params.fParams.contentSizeFlag) { /* control src size */ + if (cctx->frameContentSize != cctx->consumedSrcSize) return ERROR(srcSize_wrong); + } return cSize + endResult; } diff --git a/lib/compress/zstdmt_compress.c b/lib/compress/zstdmt_compress.c index c9c3e8533..50c98ae93 100644 --- a/lib/compress/zstdmt_compress.c +++ b/lib/compress/zstdmt_compress.c @@ -235,11 +235,12 @@ void ZSTDMT_compressChunk(void* jobDescription) if (job->cdict) DEBUGLOG(3, "using CDict "); if (ZSTD_isError(initError)) { job->cSize = initError; goto _endJob; } } else { /* srcStart points at reloaded section */ - size_t const dictModeError = ZSTD_setCCtxParameter(job->cctx, ZSTD_p_forceRawDict, 1); /* Force loading dictionary in "content-only" mode (no header analysis) */ - size_t const initError = ZSTD_compressBegin_advanced(job->cctx, job->srcStart, job->dictSize, job->params, job->fullFrameSize); - if (ZSTD_isError(initError) || ZSTD_isError(dictModeError)) { job->cSize = initError; goto _endJob; } - ZSTD_setCCtxParameter(job->cctx, ZSTD_p_forceWindow, 1); - } + if (!job->firstChunk) job->params.fParams.contentSizeFlag = 0; /* ensure no srcSize control */ + { size_t const dictModeError = ZSTD_setCCtxParameter(job->cctx, ZSTD_p_forceRawDict, 1); /* Force loading dictionary in "content-only" mode (no header analysis) */ + size_t const initError = ZSTD_compressBegin_advanced(job->cctx, job->srcStart, job->dictSize, job->params, job->fullFrameSize); + if (ZSTD_isError(initError) || ZSTD_isError(dictModeError)) { job->cSize = initError; goto _endJob; } + ZSTD_setCCtxParameter(job->cctx, ZSTD_p_forceWindow, 1); + } } if (!job->firstChunk) { /* flush and overwrite frame header when it's not first segment */ size_t const hSize = ZSTD_compressContinue(job->cctx, dstBuff.start, dstBuff.size, src, 0); if (ZSTD_isError(hSize)) { job->cSize = hSize; goto _endJob; } diff --git a/tests/fuzzer.c b/tests/fuzzer.c index 342c953e9..cdbbe8a22 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -524,9 +524,9 @@ static int basicUnitTests(U32 seed, double compressibility) /* Decompression defense tests */ DISPLAYLEVEL(4, "test%3i : Check input length for magic number : ", testNb++); - { size_t const r = ZSTD_decompress(decodedBuffer, CNBuffSize, CNBuffer, 3); + { size_t const r = ZSTD_decompress(decodedBuffer, CNBuffSize, CNBuffer, 3); /* too small input */ if (!ZSTD_isError(r)) goto _output_error; - if (r != (size_t)-ZSTD_error_srcSize_wrong) goto _output_error; } + if (ZSTD_getErrorCode(r) != ZSTD_error_srcSize_wrong) goto _output_error; } DISPLAYLEVEL(4, "OK \n"); DISPLAYLEVEL(4, "test%3i : Check magic Number : ", testNb++); @@ -535,6 +535,22 @@ static int basicUnitTests(U32 seed, double compressibility) if (!ZSTD_isError(r)) goto _output_error; } DISPLAYLEVEL(4, "OK \n"); + /* content size verification test */ + DISPLAYLEVEL(4, "test%3i : Content size verification : ", testNb++); + { ZSTD_CCtx* const cctx = ZSTD_createCCtx(); + size_t const srcSize = 5000; + size_t const wrongSrcSize = (srcSize + 1000); + ZSTD_parameters params = ZSTD_getParams(1, wrongSrcSize, 0); + params.fParams.contentSizeFlag = 1; + { size_t const result = ZSTD_compressBegin_advanced(cctx, NULL, 0, params, wrongSrcSize); + if (ZSTD_isError(result)) goto _output_error; + } + { size_t const result = ZSTD_compressEnd(cctx, decodedBuffer, CNBuffSize, CNBuffer, srcSize); + if (!ZSTD_isError(result)) goto _output_error; + if (ZSTD_getErrorCode(result) != ZSTD_error_srcSize_wrong) goto _output_error; + DISPLAYLEVEL(4, "OK : %s \n", ZSTD_getErrorName(result)); + } } + /* block API tests */ { ZSTD_CCtx* const cctx = ZSTD_createCCtx(); static const size_t dictSize = 65 KB; @@ -788,12 +804,10 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxD crcOrig = XXH64(sampleBuffer, sampleSize, 0); /* compression tests */ - { - unsigned const cLevel = - (FUZ_rand(&lseed) % - (ZSTD_maxCLevel() - - (FUZ_highbit32((U32)sampleSize) / cLevelLimiter))) + - 1; + { unsigned const cLevel = + ( FUZ_rand(&lseed) % + (ZSTD_maxCLevel() - (FUZ_highbit32((U32)sampleSize) / cLevelLimiter)) ) + + 1; cSize = ZSTD_compressCCtx(ctx, cBuffer, cBufferSize, sampleBuffer, sampleSize, cLevel); CHECK(ZSTD_isError(cSize), "ZSTD_compressCCtx failed : %s", ZSTD_getErrorName(cSize)); From 88009a8ba27820f9a261d1f48cc8f307eb546b35 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 12 Apr 2017 00:51:24 -0700 Subject: [PATCH 176/305] removed srcSize control from CStream since it's already done from lower bufferless API level --- lib/compress/zstd_compress.c | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 110698c2f..d6774372b 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -2971,7 +2971,6 @@ struct ZSTD_CStream_s { U32 checksum; U32 frameEnded; U64 pledgedSrcSize; - U64 inputProcessed; ZSTD_parameters params; ZSTD_customMem customMem; }; /* typedef'd to ZSTD_CStream within "zstd.h" */ @@ -3034,7 +3033,6 @@ static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, unsigned long long p zcs->stage = zcss_load; zcs->frameEnded = 0; zcs->pledgedSrcSize = pledgedSrcSize; - zcs->inputProcessed = 0; return 0; /* ready to go */ } @@ -3226,7 +3224,6 @@ static size_t ZSTD_compressStream_generic(ZSTD_CStream* zcs, *srcSizePtr = ip - istart; *dstCapacityPtr = op - ostart; - zcs->inputProcessed += *srcSizePtr; if (zcs->frameEnded) return 0; { size_t hintInSize = zcs->inBuffTarget - zcs->inBuffPos; if (hintInSize==0) hintInSize = zcs->blockSize; @@ -3271,9 +3268,6 @@ size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output) BYTE* const oend = (BYTE*)(output->dst) + output->size; BYTE* op = ostart; - if ((zcs->pledgedSrcSize) && (zcs->inputProcessed != zcs->pledgedSrcSize)) - return ERROR(srcSize_wrong); /* pledgedSrcSize not respected */ - if (zcs->stage != zcss_final) { /* flush whatever remains */ size_t srcSize = 0; @@ -3289,6 +3283,7 @@ size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output) zcs->stage = zcss_final; zcs->outBuffContentSize = !notEnded ? 0 : ZSTD_compressEnd(zcs->cctx, zcs->outBuff, zcs->outBuffSize, NULL, 0); /* write epilogue, including final empty block, into outBuff */ + if (ZSTD_isError(zcs->outBuffContentSize)) return zcs->outBuffContentSize; } /* flush epilogue */ From e80f1d74b32ae32859fb845e032983f50be88433 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Wed, 12 Apr 2017 10:57:38 -0700 Subject: [PATCH 177/305] Address PR comments and minor fixes --- contrib/seekable_format/examples/Makefile | 8 ++++---- .../examples/seekable_compression | Bin 2557127 -> 0 bytes .../zstd_seekable_compression_format.md | 15 ++++++++++++--- contrib/seekable_format/zstdseek_compress.c | 2 +- contrib/seekable_format/zstdseek_decompress.c | 4 ++-- doc/README.md | 1 + 6 files changed, 20 insertions(+), 10 deletions(-) delete mode 100755 contrib/seekable_format/examples/seekable_compression diff --git a/contrib/seekable_format/examples/Makefile b/contrib/seekable_format/examples/Makefile index c560eb075..b57774ef3 100644 --- a/contrib/seekable_format/examples/Makefile +++ b/contrib/seekable_format/examples/Makefile @@ -23,11 +23,11 @@ default: all all: seekable_compression seekable_decompression -seekable_compression : seekable_compression.c - $(CC) $(CPPFLAGS) $(CFLAGS) $(SEEKABLE_OBJS) $^ $(LDFLAGS) -o $@ +seekable_compression : seekable_compression.c $(SEEKABLE_OBJS) + $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ -seekable_decompression : seekable_decompression.c - $(CC) $(CPPFLAGS) $(CFLAGS) $(SEEKABLE_OBJS) $^ $(LDFLAGS) -o $@ +seekable_decompression : seekable_decompression.c $(SEEKABLE_OBJS) + $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ clean: @rm -f core *.o tmp* result* *.zst \ diff --git a/contrib/seekable_format/examples/seekable_compression b/contrib/seekable_format/examples/seekable_compression deleted file mode 100755 index 88e1d2949b157168b81cc7c090cc29dd439dd425..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2557127 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

@3hU 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

`-evC*HutB?YDj=}8j`$HJd$f8s7i&Y5y6j2cc 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⋙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

    WO&T%e1ybc{ zkG{ zGtKl9Zh(2i%(o5gaL+mpJV)~e?UfP|~LTQ3*kb78mhKWc7DvQ)q1kpxXAeHO*SOPn_p}jRzgFx;``O&JPSv&Vqz-iD{ z#}XK;8{R$ck<_hWHD-~ISDCbwkJE1l_g0^eICO}By`AvICbX{wFfnCxgi&>4jeXoI1JLvgU`hg*WFjP*J(JAffJ4kg53g3F z3pN7Maxf$aW^7(U-5Sp(E(88we3NLJ87afcc7p;#I!8dMV8jrVjL~!U@QfU=2L?4+ zcX)aJ+=OQ+3@!1Q+a6bLXcUtk$Hw7MA^YjL@)#MkgcUtQ!?XRwM@$*sBgYsdTE``z zZm=fT+AhoKstrL->Jl^Lc8sxlth@5!DJ=ccOjwvD;X+~aoflv*a%d=x1BP&BAIvzD z%Q!PZct4)p=oDek0?=lhdtBS4m2vtIiVo8gLpa_;-U?SLINpwsWp3k}sC5hyF(~K< zyN3GbE@c%PF^iP@jAX>v%+lISa3;(G4OcH@^m$;l5~u}VyBR!e8z?`}>O%Odop)u- z`U|6u$-S+S=|fO}=jOFwf+dpE5vrjEEH}m_m}3HEva!nsAp zf#6Agc}(_H%o@1C7O2*6JRjO7;>C<+&~VxzDaErZG0z+;E0#7a5VMbs?U5kXz+G^Xx}FcLB^-J*nwyF&*TCJ}qV z+!aAB724a?%wVK3j3%}uHk7WUViLkq9tHuAC2Jy=C1F8C9O%N7&Pp>Xmq7?YjQdK@ z?`E=qYsOASLTRF#(OSU(J9%uGD}6~l@j)<@;fOGA_DXvtR!DKW>2jPG7u1;Q8}^yG z89C#V6pCmu8~x#MgR#R5b%n9Q`#U87wBpH3FM2u-ZZuMdcIb!}as;8a!nTJIAB78S z8Q6n5Iol45PO*<{UzMJyi6lmfqVG8#jb7vI0*$38B}8%Ig#u8>h0|#AQE1#fL@icI z?TNp{puI?6*V0voo^%4v8gN?lh7HbJP^`8 zh?We&m>6*Nl{N3d;K zH^Pi(ai*;$x78@P|*+p=qattq2jU8f$I3}#ZK}2cY_0qF|GI8D9s1;>+ z%q=01xmkw48_jkU5A+yo_8JQDK!>W>@HW~G2&p8uOST%rwuRdYlSM)&XeAy)Cu#~= zI=I?9#2{nXfU{9wu@;rWMrXtBlTmTlRMa04R3S zsR3IwhvbFCwB)eqq$@PhHk}c(n!ysu9TMaC<)A6+BehR45(aje(}6XhOA*Vpmn`WW zaW2)4OK5G}s`3 zM$9*%D^_&O3e}ksU7ZkFbGfg0*e&=$w=B_t8(i@Q!D7ovx4FP}ZyPBfYE{;5crP}v z5IaIh@@%Q@b0a&okpLMKRTR!j%4u zV$=eP1A`Z~p~uK(KLHYUtBeVPY6W#yFUd|9d%le0kXkV2WrRH~^u}6?kRfJZ^W=kN zhxLq@W}TI?1iTuxPc{%6zA>#Ba}y}@g$^Jwq(#JMU`m=zbzV;BTTA$`NnrxY?cx1Q z{IPBbI|Joz(tGW6C?=wOm<(f`ulgh-rha2hL`@UvHDDumPO-`(JnUn2>4Z=c7EPH$ zoz9|jpo@iNR3(*|*H+1n4H?p*3D<3>q|r=u9mVE=WM&N-SLTM2Z=AiHK#>WFT-_3@ z9EmcDy^nktAj1_lU!)S$jiD;5DVK)~21dvYo~9rKKQjKVTxbj7qjNMjq}-&H@G^rL zUxJ__QbX(r-(*V%?jOiBwW@y1G$Yh#HqM1WK=E#sW_A${Dj)2(7AB4RDS{3U+R^%K&Ioo3(&B z>8C{RhP`{K8>bezx}&$9IITtvBUb=va*?9Jem{SJL6)u9vd|H-9EjMmk(PP`yM(zY zeTGd#w>fLPi0Fo9Yg(5IPdkL=g`qSYj=pdU)7c~T(?Mf&jIHOLIol#1Z*wx8%VagX z+d^sERp4|Sev%^`NRBkk(R*;r@*jMq83IDo$`#YZ{VX9a7v_2yI#k3Ndjo5U521)| z(Y=Kd^(53Mj=snO#SO{VGJ>3G*vtoxVQOKg8UgT9Nz4@iAp8_kZo zQv~5Po>F>qc!|Foweh6lY94DpjA8_&R%ihpQPw%*k{0+&*hvm?WC~ynpkAubRA41i z#>fp}QhbxqPsK`%oy?MPq>Z^NycRb4#s)^8C3R$fz`$xuUGq*QHJF^7Y> zz7*GnrjbDJMTjq+6xLF%-Dqq&$8ID+3&KBzE+O`k_{u<;_v-B#YdmFUM`s27$i!hPwb27A0@u+V zoi>1@>2rJu7>Y+Oq@+XL}%25nGf(!=EYB&?0-(JUF9<*GTGfm#dCdKxlh*8I{I zU~;OWE&Vq@5@p^3O%9Z-KHl!K1w}0w{e3y8sL2l{%lERCMSXZ0_uJDy-^*1N_2KW5 zX}Zv-uyo z8DL!(#d?fnOp@PmUd~G#KVZOTkD7G!pwAz<)JvRXA3k@Mmlz#9o$V!t22Vw8VQ%S` zmVNw|w1s+U>G&{Ag+$+v4U{%}eHKh9fI)Z9%0o|?Q_vT*gsCl+Oh{VMN>#F18` zxnlQ)tarpg*WY;eE9>t3`bDb_FKUN6ESdjW#|J;L<5&0na@%dsPX4$62*UP9`DBvh zd~*KQmQ)tXz2QYocS!4pdC=yiDOJ=mOQq})kfA+Lwy5RKw~#N?q*D2#n$U7O8plwV zA5GJu^k{!>lGJVT8|FXF{hqu9i8ZMmpZhOngPa#-OpA*oeki~9%5NAY&AngV0wql< zN*n&CQ9A2mPRlYOLmHhgxyAGys@1 zsTE=QpWSPRTBsCI=gI{C2RrvK(B%BR3?OE$pihmGi?xC#e-Q?NH^&+enp&e!B&4x00e`lQX3dGmW~ z>!pCUn*66^`Cj3S;&`FVVYbf&nmop4RcZ1Rn^mXDGi+AXcO|*aW-ZX<4#`5FYl`}y zXka)sd5j(5DoswcS#_FJ3WKtm%-B*3GW^ zn!zf`Sr>JN*F~$Pg>T3&)YEg4RQ4*pLBQDc(x2Z(`i= zL&_854)G8FJna2L%8x#BV87vC`eI*i+~EUjEgfo98_Nc;X!1PCD!n?P z%9lxrO}297cO`kh&8pF)vcy2I5L=y4Yx@3isfu#3)d`hTE_yTQ!X_;-qW@}lYz24S} zt=_cOC|j#i`?|x{icOEN2!wA~ycTOxr5l#w86QJjijC5XY7;CBJxwcRk#|NET?N4* zttQ_Ss15)#IJR_A^HuSbd3D;mGa8y)CxAs;(R>9Ctwdqd`AXS^CS~nOtCgClwVuu( zP}rJOva*uZwvZq#gZ%BismccFbSeV`6cPV zCD)dZL9Sz^-f#=xVof^aTJcH9rA<59O_O7#w*#;caml6m_giQx6$wWnYVs~iu5c@w zACN1iLlw6g+fhu9M2mcKOG!IX)i?i>+FbBc2cj7A;}o^+dO_%vXdj>3lQFl4hPA;K zX?Oy0cwiNVjlz}Q=g#)Xo!%Rm+Dhub8_XM|!Mo*mKksbE$ELYImbcJ1ntZNgg{XKb zGbou$%sW~{N1N8rnqDiN9Om^(o8qNX=6Va`;wThRB#&{C3~AUfb11I&*rsbW`G92Y z=dEz)rMbTdD)a`+7AbOAc%pCtC~6s{b(t4K?Qv;lM@e`z_gM>1l_s^#Fal9%b-8zx z*wM#)y!Ax!SSdf;+TWF$to%Z82qzbNRFJ|5`3FNI=A5vM9v zx;VqSK$E%_i+OwWg;L_b4q22&N9=h`QB?X3u{0fC)U>W)fQcq`$r8gvU+6e|RKLBA zk@^Y9Fi~8z%~Hn7$0{RyrmPFqqn--w4q4igau<~nq9_0*bXH2*)|G527`2RgY2b&^ zq*fzO&-|EoOe?Onva>*wTG98G#=UA!3fz5V=NbmKCZE{bP@X!BZ{TaiqODk?$q`P) znxa14D7jdNYSQhYCWeyspzy#D)1+Ims@IBJENJsJslXT_>XX6)YHQN%Ax>=tZIJC@ zfhLDZ7M3NQ#WnJ9S5Quq_t~r}O+I6@=4D1T6s{6G4h)Ho~>G?$p>xLd`&8eN*Ku33XIATPz##;nFYT_liMY0fAW;yi_N;T zkH~wUD{A(`@_Lw;^{(hM18%7qrM*%wl2^)UKr22Kbg6WGE6CEM67yj7BJFo)>%xHy ziN5+!YN3d6AO%JARk@{8jSj?sAWM^n*{n)U9%ZwtH2Ec)RjJ9LHmgdL!zD|dPWr6O z4cf=YwMNJF$5Q)1Cayy~eeqMr6)z3pxOf2P_XTg_@EZHcv#wxwzTlCMdHipxbwcbK$$q;3CHO=|m}ht~UO z8MQ>vZpjs<>3U1gI!!86DAw5$xX|>2CONWJ;oN7tCvUjAlP#~ann!F%rQh{}3R+Al;w99($MGGm75P}m$M0<5f zo0GP+S}D?My?{pZKTfszf?SgKBHFT}yfoQlM`UU5w9-PCSjegr zAxHUX@&Zf#Jv6C>l#V5=ft2*-lK<-;gZx@gi3ZO8IMv#vyPQR`PLtXa$ba@HAiuWk z%mSKRDV===HcYQtKnq=AA*)n`oLNMZms|4hp-C+ikUusHsm1xK^h%tr#npx@#r<0E zcT&&T%>S91TP>dR)y7pI98>!?!4p|(l_*%qii$vI!sM)xUrb(_JXf;Jf<>Rc8Whyz zMWKQp-9t^O9%_p59%@SU5MSBEai}WQLsf|$s!H_`UjlUthQ$DRO>PPyuTQrG1vPnV zs9=;z@g8bQ^-xoc_fS)+hd7nuJyezIp{hg=Ri%1}Q|W%eaC?BfCZF6R^3TcJzXTOE z`PLpQ>f3$xG4urYYO>j8#jH=Qb&M@iqe-n*?rn$({##PKa$nn-CdYkBv#K?DqGTEA z(Wldcf|_g!71XD7K|xJ^HB?ZaHtlb_Hi_V`0$gVSuF<4+L(tU}^{LZ#TBFJJHfy0K zwP#aTpKh`RYc%-_o3&7rL-#Y_nY#KkU2@^e(BuF?%M{e7V=b~Znmom3E!3ov#MBMw zVNyz!!z@**G^x}n^G?^Kl7`x}8pMHynNEfO<{w1@+XCNGg%pTUMvuV$$hQiMwc2q^D4xM=bQ z+eM-GF0@e4MT{2Tm3ACjXz~ZPi+t~0XrZ8s7%hJy?Krg5)S ziY$G~o7CwYZrlw;%~TmIu~yRhL6JR4tTMKd*yzNeZ5<`BB(TA=)cf2~hZZ#_=p{BB zL!?r*{NmHOLyP*bB*;9xs1NIc%)v!{P%VP$N5@nOGQSv;EbVxzG*|BB5}KYRdGKj! zvQo14CFk*-yMO=o$|H)Jw=gKKNu?OI8l{(FS0D7epQamwrZm}BqG!!Z2gNn%^ens< z^sgjb9`t;WrdI_`Y4Y3>J!{_iL2*qwJs%YGX;L~)mj*o#(DY?NQ<}V{M9-S1-C%H{ zy&Ny5%dw))X%keHv(Af@*IPX`ZX7#Dn=;0RHl{WU3Tjf@Hw8bs2WxV=J=9om)mFq* zs!6pKF+*z7?KHl!QF5p?j#YprwNrS>6(?=0gyUQr3g}~@Pg!emp(dBxtV&H@Dp_n( zJRcb)45Wf{yq$*^>OB6HAWI=tg6x?#M+y5IrKL{+mNEC8V)urHKc=ltlUIrlAc&51 zy%f34wh~)swaBY&wn%K#L23DGTO>9GXc47dDaO_^Iu0PUCbb$5NfZgKr-e%SZB_|r z>n(^in!Hf5_UIAULIEvL)ueV?g1pfprLc~d84(lts|3VN7Wyhp-ejTQgVR|H1;bpY z$pHG;$VZEmf<9g*4*mB9#Ct6Ci#2(_g??ip*9Di;2VGq7|20Kx#C(-e&dUo#+jS4yhQ}aGIk{V-ks0Rxme7 z<&ZI|nSs)dNrl!AihS}pPusdKhKS|_y~Ggtwp9A1{PH;J!)rn2;YEGuE0YXL4le40 z&Naw9x~LC2@gOs9jB3YPsFY+y%P?@!p@RXk9clZy$Mw!PjJf zLM~?0E4v3};%G+80L`OgXhsW_Li1As+W`3mngcXFD1cj&|0Y>3nwqEZ0?nxS=-T3! zc1f~8lM@dxg%PsQT-1kp$s`56pr}vZl3b`tP4+v`sjUy=C9{M|T_;Uw{r^H!Hw!Mm zvp7|1QqcfTXItnuTj(n_`9qskqsjYhR^?-o{G-jP(d4@}t5TB*O9>nxqYnWkVngyx zX+k@OGeeUK1BHHG(MN)c;CU$Oi=Ro|{SkHKy#BdlZ?n)KLv9402l?{N=7jHZZs=~ua>)U=`Xl#jtl zn$)J4rK4??V9LjP=cGSd=Bm-;Uu;(8hmt(lQnyBvU$9w~nj9xte8iAGD6A!fRxu2U z)W(sCxlJDxnBK5pR%!Ab$#Mb(X(<0H+3y846tTe;iz-bjHYM0bUk#I5CD=w^jSL!& zQ=)3Byq#rBRcf-zW>sl&k9!KKK$A0M&VoxrlXE4@Xl#Az2nuRa=Q~qSpDqsyYVz7pi~97F zeg>Bk-jq1NH43nHL!H(X_32#OX^keYuvrT=sXd#z`gE5qSfj~5+pL9}TyU^~*3{Lf zGbIq!i)Jl~WJB(GDRO(=5q)8Ec@zGkX3fdiWO%V;n8p>5Z11 z7HaZ#n^mdFZIT7;e@5|d^6+U(-AYX=;o%L{r1Bp4Rg=nlD5c3eq%9i1nxa1GV3pW@ zoGlzV+cM2*npFJ}>;-C0bx>H7%CkXX&ABQltV!ixyZxvc%ECb_niHT9^L4BjI(n9l zi#4gl!}jADF(%b|KeFQ#n@6?C!?s9lS$(0r`n@d@^O}B5UcF_D%#SJv{I*U6Kx;ABIRc%ZsXB5<=wr>gs=$ll$2WwloJ=9om)hNQ8p~*j6 zZQP4}^wFJK3sgIW8K^iZ0wpZNJwShtHH#K%@)4U=smZ@e7Ehj^5j+AIDm6JXKv5^G zbdaTxDna&4oAIK0j6myCfMsl-?Ro+7DBI)dnpF5oPDWbq1&ep=QvxmWJ6j}n1i%gQ z>YuhqY)bpKyy|ZOoF5(Xr=*AjizZ){R|nv85a*luqUI}(WnK(?9X>5mN`~TP;!7gs zkUh1dWt4TWuG6GeV=jtfuJyD~DQDf^g#=or#8;}(BBfA`mx-gQWt`#ESgU4 z#Z=o`EfoW!LUXj~5>77#^A4#THjgwnP?c5D-nD_C$S0p^w5{7>m}pMWOUyjFTPmq) zF2&T*GCbH(jG~1~ses=~XD`Sv8BCgfMe?9>H2H>P#lGU#7k>@vXfnVfZj0@b;z61C z-bAzvaQ)~)rEq;uVUwAV48Jl-?k~ zq{$~GiyJMIS0!x>_bf+5@bbd4VH{y z@(XMbTMs&yb2u@{5X^Y?G`#S8TNvYc%Orj7QvcN`;)Q zxImLD_SmCVTy87YXwt11@9|uz@O4{pfhI4pSv8uxRI*B=-7RmwWh*Yw=3Q@U0bh8lUk4PiMv>HQsEw3u~L&- z5ubX;bc{BBn^fxy|G?#aF7NX_dDgmnA-CpoIS9pK>Afh$KCrzix7srCA~9k`i_oX* z(SO>*r>9TiS0Ly|7m4*0ZKYICpO+p^?7gRWkyuaBBBgrL9_IJnQ@lv5r)ZH{j{R%R3#cb+kTyU$PUFy%28( zYmS2UDS7i1ygXbp|F={-$a&dk0ZI(eH&ahBfF2Sp(dRT2+fQFtO0j`bY(oNXxg}Fh z-_MtFgS-t80stB+JMq1`dDcj3X>g(f$4;67@ zm_CL=JmmU*yp*$uM+<6!iBj-TrI1t5&|FKLBqamHV@0)yO3R)Q_agsPa6c5ThuYJm z(#ZEVbg$8er-fiBr%COW%SD>^wEZV-`&F7$4#c?{np8$JQ;}7!^^O zpx7WFA`otZfW{j--l(dsZo8Ad2!n$ta=(O2ssbWLMGYFAi0Fxkni*jxqt1wmPE=G> zMl(L3nUQgVuQ~ylFLr!?X|1BxICYC-uKNP>EB+p*4n?d_S*aGbE#9g z2F{sCSkfjAZ^ZB5v<&`R;(78)s(lIiu*dlrKEusOy#Zsu#%EZE+wyd$+E)x z+x4%M6YvWrY5;#^qB?LunadgLrUAOz!=`itc)W^4Z^rRp4*0K4wlo8O%4A1mhBeItwIee+BZOQQ)Bk}c?5Fy7zuNg*Wh-$0-T8mprMW|z4QVYvnMG~?IgGAdl3St85o<}I^f0oE{>|$G{6v#2 z&A?AF*%29-9#S<>SN+rVubf{4uP{*^c$JFwsGV0k@__QIs^`dR|7MlHHJH3j;a7v` zYYOiRqBu{`IoniPz+mOhkg&SNtQtW2l(W)ZG8(~^TM}l}#i{yt7m4)3+R%X+g43;! zCh%ev{)hTz;!ChE4P(d>lLP0P_6ujua}yy2JpibUyY2`_ktpDzPe9u zqIOe*OU-cn9SYyAe;1T{n3I=eNMJ6B142MHVP97}I!-)*La$F!)qGUI2dPY)H4kAd0?++pNQ)ggC@hYcoGh03WE5!Pi9#r^Z?`Rd>?!df2RK4Sk1x7LQr z!1;*o^R=n^dF+ym|5#n_Der9#>AY0a$*2FNTJl3?;+0R4QWJZXJ}jN+#eFZ<;_ZYv z0({CVmlfJ&9%wyJrIIk<3%e9)%>+f@U0sU3sNh!a?*j3eQ<=zUebjoG0RDZYhlz|9 z{U+`h6F=d}tB7jgVCL?rKS|St>QlvR-TcE3l}gCLf-9Gi+Lwjz_{D)CqlDsNWqL$a&Pch0dZ5<~(L zizq6j4Og@=%{Mw1)`pMuU%lL>Bz|FublW~u?l%>;>mEFLS;2m>{*{dpa9(!7Ka8`0>jTVn&)yfA3ilYz+2Q)pk*3lDK3_$HwF^2KjV1Lm zp!8M*Z)?q419-QJa>GVO>z6?hI8f=KksrLrsnAFCqSxOXxdRet_X8T>wC-8O+R z%K0^oAMO$h=4$n^*?f%y@%5D21+pO;+`RpTMZRS(CYSv#ylURM`SOc4!?L?Zkth@U zW+*c%G8*$T8jOl;eSk2|pHynep4UcN!K;<->-Fzr^3}T7EpBJG$VaexOujGZR*Ah{ zson@^u(n|T?58TkPa%vgsgLVbSwaD3CTaljFIv4EtyinZNb*b8TL_Gg@RtALm289U zbL8~0M=^fk^SH|$+7+cM#R(+lvfN|CxKQ$~#9TUwH8pm#u$FK6i&LH08+@2N#n~4MKbU9?IMgh)@-&WSwC)=* z+rmfXKX}oN6bcZl+?RjA0K*p<)dsL?=+_hs34ea=!nFe#Eiy2_c9BSFK}oLHij)?4 zm|yFNloqKdJWR_U#sB3kVAt5a6@IfV^GdCKHVSAu3-f zIzfl)v-(#=z*|k!1pb$a#(=lj$c_LBDDjOYqJeK^u=fc(q5k_If@knO5@+hAnxR)% zrs_bf`EZbH=n>@gA#UJ37SFVgXF85Yd=t$=BQ*_# zuV?{38MJ`#6)j+(?rxmL)bnEs+!+%&y8z-zhW|!a#P8PdsTE0HoGCsU(6r>;fj?%c zssnFUk&N@8)jiWtXaUdFddS^@SsX+SxHFW^U`7f_k}LyvYBq^dOnU{epb`KqF0HTt z)3qFj>5v**SX-r+OEN@oStwf!eO;yJQ6QQGK=SBG$h=3%w=z<1|b!FXNXm#220l53HPts(bX%*1&9Ng9otH6BJTa7XRjD2E5grmyp z9}hb!9KFheZQNZcrAG#{l=ku%I(3tUPM=|E837)qBDtTtGNUmO^kDWhEy={!sqlWF zSv7z-lnA^{R_kp#RGaUtyGj6h|D z0tROiGtLOGNJQ-HwHp1cmW4V){$(rwx-b8F?C6=Qjdf~YkIvhJXCP*Bj_+D5K3%Uk`mi|~0|o+}jrv>Ip%3hiGrJKWoMfmL^dye5EG|I$l|^VlMgtPN)P%&) ze%|4B>fDCQ9mjVl#|vA_iU4G-0Di(m)4+|(ORe%OBSc&xeC=t5Eg%l$2o0?I@@Emf7fK@|(mVx1}X4nL-Fi{;?_2qwV4_{E0=?lbA4k*8B_eunrEYEak;bMNREhiqd zaA@Z#y_}L#Id}tCTVPYbgGQGX4&)@|6AJG#(G>9TG1FSA@Y*1{P2sMuGA-a@6HNl! zCYk|WYoZb0znExdbi8!CV#3@5AFXvJ`qpF>R2)jtk`XLc7LgMXIOuAV8w2mvwMT;pA+nWOtAcf3k;WmTKfd>30ioP zhplZIFE`<(?TiM4CA(ta8?{;s<2PyJMl0bl;D@zd3%#g1YcO6Pf&jut?Cv*mbt5N? zX;3tUCXI`X$tVz#g&qxP7W8N^*W|zhq~Z!=8kx<-F(6}>I01@-&5!j&%aBNdfq0T0 z{llp5(1f9!=Cj0|feUPm>cF~*#u!OPN(>pFA`<4WR7Xv?ITX&x3uO$eT|Sq3P?8bKnh5X%|MbQT0oK{T0p`VEg<2G z7Vs68i4kh)wLwJqr(5_VK76-fK29f22(U#VPz5rsbQen3l^c$3x0yH!#JB8mS7qM& zXzabvi$?hM^Af929!=61%A*>EG)3Bg0$<@Vo~x{xO@ow2SZSUm1mY8I2#PP;v&O+d?mP z_6$uEqn8Nn2&dABy`XZms!{{X%!GU&A7y%xlA}NtA?IBsZ-MDVjRQ!{r zWE8krMd?d#-msN2^w-(EVIyVeud~7DjzAUo&$CkXhOLyLr|JzGDML?ni}JZUPzC<` ztW>>WD`n`Zdc#J_&{N%}e103K0^h%4Suv?(^AG&6iprXxwcn&E#^rXjHxVi@C)9GK+m22*aGl zxbeL;C<5{cr1b0$A-~!u4>h z8eC?EW5DZFv`BW9m)Dla&s(60{^ee`esmiOYJy7TR6K|Ue_ zkIA2B&7T$wDtvUNdO7Yv#1gr}1?hSAgGp-SE6!4ad7QO}(_Ng~XKOYn(&F*RPpFWw zNj@{l1V3Gs6I;Aj>qn@BfwQL_V?MX239EwK76Ouna)F2qnQe@T>{NloR<4A&A-Ke4 zPVdW>@~W>^=6$)P9A6m9A;w#|YeKQg!u-p->4z$bgLPlx0Wzn`z~hWz$O2cKqtMvZ z4k!~qyd}?N4p;q~&8`K!OGP=0Rz~Y_Q_BiMa%l@VtfEqp)>T0xV%R_=^g=Mn>y_pa_Id>5_dqlCMmqBEfyVhr zeK$0&g~Y0e%BQr8#8*7|h~DS=iYLtGthIQs7=_bx6uMK)w>tUsOw}q=%%@1HiM`}0 zIr*l~oZk&4xc}5VR33Oobf11C$3!42OGP^l3KdqIVxhH$)mvjwA=yhV$ZJ(;li7^| z-#)#p=vod%z(Y?dMfoB#n#IKMb+h?u0Y7h|3E+2CG*G*^23|iymlMF-R3!HzceH67 z{c6(#HdG{f+uAg)2ztN|2ECna8pM_Ob{0WFTPhxiReA;YhRxVexNd0o$|B0LECCq? zxeW)b>4?Z>noc&T4Tu%OUjE;u`Io0$HxKrqv@O=DeSs0QLUC){4^-tI{VU5jkO<_q z88Gm^)<5Nds~@Q=d1LV1@(tmoOEM()Stm{4WlAk5$5H%FjixP#7O+BRs~e`rsD8~# zb_}>cMf2r4Z+nQ=@2nQb;&GDO#4#=0lbZaP0{(?jZ35{;<|>d0Bxg#%ABDNJxXr}=6E zvCrRt46gWE+vznV%$@vo!r+SGMmHs%s+}JgNB}E#{<){)RsYmr2dvoncPzb6^%vRr zPhF@m6ycN~oT)hC7pbEc8-WIpjLE_Ugp*uDzEN)DB)*T^Qa@wC9}r_Hxw~BgPgnid znOzIWk|B1R+BA04`6c@Y;2|n1^x`pWBTF=}!v7Y>|7!KS zCin+d>~8C6M~3BG0$8#0?{=dO?c6=!}0z!}TD6g05nY(vVK-#~e1%W+mYd~CPF$JaE1A^B#E zye1-}{e6`mk{{iMRu=#7x0-+RR@N$wLk^l(+v{<1%{Idd-b?$omaoMO4PP!U$p1yClN!E^&#&; zw(LVG2G~*nhsryI5`U!F%YA^nD*AU4m(Jm}%ksU=gYZC|F>Fi417VzB zTT&f~UXftRh#y{zI8OTU5Ld?gbSjc#Ss#d$&O`*HUs<>O!^?;30dkccJ4b;xnP>)h z=4+M}T#6--#wryiu1R_CExdm*9GTa*Y2Y$Hg=}TCLI>_TvG1Tk2Ti4wAECfgf&bbz z4Z6t{!J$j{c!lq>Ntaklde(Gk1Xu1BjdwMUtLslKGcD(~Kpf`D8BZj~ZL+=l3EUrQ z?#F;t_Y(>C?wsruu5|tnyMIKJ1N&1niquBtXcyrlT)qwM;OqKV&IlQXP&RIta)zT- zXIz;`@-Pb!-}!?uEm^V>L+h?n&;z9>58!xSQ^itH`Hvn^O}$YI9jZ>w^?{h>_3dvJ zV;Cse#8_^#S}yv;vpRXv2WNCyOxQ{!1w@I(<3TMXy_^O5Gr<4;i)DqjJYWvo`?OLd zytcP#T%$tK18xj@TiZ0Y20h@fgWird4T4X+&WX@bDc1X5A>F0rDDOsC-x=C(^WfrB zOZHFIzlp2jbb#$pY5VG0^zXas`^o0Y)#{&^JEqc{r;rTGv;tNl+U!oea5Z1;jf2R^o{M`O7PcPm4hEUwAbqJMj-(%$-4t|9?DCK?4+Nc)oorccYtXbd>1BDwOm zu{}hqrPD>GUOZV6SKe?>0`n_x;FGM&aUh*Y%7GNMTxkM6JJjROHVuke^njFfF%M`^ z)S?HZq(yJ1)%SqKIqu_hi|TA~LR+qf1DTZ4;|`mlFSD690X)S-BS3-^Z$Ls5Z`%^{ zcp?$!wnUsGKF*1#@M^?asj1bSUP?z>EjEFSz0TOhv@8DnHMHldpBDyyz>1x}$bGWv zj|Dqm#m-*|!LFj@zqxg$Iz7vrH-HtpJKZXQ-DPIi0#@v9b#~XNJ{gn!GVpa;sxtom zIwE$BFljcmX;4&hO$t~cvB{Bmw2e_* ztZ&oU&(@F;V5NigE&$lR+U#boRTw7xjGqWIwdkLBG=ifu!=wei$C57>muQ3z6En}P zFn_J749utQaLlJXbGOf8Z0=Qu#hpwVKU1L`QGkD@y%c&ho~c4POl_3QwihM8w($#f z#>gxzUVoH)OW9rOkJVfTc|(VFt+GD*$%^Gx^&r~?V8z+(wk}qj`4t0atkZHH48&Q! zkBf<=cK_a06IYehf19oT`)G-9DT?+|mCL0gxzck<@qgut%_ZpzXIQ~nE}ZpgZ5Pf6 z^4nIsa&)0V7GYb9{!yBQEUOi;68+Xh^p_SF&WIlV{YNiP8e~xCz$X{ZfDo4pXFJjt z&KNkVD8F!aUY{4vup!p`!kO604cLAzoMA|oOI|lKQysKt%5}T^ zT2V}(hk0B!M~JyyHtW*|$){8f%deU}m>4i*$nwi>#1LIH>k$T)4EABgVB@2D&8&xm zsG4g()5V4K%DO<9^d%x7-O3{7Z*{NMgY`e^Uzr)ePpc?Dg&oakeK#lqzt^Qm>%O1} z{ArhB9F?ciN>OPgsI*d4TA8QPN>TA&Fz{3wDJqQwl}3t6BSEE+qT-*@_f(oGD$N9y zW{OHPL8Y0Z;vez%RK`-6tH?HG#-%Ul+R5ZlQaARNenh=6#?$@+A7Urq0;&r$Zi8s1l* z#p`{wystjzr1v%IeT}B_<)3ETpkwuE8?&R__@G0H%U}R8OAm%O>@j2GitvU#R-L&G zd#s#u8}?ZJV?#2)2;?uu@1?`kmjnVaOD7QLXv^*&ST*zulsD|L8qZC&#}Q?2!_gE~ zUx7Qc@^1diWra^U?gHO%dMV1!EITybct$DAGuGSNE6rOA_{-PQ!6fi^D$0GeGFtSI zc=;=2pK9(KK&*u9EuA8g%755B$?TfIiro%(T8!P1W;YJ}--xt{>O&D=MDILfo-SATe+EkzFe_v$#(fVAkvOpEcZWUqinE61~qYaK{GgSAWN zWi;qVPAKv-Vfh|95_&GwOkkOIOb2p-9f)TkzOh4tZbUEcq|lq&|B8hF4GI4% z=<3H>&*Fb)n{uY}LP&$L6rn$Lf3ZgTQcHCc*fP-w@P6$yKiLM;x47SKMCdbhR!*0{ zwMjaI56t9RSe&xIP&O-6@;zdSEl)5ApCBRTg>r5v!@6$1|Kj&54f$FN zy$PgVEMG+99h=4>ZeEp?q zsL^%)D}vy!8I37m=&=s`bM~S2fb0BuJGmjnQ%wrF;(xgVnVUi|FRFBXEVmCX@*g;M+ z_Ao2WFX)Wuy)zy8I8}XP#z_c=X{S^k|Fqlez3dTTPoK!x$Zw~ zJKcTsd{(ygz#~<(hgszJ&xSMPC+p|d6#|a_QCIOy{NFkQi8jB{eVnp{Z7u_|TdZse zpR5#eH8Au<4G5uJEoMPB=_Nmj@pk^i2D{6Zu^gj-xRNSc(}{f*!d@g8MLCH9GJayW zzC(i^L=XS3vH_d`61JH7%OvopAGtpWq!+PU>$-fjRYP7s@>)CyF#3fR4JbIPxic^d z9fB3dn^NiAD^e)W z{fl8f6_*J4ZV&mheyQjFu_f+j7e1J^r60-tlB2qwoli#-4~{%J-;+<;10zXMJsdXp zpej!2D>5nGWS-&@TB|dW{8bi!6K!(TfpC$!2SQ5<&3`v=t-AQ6xf%t2Mn(C_+-OGY z8$l7crAv|4zXe6$wk|~%sD4WSvStJ6ygaX@MgO7*r1Mgd7X6DNuu8?>vs~+_G*VO= ziL5kIR2rG5(nwM9_bfe?W{OHPL8Y0Z(o9fkrl|N^0-nlvipqF`%6N*(c!J7!ii*EU z2bI6GBAWnG9eD{%7?rOEMIb{`D$@FHPy|+M#b33iKitZ}3y9CMhq&G`BsFrO09ZBj z!F$6VBRzMIJ*stX!yYqkZo|=3jQ%bRIUju8vcjia7y>R-QJ&S&jMm;k5x8%cBCUgi zBJl7o#W)5}rIn)6N>FK~sI)RqrIn)M*CkJ-k)qN_P-G!j%ADJp(l@>H5BD$N9y zW{OHPL8Y0Z;@2flWjsY?JV9kVMP)oeWjsa2Pfe)&DXbzu3bkAeXzgXyI02+ON<~@+ z1Vv!AZ2h{_cMoyBV@PUbT>?TTr{;tAhCN1l?jCzo>)eJtX58F{qp2AExe2P8em<>`@Ec}l{U z$bn^8T9vb^$_EvxeyJK4VSL|vo%%ZYb7oDlF_yB)yqG{m2o>wkW<*3A=;_Jy~*aG4}Ze0SazWn|LU(YvR z4Pe#K?=~>}3o~p2=~+(hfK^|9yMQmW<@g4~P)?QoEqUz7WcgEhv~W@WR30rH_I%m= z5)FuyN)$kR^qt*Ve0|^YHU<32ndPXUyg-2^O3NPSuO*DD3!Y>37kX$F8@|YzvyILBifJ;nN2mZ!H zQ@{bukkrbC&MfKbSZ?aTbtamM(^Ytj6IKX)%BD!!J1vV-KvI{;V&N^$V;Foj_kXqf^rg4a_k|JE)p%IpbQ6Lk)Fs5;5)wmY@`>KZYjSw1; za3sH*Ex&|QFrfkWqFWkpFEnW!9=L36(}2BX0SNcPn8uc>aTJFUWbv2B!}iq0{&o5{ z->`viumiyuaFdDZz|-tNItF~BiR!>T>@YCKdbYxjnq$ClAgu#;we@~1J}Tr~IjC}Y zkdT0vt7WbR=8z$3q?>fgZUp!#O`>R#S889h=FaL0vidibtb#eF%3TW}d6W0t0I#*U*k z{LxrMyV^(jw=wt4-%WRXBP4piZ_~IW2ybmOaPPFStOMU$)uXW?=m9x#C^}()zo=tg2x*+9^RA%Hcrt{A zkOr$5X^JOXB~6L(zO|hg+~2z+JI5+UaF0Bj>*g0f!kYhF;j=ZNuM3FH8_;NG*8~iAcU)|%WFF2s}{58y9k7Grm$^<@?pWErX-6~Lcye?}IC{A{I9dNze{>QVojk}N zos0=RNb`8ze30gG#t4*PWM2ruUs}dmEc!Rt4y*MH=zS|C9^3$C(h(ze(d2J?atyu8 zSk-|XR%Nr7hh83)G7fB~*zT|nWFymU5RcO*wI|%mT*K$M`D_C5J^M0_77U^_rNY1K z=+ukamy}}n!iDrJ`f0xI%F}NdZSvM_g_Lj~U(II8CSTO!6m&a@vk}F|Dp?-XXbop^ zTAtOt)b>b=6Z>}nU!()IDhwE~G6Y@gWhkLphu~`Uh?~TU7OV959q4P-6Q_>l??B6Y zJ`%qJjmv8+o(7P!Ai194e+PPBy%KU*xFP~PN<|CGx4i5(kRdrLzv;mo~Is8JGf_Dw1FGo|n-$E$Bglt`b8+XyNSF-PXN|sKZ*! z9-I9_VkPxfqg=P(>VNU;_TJSf2X2e0TZt}#$MHEvO5oGVM8ts{y^~RwNq}$?m<7sRT zdT=<M;9c$QN)o9iT;WnlEc!aTUmaoAyBUb+Gwt0qJIMF}D*v4lR22 z6!$-yZcJ>ys^1F7S&t4eIax8_@OPGj7LYFG1#_2YG)TwLqjK}+{>Y4Q`zr07)v3%vSUUvK26s-esB?(z&zP=vxfVeO8dif{3#0}4w zx-DrjfBT?UCtGY(-2sJ8zHTZrK&TDYI{w$qy+Qa+bA&CpOlJ1pU$B#j zI*{`Q*$an@BW|pI=zKQu`48r^3B>p8dttO-5bYH!bY`7uF}3auY0R?w)0l8Xp?ejf zHJruFGC_Gb;belXmeS=Pq zciQ1;1js`Sk}4TDc>|xDTibM9`m-jVJl_msd~dk%Ft7-oxiMDs~72CdB^~`yNc3T^@gpKp{ME%8!1Cib+Ph!ZJ-ML zu8C%VQx}$j$>RyY)hd!31DiTDj=jkAfaj>F(4%pS=}iIeFwt`0t{3BN2DqDw$~4mY zb5mRnJY7Z7?Zys`Z>_}Pa^PDoK{N$iZ=xCCW)ry`J&AmUWvvAyd-4_zAbsTgTNy37 z5k(+9mWs9LAD!Y>)(jv%bHhfWheoQ0-t~c7IVu3@K~j51n+9G*4@d}wUKBv>xtgx$ zTiWVC>ah4#Pa1SCyOzy^H0WG<*yflNnh7rcS&);I!%G7fAl(-)9-%?^(!;hIjS81_ z9U63BXwseFIw z27M$<(fPGi;)NEowDdxYiCQwG#YD9&Edr)%>4X;ZRj6!~U#E~xkuYsbeOjcuBuI;V zmsDtx{ZcnBOVYoLY_y+zw1PGjO6*ftL{}@C!>5+;G z7xzG77x%#b6KR;x!kx4Nt^=6~QiJ|!!c$cLv|tCU*!i~uV8`+$&)NXVvGlhgvDH`+ zuVA9LvBTo^50?5dAS+m*M}u*Zei;vzBM|{vhYBh*7#Yz6QhM@87BE!d3g9Ly-!Wh% zNB+slRhrW<`$mEEBE;6X!;kDk>9XSA{{-(_)&D!f*a0hctKC8Cy{dm>ume_jZMB8% z8jHIQtoZkD@5VpD%7S`sSH?CZiOC=)j_zk zO@q)RyFjv>69j**Y8x*$*J;LxMu-4k6aw36fxVy_MIyAFF0>+!L~!zWvJ$&05WKTZ zgNWpW8Aw3F5%}q~Y&%C)flbcuX{P0AT28#5aoJndi&rZ~y6CXb%;6d}qBPg}t4br+gnx{!{i8tEoH=?! zjn&9{%ij?h^@a}hFo$c@!q|+)gB*`deo4K&042Y9n#IG^{b@X62h2KR_#JjT-2HkF zT`*z{AJB^Fx3hSFxUL;%@R2xpF|tZ*tGG=|ZHU4hY}W zLxdcqA6>ij_5nQSy!hZOxHMnvO0!969%09xCXmsJ*Htd#_c-=E zt-649jc@f@zMa1_teA0x3zgJmhB~8tP_)eE@0PTh=(5Cu`f!%jYU#41a`v;OqzR;M z`ectf<%C&robIZ)E&`}ieVEOWtGF)vG>a?t=~0Ti>$i(&T+4m-7+3j&=#l#Wtj->} zWCvtQ^hu)EWEs;*(|zL$>7;(jqq($M8oRBKvS996Suo2)=1+*lbF$ko`#kLpWS8(|Zh+ zE(uKRhnflOnd#kwQ#}+_=UZaM7`Cg6EEW%yQoMRRR6p@@@|X~Cyp+{~w zR)}7lqaSJSgStbK9C@m-QXhT>EXR!a00tEBq+jb5!;M0}~fVx>f1 zu}Z41ztukAmCRwRl;|r~N%h4zaTHDV6)PqBid9m5HMAz!D$DKz$hnMs&uF!LlO+E( zNq!9@aXi(1GzJ_LgW(+8JPg?29&9~gm^fT&r`Rxg61KgLni&&{1-@^>7%&r`;RvTz zeq|g_8PY}sa+(P-{EH~IV!6aBf4nX>OFeRj9lgq}jJaIP71h|Yh^pj{$Kg&uRzn>w{je@IGrQ zc7^;kc0SPRWkpQF5lAG1H3=_{r4aUVMhJ3`1PJjWn%M3#Jx+67M@VoC;y@`P4wIkM z3p0syUg;dwarAS`K^@5D{^ECc@@?gkEbgQ7QA~QR>#Idy|7Cr(fb>~BluH8w&&z-K zryx*nXxp{nZh_o79vZ$#FTZwwq$>f$x!d#zxWpCteokL3i#0t-F;3vUnkBL!#|t2d zktZnic=4cu@6#*}aM*h5KZu$ZW2n=_qeU}u=vWb_9F}+5u5#AfO7o-s@^zW9->jdd z)@tm+4dprz>hdVdD(9;Z#`JN|!b- zK*-51cR@x2rt$!gf9lygYv2qD$)}~cwZ~Rl26j`2MtBOK4rEy^^f*M{q!XhM#<@@2 z2#xLJHs0J;qu$CLhvU9v@fxs8w1K4vd-X@0iM1N%^K7wfdP>b+j!9g3sE-ZP8vuF? zq{e!=ze7(Kg*R}Eii#s24QPm7%wd+0Z#VB_KpYmwM;cp$UQCo2erGD<5O|Lz)V<6T zD`KjIMr$lk+n@;iU6&%Qg_kZXQXxwma90(TUB`*?oLec*t;}<7r8u`T z&$*T2EWeGa5kORy24I!5fBe{UZlpLj5}X?;&W!};Mv8MI!MTy*?4N9nIm^nvz2ADEuFGRWDQ^6p>dB2rE-8Lt*+?Bh2y(y(d9o~W_DZK zG^X37uza+b)&~NWt!)}mDphI0O3DOSA?P39@dO(X#HA!)O`8TzA*C=haqCqq58aj3mR$40hAK*;1hjvKWdI$oj( zgiNVO3qGO=zM^m_#Cq$wC8F`7N&+v0;|W;(xJSF@0jgXEhP;V(OJ zujw@gz9H4N51V5EvE7274}8J0vKN?Y48 zvqG3~=Z}WC!-H6^@6ZT=&HyWf*LUI`5>HFwzhbwc9otE#8{BaeOUUF(#;k2ep;2J9 z2gj^1?p1`J*Ev3v$3C>*t7E+{=;z`f)VgF&PF8Rh#+yTqRUtYef z_(;|v;8`k~pTEp_h}KIj{4pw(@WnPxZ(>=yPeb|d5E76O%WQtL%Nr9G+>3vT3b_g zv$AOcD@4}X=VkG7MX67+4@>oR@$rY__EG~bWHVsw=+0OvK z8M?hC75*)*hjTUjb{GsGOTEy&u|s3Q<>iVl!kgMO7O7Bj0%Qd)^k}eom9NB;&PmJN zav*6PtX;amPUv52dUYVHc)^cGr2uY9W#Gn6d=w=EjC#I?fdiFmH(K2-*CgkP;Hyox zGy_8^*8^ARZ8Unv={6%*1XC8N(~^s$k{zfWWqG2bf!dK-Tw=K_j-E&vN=fE=8`l$U zTt|Q$VGBLd^Fob&cR6dzcPiz*)(kNTk@5gSGqONjNz;0lrpPAFSmH4ZiuXK=XXe8S ziAN?;+gE^whco1%?@8bHWLz5CwIi%$$TxP_bKt5F_Okq`<+}l_INNM{mWnffAcF>bC>dX1 z#o3yavo-Br%YeM6}KJW`}E%hvW|Z`SQ#(Z8CmHzKb1NS^Qu5A+?pYN~3&^ zAb*QGR`7mS_c-J^-$Q}= zJyyOefDo5^R6E*3dAeij)B@FXu)GWN(%KUF+FbP~!_P_nWH>g&n*U_D*yeBh>BAZ; zG9*74zE$4dysa%i86JC(%ks&e!p4>|I8+|XmpgHN+ha%Kj}v<)dg&#f?p?n(-ir#O zK+POpyDTsJ2N4XG4Bk@lKrr%t!0LGPB7rI6EQM1#;43 zx^X{m(tg(1rm6}2zKU{wZuP_tGV+B$U?olA${9pN56Heo{QHY(Z`01V7%z7T>mB+L z8-CfY1EC^oIu|0oLi;GV$;j;R-39vyE)S>M|3Tsa z>#+r-XSsh3#8K|v?-TL$9P`xxRt^1T4MV~&Usy_u&dcv>&>{)tn>lHbhf08hShSz#s5Z7kGLXxRXb0AFvSF-UwSi0)VT9viR` z;9Vvf1KzJ9dG@`N(IBM6<0G@gv);&$Kz_Z^8u3g=U>dq}*Xp03Ir=lpRUL?xoDTXo z2tGsg4-Ixe246-3SaG$(U7*2LX0c2Y3wDXeys&z|Sxo}5mP@kv?=8vumXHL0lSLU+ z{yRzoc~%GrxZFfdAb}0kRxJVJT=I2Qr{cMQFHj#S%OxOS#m-+{JK5QdChWGmq_)&< z%(&NqmHzyjtjnRml;LB7d9RkUdl(ewgUf{e4I7>*Abn#^n>6eqKKH#yu9hNzBnFkSE8rKE=^|rV2f}!_+Nyk3man}- zNErDo-||BVrm{>EPM0x(k=Cvc5&o-1T?Z1(qx0y>-XuopjSq-2W)nwyyGHvfoA*s1 zQS&;u%k{h2TvXO?WurzOc>5OWAyw5k8D*%GHr~#LSyvLfEaUp_17p#`$|xx7{1aUqFWR-}F;nZbsWmqRYypI#YNt z8G&^d3a#NRu9#z_i1Tm%W0$U&SUT)Vi1&Ztr)LQzJ{We(&O>Dt*V#k$6Bk~%L6nbJ zT!a1QJ}yRHfr| zA|%IiR+o>5Ekrz`qDAja%*00EX!ZR9{VSVeU^wWn8n6}nm$tw(fOwakmp@0u*>vy^ zJXJ;V_5FDn4LTJbv40WOqJJ+@7Z>PX$$X?yTdN<1&DJQ#xUkt8jjh%>t8s7TZmD8b zIa{3D@q?kmuBSYyhY=%RN_+JfNVzn^`{Nq5T+smjor>gz4J$JmoC1p;96~PLc`n{R zX5Je>97+iU@81r3c>kJ}x#Lb)VSHOy1x{{ni4|d1A||*Z;@DLotVREDMpRO$Kw>EZ z#2L{P!pK+cU=6ilA=ioHu{QVWz}Km0kvwT78@Xa5?!WkPr20C>e71meGso^ArZ9(k zOfO#dD`dNN5^*wOmU2o0lRYdMEg-!XZ}y=_Uu-elVN*Tk&W(te+nxI4M4y=hs|+rj!*E;<0OFwqF`-%KoGV?5{q-=w1Q8Z#}pC60hFd#2H70I@1A-^4?K z-7#j@1Xk?)MMZBnjvbklFBb#b7WUnYPx4?feQ43gMc(=P)N2k9syAj}@R5TzDjDat+QXh%SAhA&4 zoLuGWB=|n-a0W;xvI7Rb$oiY1zqgvG{vL%yn0UBjJB|$e{2>i?{(K83XcCEil1IL2c_fT-1mdF7DM|Mdyig={&l2;S;$!c;RCC zc0n%#nD0?BI9E<0k<)3H!vzV4*Jw805wZgOriq$Bk~1Jz4ZxK|`TGeUR39HQUkzZz z&R+`tnCky^ume`?wz`4&it0m7HbKA#^rL~=#WgVXEt3X#u~ty72BzSo7B;tOyrZH> z`N1tWj{iaJds^G#?DjSds$8y&0yFhr=*7;yti zV7%X^<^=F&6V-wAEYBGM=~qasO^}#~Nf6JVd7nB&A+97_j7ErNYHmmMBh?YyWT^ul zp=p-~yyYr{d@~$D=(J9!i8b&#HBVeV@Bxh->jE0XcL>OB&dyS+^YrgvZRNa-Mn{FR zV*}n;)uX{kiE$^Raf@ooSL?5m+Z6I$1GxYl=krQc!bo0qB3~^iZeYf$n=~8$Y*`rv z4z5{NXiK7iFL`Szk_z#AMqK^7xoQEiOFv0~;qKO91Gu+#m`_*m(*lW2Z5s5Dn%CZc z6aSb=K>k*ww;NB`-PDP5`a?~?k1Y|SKoTj-^8zqF)1Tb2ITO}=iZ;)VfPueFA_}yXT`=i_Lkf&;||Mw zxMRaq31D}0C@~$Jtv#m|NO3KKPb$*2v2XfVZhg{BLd3fQjhE zJVexs#pxE(2=D_YY60;r9d1ZP>8CSMRxH=2Ed8SfSdtVu#K+{1QHE2-)SYsmQt5xQ zo9+_MW;ZvmoKWZC%mK+6)@>6=q88LH(31%yH>k2Kx%vDo-)0gAbeSrx4W5DTGEoCq ziRq41O#W9N@c%F7Yy^0hiU#BhabP--ZIdr0cbx?1%*%W?0sfkGI1Sutq7}e9ti-3E zqbmfTGUbvrNuz@Z%HfQ+v^)wNc(P{%HD z2eR-<%utub0mQA8&6ZTz_=_Iv)d>d$*@^=xYH_wBaeSCyQlF%gO-y5|A64%w zZ1yY%GK*w)4y0o_*8o;H>`ZW24u>$&-D%1vl*6YjT}@ymUAL#wb-P>EamF~xVFg%m zwmNl;@~5w{vw_;Nw&d+^8EXQcZK4%G(j?a?ft8GHw~YPDOq*ovP#d+TFJ*Vnk08UG znDTRC$~$tWS>DkK99>`}R{#lWuy$D|qwzV_lXEZmzOHH{#Da#i8eP1FPuXX5!*A7z6mv63wMD`?)X ziQUPTiWwj&k!#^VBFIB*X0!+*zXTb@Q!4t9+`-#Z8;2SNB9F}h={a{lZaglsVi^OH zj#9BwC4sRc3?m0_U*i}S0v@CqRbV|)@rRp|9G zQczvva=H7td634}&8`6?Gjb<)^C0!bB3%e+9I5kCjxZbLY|FohlU^>d zMQ8+gg^J`woU*&5^+Q{IM<{nPl(-f1EhYai7DOFL5ONn6NGLM%x3p>OYD-H4NOThF zmO&bS-yQ0fK?J|l4+^g|n2nm4ErSRMrqGU8EyB4|!wILiNVZv^O(4+)WkQ2`_yh^# zCTGd{xTVjMf24i=#QLlQcd_AY0xN;}`*FBp8OhIC!LK)8GeC+#&PIVZTV2c$9x2V8 zfuCSU+Z8|-K3N!na4jsOL70u|vy8+v@RcT70Sr{9`=dJTsjhf1RH=acOqL!Q>0Ab8 z)aC3pA9Z(jOXwAUPfYlGVh?}rj5qe@&Ujo9A8l%33d|EkH~s{RAP4pYSuE+2R6Om8Iib+B9GzcOQU{vURly++8PUVM@(cc=&5I@HkOt znOJUa(_p^Ii7{~h(8K0-JcSD5c({f&(!*>OssmY|#O}rp4Ne{893IGtgXpd6&|u>x zdL-#rwp2_2N!DQPk_A?gkGIU#fh=zY6&l1RT>vY|^ly(OR7Nsi{lLT`m#%?CB7JVK zuplgYz^zI*cL)BZ$(CkdXjhkOc#AtAA*9|Wc~DfA`PY-75!@e{d0$7m+ipaDzYY9y z;FE2jRsczgTyX(b^0TG`i?3+l(Tg7ulM9(AKzK(MhzUhH29lw~JyELkO%~MDXB9$M zW{daju0D82D{#~b*#VN0;_3s9>s3#-K=P_Lx%z-0Fj?W5xT_DI$rwz6qcQKuU1Vg4 zP?fVApV8@feTc1OHf-#~c7-suWC6a$rgIC()=_tdw-3g}kl5!Rq;8g)yHViHDiRty z+Oexb7<0qf8gn)d#9r)fH)jt7Js@c;^b!g2Co^<9W6s=}3MxmMN&{HwZ+&V2{GBbF zVcFk+KlG$xId0_Lv;73qRF&l5p!5n=3g2s4DHrpR;-Y-<*#+UnXSq^5-ne|Jx(qKq z6K#ZeOkAFh3pY`ptJU=<^l!e*gMXRbgqR(hX>*GNY zcvF`mt#1ZJ;2m9xw0;m2f&bB^7$?|sZlyT45}aEp&aKRIZlyT4GS9h{;_TlE;W;-_ zoEr(wjTGlbf^#Fqxsl-9NOAVBqwt)YDbCFV=Vpp?Gr_r;;@nJdZl*ZjrqI0z>t^a2L8|#5ov;ui2ST_2dkzgbw>Z@{XGOTNv*f) zGh|QU7|S06oO*XyY$xY;oWvy_^T)Jr0j^8}`@>%xySI=R{cUkYYnVtt@nNtMpi>dl@pxQtuA){r5Q7 z_HrJG_t;s?ZP;TQHMij?ofBa?$J?~Vmb`OW;Zq)F0bXjN5#ZZQGzz>%MWuUMA2mhy zL>N6DZz?U|%SV;U~viV3lgq zsJ_o^o4}A&caZ|BN0`+(5Povm2Y71G0=~&aQ$YC1r8!`g>bOyTui1_R166ln398RG zs|g_d@y6^FNK zc`LW#V-*~`VLwDCAS$W!phHgjOAl7$q`&lFMRfh8$6NK29;}Ex@sI-_onz~i-@Kx8 zs_IC3fS4t!2dmeZRTCJjT=ihZaxCcq1}j%R-paNg!OB$+Rzby8&!mz^WuWHMgAO_A zFFjb16R92`oyiU$TGVS;k(2(?(0fd=@KN~>_rI|cY60m+o}mNcD0lBKNaE|sMzsN~8v3h(7}AXl6!70H#u*?{ zbW?nvp9W;*>b`%L`CfH$e{crGUYu>K(Wp4{UwXqC%H`X6 zX)&nf8+U0DzP)i50X>!Kl`$@0+?Cvrq@Q^CVrKF6G%KPp z;6Wy808hSlSrMI_bO4(sY5?0RDjuYy@o0;B5_pk`#(*C-(ImOK?cMY_2E4*1*d&nA zkX#fDEC0jxY!*^$0x95kOf&}1CF_jm%N2e*i2C5ULreUZq11syCsX8_P9?Mg zKA`$F8?QPrR7eY0adnGx^(b|*Ft`F%?EJmM$Eg0E!46ol+u4q(534>UCs&ezFVv6Z z=CV-4(p+_T3&I#bnW@__whlwVD`&pTe0B=>1 zbg;HV?Vf8jF#&v!iV7k$z8Lg?S6JD%fY+<2=pxQ8sXW5cRtHkFg&hq-ktGI5D1{!4 zO(9g^Ehd@(61|Km@C_l*wH=~*rA0LX90^g~)S*Gha>5E+Z>A%_51FVjS32@ruH=IX z6S6=&N$8v0L}5W>R>`g=N{PN=l~iAgbw*N^$-ZKxL|?H=sxNZKcqjXcl@fi$DyhEKy7kF@ z-~F3T#Z9$XMZPNFUqFgdW!Dr}4=Rgnm9P85L%o+33sic?j_=U@NH0fs+oCcC+@>OV z_4&$-#-ohp7&PAyM7_LkHSZ%p92PINq_I8d#jc1U5EzBPj1h2;TVX{^vX=(ps0a^N z#8L=*Q6L1lQUio|@)ey|>O^9(lOysH4V0x!L~(B%Y_ZCZ?Xg+$ByD7f9Y@W?=>sbN zp~U2DIrc1~$ZriR2$eM#8>TCf6g;>dv6>|mui{D^mFq3qG2jLj6~j$~_#|<0aEV;v zeZ6@f0phSI3K|=OUYuaU{D!HtAaI~f47XNcMNGmGNF+sgxT55eC5|&fkP-(%JXzxG zqYu<$pwu&vow6+H8#=64m0g8@mB~Ts1k1e2W5ZPC$98gjh|Lm*46NQ{)W_n1K?Oo( zjmN&zm5=Q)wmVq0aiCWRFAN-j(30WuFCP5|-9Fu^e-q(t(#7$cVUYyhp`!9qAFY24 ziok#CQlxcnPz2uBrAX_yK@s@7F2y( zDvH1==ftH;r2jAcEe$FCEBP8t^t16^jXt&ihOc_)`@XdNdBd z4!vn$(?lzPZ!^&paIK2Ui9+kErnmxluZoK8D~-dir`r|45fv4BG+y`t(*q8F5YaU7 z4<=dxJUA=0#M><$8gEdcBoBDGiKc+>G0_U(-J>o7KyKiZUs#C<2qjtZf%KTT^ZyF%reWRHf#;ZL6iA)Q&+jb;Q>F59%*9~1 z$Q0V{W_WHKGbWDQ60#VBw8)qi;*Rv1-vLE4EPokO_H#uyx)4fQQS~u{+YoJ zSh4f(scftM-81yZ2s%Oik*K(;=QVWW3U5Ocx|;M z<{XnsY@kM>{#q0x}PZPG~SGa$7L7qfn$l0&~Uq-*q}Jx7qlN0y#Y7 z+f=vP%6GXnmi1R^@{3;M?B~7unIh9arSe?IQfNQoc&yp z(oc&yp>SlJyXvQm0(QiB;C+ZT+k}S8BJ9^Y&-zXxo z_YqrUC&&|9F3|!ubdfDOq!Er@69jdt1|_D%M`$9VoPHRz2#&}^D~{)-E}l{1d9NLy zM}dU6PyVZ;@-vAYy{USM{?*?*BK%KU_@jh>tCor68%P@VR;SnAAwOptc|?coJfW&9 z%g4I;+BFKKZ#Jjom1GJhAbFk5JWlv*y75TNraRXVEs_@+mp`a=q1?-kN2~kOc;F70 zb>Q$joH2qFaJScZFuOmE$92EzSkXl}T(pL>c!0Z%Dg#`=GWy|+!CtNtsLKf7aZf0p zW?P#X;A?FncIIU?E(&@S3=^?C?zndPViOlpUq*2?^<@;7)hTY~FQ=B6=JMnrg}|Am zh=BiDJv7j`&*t}X)@Sk}rZI=tIu7G-c$cj{<3MP|tIwtB)#u~3`iuji-ZlLqb99KP z=Td_Lvtc%;=sezvX`J*>RI{0*bGohV<3O@In>jjf4(8O|4}yrQdYkR>#^)NZMKzL_ zzY8NCy6#WoQH+-&JeBj6R#=J{!)~ug8MUsdfRorI%8X!_QLn`YVJFbZCPvhd~i)%dCXSvBa%?Pj59i=>U4BTd$%y!!j zA8hYLk8)?@B_awgY)=wb!lz?emnp5A^snp$fo#flt+i!~SNWf!d2npq{K8rOi| z`QIdLkul?ATjaaww6FGyY&ZQZf18cv0wE(>HH?S0`Wa{VocB~eiwFBk9prU3*e&2+ zt0+Di@Z;X)q|Z>JM!MF?NEaRI=&9A6it*hr^4iHq{U$f{bR zUF_m=b%vGV?J}g0Ice)76R1+>{e^}$v*MsAw5~bQMKPW)eNc?2(bKBq+qW&yKwGT8_NjA5fjI9xGJ!uv^x4*eS9SJ%q$! z%Chv(5`;C2+6hsDdM-7I4~DaNp?e5n#S7nY)sjneS`K{TgW)V*3?BlUcyW9fY~ls; zA+XscT0oDi<|e&>#?s*|UQr(ck9dWB2rl9k_aV55S77eK{iW^r8+Yl63nwUYss6co zT1&x;h^nPtM0Bn0P8l>2aTks~q=@1mL==#whL48Op-;9 zJRhwUE+XJE6&0p5Lgp^%WP}<{&fHB-@prTO3HcS%IKPMMrJSbzl|k@XiKZe6p0}vUuv}GNuECp3CB~#y=NM zrOfS!9(EW4h-iBW3lKdU&lr&?cqN5FGL%zybtJ|n>N zPkc!qD}u})7K+VsS^V#E%@@7p2TgD1e+eGDtyYY6Ag<&N;VS2=5cVSR2Xp35qUaCj z<)hOv(^A^Y{nh5)otopmV=5y+NXhEszhUR?8radBY-zmRe`-I3)9)gW9i$Sc9kEL4 zlP+xcI>JU#)G%F zqu`!jOl-$uE7^|aPg}_`FYYva(yQmn^ToYX2uoJ!*neW{5tA$Tez9SCM~H18>k{}s%EL_V82 zI%KCOO)_?cZQzVzU$5# zisinG)^LV0$#tJ%({8R!`J1Ko?{;IICi^?hqN!3@z{15^$Lv(zbkBP z23C=4CyrAVo((L%4MX4DU!=#_qi;N6aNG!`=fc*0l^hQB0Sm#r}CX822*x2 zK-kIFA|_Y(gWb(le>6fyCVDacSDmu5GnyCc9HL7p5B5sU0C}QT24F*n1`li}o)Uel zx?%uiCqP=rNMdtvbz~LxC$Vw(Bn|=}m9IhdQVHZHV*iPq9geXo_b9PpdgtQP(NB+9 z%@T@Naa$r(uD1i_ByfX@itPan;*-5a99&{cg!lF4eGG`hViQ4QW6+C-g)qNiDoqI7 z9FCs7tcXcC0*Ryu4_B05a!UjSLXa&H5aP)#(QlPsn7uPh)G#M!V(PEeGm)$HuPg&V zzUn7+F24_Xd2NYXf;Q^{@O@h?M}c>%sC*{_t$Twa@V+iZTE7j7z~6N#(ptD-SrL-# z&VakBsO&lp*mG{BIJYv-xs~GF$~@;*igPRToLec*{^cc}b0fvMk>K1&ah7+IXfh!x zcgumWEZ^Ed3!ipq+h zHNMdlfv1^h6nM6Yrs7h9#H)=&3kVsRyg>TMNwhLrbR&vDdMp)tb-KnnZ2+r=en4RR zTr+F}zhWty29l(lL^GpB4|0n#=5>O){hIlj!Rr4CBH)*AB=;l0V=U1#K$0Yj5->1u zOBvCSa5;Cyiqs}fXJ4+oPqwtwfjBB&r$GakvJ2VKrh)%LlLk}^%{W~n+TB>_7D!LU zOEPHClQ`Shp+QfDCXGtB{(aBna@{6Z&$GI%-R!rkQQ`cQd1QS+vfCh?OMp_w6XW+G|w*b-@l8O1K3ivEpizTR^1x z$#SJj{~m+0wX5?(m%xhM+D=UMGSzPfJ7C3by9@Ce)n`h`X*Te0^`rc4?cm4XK;9>S zjI@Nkp+lo)qc{O%fD1hu*9ARBn**4foB{W;IWq(Nwu;2tnhq1`S}T+Z;3gFnz0zQo zNf$t>NESEX6KoDn0H14fP-3O=#h?fLanQT9LxZ^}y#nht;Uo&`hw)0Boia4nhfxNS zz9Jzs{+Fe<0VIKi9*tW=f`Q*O(FBm3ON{{E9@4v}LoV0a`;%mA5SMJMqDiBp%~1|)Y>w*&@c5X9F_ZnAN(__hs01IS`9B)5Do zkP|uv6>f=#W;vvM)#g@%6_4dktUjojCQ~AMs^*0}iHL&*SrQRF6OoAG+&u9^<*dF? z|H=!LzN3&?E+QavTvD?>G2vGvCj9yi70Hw?j0uy;DxCnCs)Zg6rkCgenOvf`Au-cd z_~=)}(LYT?eV#?%^wGPQ0UxB=6J|B;t;QKZ!b%)OaYXm|7aPmx7A~$04QH{_=;N*F|r$okK<2G7#Ji2#`rt1VO0A5h3)K+bk35GIU>{Mfm3 zqS&{T!NyMPy%5I!3t`+tA&l#TKEmvnq=&&9%84WU7vLvVRO}WxFj46Fj!XxNycT*i z7){Y*o|9k^G0$!DSDQ4btKlyBoE%!ZB#ZZggSC~NjBwf7iBBM_L!lRE;LnsZ+>>dF zmBjg6eo`-|CuI-l@h90=td!`>tGH$F(^`d0v1C`VQlcxb;<_Rll9cQ!R!Vf`Ra{pe z*TG`ZTqs8(*c@PYzNXxT!hW1RbOEnN3c`3nD5r_p(yFY-0(~T3nRZ z;=1|TcU+JQ#YR7(EKgU%A^9k$MVs2Jn8cNJpYAUnu&qaf+46p9`Qu(}wYz*vv9k2W z9-@!qh!RUL3FUB{-(@0NqDtL)!j2Rd=X}s%&OewF!Et(Jz8A(cppsa;VRFAFR;KK{ zIQawm8M=vkK7Z7rL}jpcsed>VrOL@&+!ZRXv~h0w-K0BKV@2=jOC*Hx;Cy2>3u&+7 z9{xgQiN!Sr+*d`#EeRS#BIomQig5Krb2S3QTJABf%xD}G^kP@Ubdssm$#xb*y{rg8 zI0B)ayuCtiu$60)KsXk}zRD87WcjDoza;#_D!U0}ZW#Io8kM(3=k?qzR zz-Oo^KWtwW9XsW$f+Oz!I@d)6WaE%J;2(O!vSLI@rksSK&bCNI@t||kgjIez7n`NN z9)TSvF^LI;%A1v#EN8Lj#0rNMgvvE0HcYQPq~O8zh}A5iconzupz=bCb`1Cu6&16U z2JuPa;@}dw#QQSyJ_5vHF{f$NgI+EAN0{fC$|wXzt>WCbRB0OACa>*46 zXM~VH(V=W0o?M~sQKi%h#g&wNg<_k)IZ$eAfG^InQxDB)cc{n(yjuM4$PX_oM2`d| ze4V5|B`rYQOFG-OD2HnRBJ4%%dFm|na4(mw_SSO*Ss#F;Q#RP^J8?b>Vcg+y>IBPv z2Liiy8;P-Au3x4q`L|8tf4B+V%R;6b;ZO(u@$}eE4ok6F;`|}D(p4(zFTF}(OEiNu z`5_k?D=Z6csm88pDq}!$C*|RvAjgg_y86R|3_TDE@`_dO>;^5TFItVdv(s}l3vbiE zGP8i!sHnWXP3wI@5qMpfBCWp;ion0=Ql#~npa}eImttH!o^vb3xs`d&trX{0<~g@g zoLiaa+)8ow&suoSjTGlbf^#Fq`Tu6`Ou*zSs>R=#Ookx|AwYl-0)!-JWX%lfL);gyY|H1c>d+OBL&#C3sy?v*Wu(^`5xstHClCn8|v>)0m&x3cA;+U4> z=#}F*MI+=o&GG(MEbs}I?FHaZ6eUks&jH^0;fyT(`ogbN*uhFAKc$(i;K@0vtl$MX zYPEtJbJXPuekVtT6?nI*n_?ursC!C|Lfxr33UzC96zV>kqZ+AOY@u2TNO)w42b^w# zmI7j&EQ+&$PsqstG1!p-FVD#Ual(-SUz(Ev;+P`?{z^^;xK&Y(tdZmG##YAs?i_`w z%W@QEy)H*#!qqt{R7d?0T-nUONJCC=6Auz%u4OS-^jZmAPq&=*0KQOB`S+#V8|Yv$ zZi*Zjt#8jxZd^Y)SLXr&UzZdJ_T4!H;QNvcU_Y2M0DdgV0QPG+1K>?b2Czgs^f62r z7NIEd3pz-!R1T~F;i`{wBOP$H$uS);5)!vYblqgWM1rI_CL#$5qf(3ngsZucp#W!OD50s5x2cf_+Z6Z@T_x|0)rAEJ3A1`)AcZ7tr4^x+ z66}SwKBR$!#Ui08BuQ!N*U%NV5TPbWSc4LpLXwmw*b6(8P!l99Z3#^wNlFv!O*T9% z(&MjO;Ej;m1i;mX>IHn0qOv;&T2J^dc%$9)*GP^QDT0c~*Eck$5`bd$G);AO@gn&hx<8Ad%w1k$MDFc95+38;96|-lX-H1#a zH(jN2b~TH8W_&cW^bozuR8tE6gA0E}@OJg&Qsatl?qjarm@7)}HELn5Yhlb4y=Vs) ze#t4*_EPAXM+UByJ{{>BjP*soBcdK|x4cvU3ntu>GU1lcf)c`Vm-}K9qVIFTHP=#E z27IET41hIvQ8zbby~L($UDREo1FYEAKMku38`76dF&lqA3HORHUP^ zBBujfWvHcqZ!WUZ$m=q?nIv$z1N}IVb4BN*Ay-@J=vv6;aA9^S2XZ6lF@Vo5hLr}n z5jwy(73pY@8=(WlAs=gOBN0Le$hh!y40{qGbb!l>fu%trgbt8I$Qvnu1k$SrL(-~g z|3H=U7wJlb<1bdViw#uIg=}?bg<}mpZ*!k*SDp(CyrJifL3fX`8sU&ImhA&sd=eoq@W7m@9{k({X48EO#lb%rVfzR6IF06%P~GT?t1Y7j8Dvy7deR;OiJ2K<7d1_5(#qE!)Za@iGR z=UmadR3VpJ0YkRbsz^3-ZA-}|X_T72Sv??$f+CN(O;MR6n`B5x0AG^F5^%Yq{7OZG z3<)dXTa0xP;CqU!G)SAU0$y#bgMj$$wcT50wj)} zl}OI8TdgwS0ft%xNW493G##lbREq*4hU0V1@j<{N z6(wZR^7=4x9)>U!<*W+pr%nER8U^%Ot*Mk*4`Tp?W`&}hS>a%EW-*#hZ(`QNDD%*) zP?R%^(Ze`#W&wXwOD`3zhcV_uvqDkMtZ@EllTXAjfvj&5tp!?lE3G8^0Y7Z0Uciql zDmz7?wfofzyovI@L1-%Ct0t=t@I8w1Z+Or^RQhNIX-iC61+d5%pKrkUGGpumJj2Xb z(O#p0IdTGllS@8^lLLU~8>$TWMn#EJD{3^bEpbqCl4{Qt4gsP#KX?IihUpywL`6;< zX5tjpbXw8hkmw!S`A)sc3=4=^@)x|Z;})ZeelULf<}6yj)~lPU6~xrUp91et$M)81 zyL?OSyc(?o&Fy}`r`8eDdVvx40iraM7`lCuGLTrYHYCBiPjMt0Sa!l@ay)=!PFACS!b zPv!9=Y6AKbd1Or1H05oj6KlYqwYrJ2^fv9fM6VM{7pz&+S)+NWaaI5ks+3`NKCB$M z&S>KwlB9mz4b2u6Cy`!vUX4~RydSVI5QH$aYO2oUN9$GgF@Wb7stkCgq51*07^)2T z2SfD(PP%4+*DUz~e7m9g0XHa0-WJ#>Rz#{iE| zlvhJz;Oj+hT7!!9q%=NRM_LO0BckV+e|>;vd}#?_+b1)%^3(+I%jVh=K%B`INm{ea zwLZXi7-|XN)r!h^X}wZ?keLRs=vMsh5NDFI%c&(#~NG#LGf z$;+!K1^+$lG!SS}2uPSk;XSKq-2XXZ)bk3B;R*}yARy_G2cm!^!%L&_S7Tj55|&vK zdfu!c$q;FPq{2(1K{A96kW_d&8YH9Tyxg<8wY=Q3I!H^YS0}O;<`WuVVM5z#TLBiF z4*7dz?r#**W{TKF@cw}{2iB5<6~?|`K^Xm9U?TqaRKxh^Tb-{0+L0E{#0PHC>fm!Ocos-FeYmeP7Q)(_LTQExvxOI>G`Iv~XXbTY$u_rx>`V{NAuJvHMgX%INXhCu-zdVjroeS>*l z2E;efvm@o`j#9$W9i;}2o|15MM=9axMym{b^9Ile96hChqo;(9?yFTv{H3lp*>+=x zmF#j?v2(_!N_LZF02{mv8fRO{A|qbq=?RVYg5r{eypb<#WaLB^@JXLu;Hl*WY`~uy zY6y^hzEH8hzt{Hn%K*Q69kP}HGH8VnkS&lH0m#lrr~uih2o>N)9mzrkxXDmU0NGm# zRiiLaaZ)GS+Gvnx`8tlSuR|*U?p8M4dS!4irlJ*O_q-f@H<7zZsWoP64fNw0)(LhyaY5B!zHM=QW zx3RAnSEKQCQ_};ux=2T(pk{4a%_dy{uhpvyH!zE&1nsWXH1f5u=W)8=ankH~fjLB$ zw@p|I^CfQJySjh;vtHXuU9~y7{TZ-rP(R=zLzMy940+pV+@(7s`3o9%Wqvd8gsG?vm-PX_LLJ|=nvyMSY3^Db zmh4fQu5R1ElmW?}DBZc5n19vw9D=05ZeBJRfF#5-U#k)NN**adf+KV{R>M7_N3v@h z)v@i7_{0|8s|kJkn^y93KefPjxv~QOxZSZV1^lj}#+RzYH5!z)+*AM(7guY*d6h1W zt8{4`r=@LLuq*5aXer=36y@!p!QGMALCt^Mig_vEO^WiYG(M}vEUbVx+wFk3N@K3o z=TgAqYzHrNG$=BW$5eTR&d-9P>X;}673Srs8tA864$6RcTMkBctR_ae@4e4d2T!>^ zOR3BXH+(4rUvF;qUaa6%dUc5hW{Asu3imp%wzBMfmfrTc#Ysv7D0t;))KYBW`vYJ^0NQE^owoWZ4 zp#m%vTR48ig8OwyF??j&IyopK3D>47HT4NCBtfnlrmC2sJ#rtd7UpV+4FG zQYbRM0SgJ;xtfHMFk#-gnk2kW=N2!KMjBXa>8=BXn0O!xNC*pyYc)mB#DzMHjgl+$i@FBWvZK+(P!%^}S#VyXXy7 z8tfD5S;ASv+rYd*?bWk@Wvk{{LU#{SH=kkdE&*gOEn^Mv@mj$$7Xbc7Q6ha!mBy7e zB@MAT{$`#Se0jfZD~AB{NilJJALR<;pJ>Jp0De#{5VP)DO@mzRo}Gx_v%2GP*|Qo7 zio`?&bdV%3mIfJ%xN!8n>Zs?UfvXW$nAM0Nk=Xdo?>vpkWfq%gBSltUY9y756#RaU z3M(e&>iKu2{*`g{jKwwCKAP_SXfy(@Q~~w9V6V|$_a~*#7B1QtThA3*uv5A38{5c} zl|=Hjsrp52+^E;oIXE$avEC9Sv}R{>S)o@UwKti-a97}w6eD%#k&VPX--1*I#5dvF zSVhCO9j>#6~tYPjwiFw?Qi4pR=Zy~D9QF;*Rp zfyR=Z$}A6Sc>hs8cF%K}%WOtl_%Q{UCK8{-k4YAXaX=T#o$NBvm9KR6?qsgMCzc(J zKV;u$d%mHVE1F>8!h6a6_SM*g0Uj9&^T^P_Zm>D46~c$*nk{$vnMM;3~@Rmrie zt@O-VXqdx=eE7e&?J_DM$(<@1(!|z#x^mCc>x9zDXUe{WCPvLGoqY27r9J0$l=kYJ zb=0)d{C)OZ@cg+m=IvVQn!C@Oar+$g`v>-V=ggBIJF`Z9iD~Pcq4ks0By5mFyHI%I zLE#evuJNVLxsg#G8R~nUa-N}A+5d$k@!o7^`SCl;NIj2S=k4aDHTOyK3-^HYt0TC!~iRIFd%IT2)3R{~Z z+wnZNX$UR~Au;2@g_{MB5#&7%MR{bHbcErR++hIXi+2!Vc;k_wu(j7O@Tg1dhIRn( zy@u)sEZpN>Rpo~EOgpoZlRp}qKgf+dn}__YV1Q!*_Tq`#eGDt8+_4&!NSD&yVFwh* z_sCF#M~2Sdpw8c>SBYdu@;>dl)<`PfR`76(Pc&!8Ts_;AYp!wi03v9n{Bb6E)o7jk zf05h=^LLRBpEs_w&tA1s@L#B-e%Fk;KdIXKUa)`DUhWBKV_7{HSnNHdnY~y)J2L!P z8iD8JkqBHX;oAq)KFQvu>aI#ZRj;WHXUJIZ01|Mdz8CDnbg#JyEqO{B3bntYbZpdUN|^%|0=%J$+73%h8L(i2iE-&=G= zB?I59^rYGS$-S!klY1a8$d8}_(P6YG`n;aGNf-dpZM0(hnkVGb3SwEOe8NP2!*%k^+SfD_1TZoG zhDzMs3~WDCc83H0nI$*cVj+qG*-PwLKTeLYy5}z6+14@JzlTES8`S81wbCL&2yl3c z_*4`W)&X=xA$5Bd)5Jp|q>=M2_ZrQPwvIXN$v0`Pi-H_2S24+8Xg=}%ZE!m{(M|dja_&ro8L2wc4>-?ovbJuG6#mui2xfe!#CQ%DtP|-vVP_iZT8@eyp*Q zVvK(jzg6w|VQwnmkD8hqGxntzV^d?sN{TTyb-UU#{u>K?z~x~e;37kn0bggRe!z`} zDg*BFO_K$M&EdMybnlT|i3xe|Ob4ap2H= z*a9A5s4`&gW-s7^?`1@(D^Zwr;U^K>Q49- zq)a>=4JH!zivS(46v#uKQ@nx}rj#bZC02#%xWt0H4DglTwMcHP9;RWv_xnU>*()@H zEArq1e#B7A0DotwrTc4?39@xtPr%0NBN^BQw`Q&oO9|}8Dh&o-Vy-?^ zXObiH83nK~qwKI5g^9dFzH#lo=0KY)>>_yoH3twlnSbB_4AYL7x+9u{kWYm%>v&=s z%s3F!X43J*G*~borcID?1HlkB^Qjd4_e2falk(sJ7J|2Sm%z`K1OaMc)cGd7n z7bwD2{5}D=LI~Fm(;#HI7BIw3;<9#_20_co!4Nr5PJ^)JaC=BpOw4=!*Lp2`Rf{x3q3g_y!EdLpJLi=S_PW!3vIeEga z%dDKP*Nt0F{sYDQDC$Z0a@lZPXpMe;-lvg#VLto-DIEW!avD^Q{8IJK5)DenQ_`Sz zJS7bhrineBs8KwgmWppIugQQfe&+knnvR_7L_s}K8x}l zsQ~OKB?7Rc)F1$-Bm%Ib)F1$-Bm%IblnB74;RXRXr9l8r2?KDkI!?YMs#n?i^(-rX zej$z*WXzM(c%cHpp+m^-o486*E2H-h{ssNsK zOGYhu-cqIUiJxRh`C+#F7~LI5&92?8xLtml^@&)(3l-&M(AciXq6~0%u3fPZMXt{U(!4A)%){;lDTuATbr%&K$rDp~>GWT^7h3VuefPAl*? z4A)fy-e$P&8t|=#JG$2OolNTjy^2=AlMPj2`dVze@%c4iCVzQ7W2){mkI<_;nqo`A zoagLiRa<&%rrYHK_|=B%s-3IwTElhMfZt`fuG;$*zS?lzHQ+BAE`BQX3Uv_g{FuN{ zjP%G-@E@G)2;?nzw{AB0S@y?N&kcI@Kg*`^9gA{5;2lLe8VBgfO8&EKFc{HUI=jIu zUT3MmcbQ=M!8VN#m;pV2KQGeJK%E$%FRXzn7|`HVHqPE2uTHp(0~3AskSPp)>cP`y zUCchGS65NMEO?T%%c?Z;b*+cvCl2%+FEA&1^6LxsYx#23d(V#+_!^j`8xQxS7?@3l z*Lth&E`F(383TZ9JUktZ3ATgi1!QyK>1f=Y(*d%CdO8~UdfJ=Vd|Xx~#D%@YWzj1K zoY}p~HV$P#LM=)+RteCjY_S(4jo;^+0ze|>nfK8R+hg@AkpjG2gC}&`s^LD&BPZUP z<>N5DdL%w^B1zA)#Cgb@SV1^UjjH$Lm)4C4%cWmKM@^myV zusU7>c&TMaLPFydJIh`Y)}5%J>X=mpRY-{k*JwTno+NLp6B3ZAZ4H`amwJ14tLtM6FzbKpb;PuHvGRG7+?m+u(C)Thfl@JDR44lsZj zpu))*B~aegw+2rDoH)WpwF~btw`QIaNzYpf{yRnM_zPCgJ-I7PGfZVmo?hNpL%VPx)cZKSO>Tx~;^~W<%KMr4M z*&h0M;I|xT?4iT#KWw=59Ha$so?d0moTphnT(9Fx!)-Mh3`;?duTc*SLXYHz{bE}T zdI3ii<$tzKB75LI+4q zdOE5xAMHJWjCx5AAQOO$YrqR^9_+Z>VO}t6{A+41kTg?mrQen-D zt<%l{Du9Jz3&)RGz(U6Gkpek6C?g5iRaI*0Y%L@~ZWyMjs0ojJy(XLjkOTl061B1V zZA~PzvjmOY6^8A`Y8Y>?BoXm&cnjUeDr0??#@sX0U`ryJ0m+f4qmj=Jz2WS@rh+U< z_yF^SH9OuROEMV&<_TMPj81q<^eV}Ji6-Dw8`1rM`8XBNZ;Ha(`#h~!idv)*$2C^W zWx%|cd*gsbfjC18@8$NqcmR+`$Q|n^ObvvC}|27^C0JU#c6C59n2b z5BOr8wWY!V57sdysohegag7bMA$ED6H?xKa)JrTnLx5DJ=%hSY=o8}_OAA}3O{p!@ zrYb6tFV=3VQiy2q$gmXI8h_iOQNCBfxfYFb$aSNVYoHdpp>^ZptcvNa~Q^ z;kqBbA?^S(q5_BpdH+xjeI0Ql^AIj23mEDNS>t&0wyO#G)~c6y_gIR10ShU(syepx z{!qGPX>%%{nXV*e78EDn7^Jp)x<#)x$- ziv=xfB|JG1tB799rvep|Pn~q6yR*cXSXqKQteTjz?-GLzgOVv=6M~1?ZziB-|#hs#8nUO-%Lph02 zxhgWMQO$?KH!A1*^*X*JM}#y!WJZ+%e^I2P!AS2coz>tZ5(6(Z_LbvmG~S-80lc|L zN28#o!Fl5+mF)}0F28A|aeb}^upqI)Z(4t^Ow`S;@*2MWrZr4@p(~6sQqY(9J~$*O zlHZty3KG9*4Y?XmcA-gk2adI-53VMT#3a!!0g_4fsC*gV?uwF|L;2OIHu9J(i(?Wk z8%W>9q?Z8^=zrZwG$NQFm8CB+tdeFzPR=^;=)ZS06+wJk3&%&*+>{($t@8e zzIX=_hBqD=3j4C&_}{Eow?+ZqZn&-*@J_>Z*MQkvyR9ADoYe9DF}E)8>Qh_5)ATlR zz-~ka0GAr7AMhqSZIF`~8r(O^nuZl;+dgankQ)+-8erkxWG^*wZ@ngN<`PRL#-FM; zt1=8X*_|fRJrenZ!Xtf93jc6-yxC4(%HLA(uQ{r~;K2`T7hEFxs_M>wFBN(iuQJ&c zKxDD|M%*8BQa~(|-M9S4Dz*E55PCx2!iad^exybeZd9*IMLphbhvMXDCLwG~O4ycY zyB)qn;tp`hc|Ho_$B^(HtVc`oK8yrTey0iI=={tHkRuJFMbSnp^fDm2jW!G_1|U>| zy0(fyxIHrDY+~Wrp*IM`EE#uz=xAsmdJx)bXhXi%Jv&m(U#M6AJ53rd&lB=m1+gsu zo#vIA37ivu!|;Vl+?)&B7G-xh;4fNodm4$NK$5@Hr1L3iwERxfS6cKtO$cy!tlw!u zM-);wB{5Ar6ha#Lou+%P-uQQ#{}BZ_+BU}|(P2FC{hUfm|4x(M8#YlXQT5XD)t+DGrUL$|sj1r|#=aC|Y--F{NioKz?o^F? z{9-}L{U=cc+*?uUsP3Y@(~L2iK2FxIWjP68)lj{F!-k3kyvckDXtZ;5D$RE* zy3?Ttq^A?i`?Y6R?O8I~3F?8k3)HyVvRF)XTq7b**uag~zDcE#l68V?*G_reMmtBQ zj&_3GQ+o@>fZDgHQ9N*L;0`d2YK`jzc~DX#*txMrpX9nBy-L*pzRys-fHg&>3+d9^7kHWP)9Zv%&uIJTbeiv1bf-i2OV2bgpVyvA+OtqP zr-J-ysuSe*Qk@`w7AXXCN7Mu6m|tc-)R_YESnc#x;au>BdY=pbtW)7v=~bSL03yJ} zFCO|Zjv8YHu*et>{cDwhkjaq_;2*VXMrqdv=8Mw#kMp0anRwT@8SW=sf|*^Lrf%)8 zS78U7W#JHNN4!FQ4+avWn#nH2#e<=)@uk-e*U&a6Hz4FA;rP>58kydRW}C&Y50J!5 zc?14VsYFtHDCvjVce6=~7W?DX*5~PULTS(yJIz5wPxR=?i=?j;+}E^kx4h3)66jxR z_mo2S$n??n(8$$Fw??mSm++Uzoh|?G|Cdq?><6?ru_?StyKcxy0I^|WscWRuZ`$TY zi|X~t_TR=H1?l_R^)tOrkOwN=v80{a{TIEuoeA*$hU=;UPx#dW9}M?$A8?1^j*cvy zs(nZ5RjL5+sfx;GG+KYo2?77(cbQIkUKS=I7T<^X$D5!5Ky+kt0Ife6VIN?wabRqX z6*MByiLcaXonhiDfVudgQ2c78ezRUDlom_<&ZT*=qB|YBEIqTpjB3wB*R$f5lIsKa zPVL*x`W)*_(3fiW?$*6xyW^ip$9t7v?|K3e;%4D#bYHGy`__}KxJ`t4LI^&fgtOCx zB6peiaF?^86WSY;XKtEj#hpT(b3y&}=9=Jg;pl{xA{axH{D~jO1@)pGS&@3Kl`j4( zT@Y8Q^DEUp<%0UuJ2I}>LOQZj1diP4{<+7AZkNA~>wQulXm3@XT}q?V#pBV*Aolys zgE0Z>#YL{b2FY4r1k~SBu6gd~l|of-`^aX8t2ZBx-zsM-E<@Y)+srd_8N_MY(TXr= zSL<|Q!a!`$j#h+0%TP99rQp9uD{;)h@MtA6lK-KV_#2%If3H{BmH>Y4*V*`zUn4S^ zFkA%{4mLprMP6bSe7w$r9i`Q>C7NBMje@E21N`o})za)P_s?pfK;L~jtHDVdJ7Uhz zte7ISa-3Sn>?^x%)HAOM3J}e5p15H+6t!n0Y7nWv)L>4M4cwho-NAwUqjpY~?O5Im zmMOlDH8ycDVdBPN?ES4yuysPhVAXYoWmnf3dOkPuyoX$2mi$Tx5ThiBS4BaLMJGJ+ejq zN$jF0XjTukaa0C;i=irjn+;V4{Dz|B1bTBDx!Ok;H>pX$XDUkQcDB)Yo+8`jC%}8v zXjO7z*|KIDHvkQTWiCnbbgP=G|- zo#00qK1D-xhF;yB3h+w9b=82^7_PepOqkudDpAJ!)LH9KRmMGfO+*NiU3C7MQtFb8 z-}?6Ky@B|KE2Q5|=;w`1js>&o$m>+imJh}xYIC#hYQAfCQT>278>$!ZHbrGOWwiSL zu)t?FaR(NQ??XJ?+3gmsj^87z5Adai8UQ@YPJgS~kWxS}d02s;I z6Qw@AUbzv>Ef`x4{M6ME?h&8d5#Hwh+2Q_ix4GhjzqR$ZyG1|3Awdukmnc(62;QS{ zegxv|me@wQAjOG+9p+;1bj_10s@4yryvl+)hl&!N(@JeajIoD4h68+2Q}Q&r-l`-_+(zP_0ZgU$9ZI#YmSvI3Eftd?n5w&; z8EHay_SOl_0ZN+{@x|l(iDzb?C>ODw8mh-B^?|;Km3sMl;mRg;=%G7a>1WB$@`Y%5 zV-6_U25~;O-W|woaQ|H9{@LjM`GEUpll-&Hc@h@y`!!5>$S4lNMK`YbsA6C-01)8Q zevN&H1ZVbd<~BrlJV#F?n|d5N`yvhSTl6ZADFClE)BxZo6qRj~8_8N{jD3Jb#`qy5 zM!d}!D}e7d)Iz|btoXSkvY?h*azGg6>D$84_7AF_t4v=HAZ+sf*24Aei5m@giIPkn zfbgaRc7rPWohgm(LvPlum495|O_m#xa}|7BjtafGBNE($j+|@pSX^!{h3yDaRtD@+ zl)P{|u6C?826MVlJfi+&-u3`?DA%r~*S4>3oA`ii2%n~?lT2U_U^%94r0YQA?mJGw z+`{Pn+;+h6+%c7-JCl;q?;E%043A^}$#su%FBipmO%)-^dH z;GK1Zw0`$jl=K3AP6abPaVFe^hE=YYfT3z>>C4fVQ z>H{R5A`XyTX5wg(P9X#&ncid?q*LesNv5ZxK`t{bwDQbG#~oyLr;a2!yaIfw4U8c` zQsl%}EMa|?iXlMao)OX_wDJR1K;oJa(%Q>HIs{0xGeTMfTqFX%QAb8bNbAQogoXf# zPDV)UIvpD#5pc$Ye7ICYkO>8;;KUUh;jI4#~--TkDykIBlq*~8Q!F*pwYCSJlYN(!<8lpt%d%?a?E395JSO#r9Z#Zb{d&8kx-%EKN z*OApyeQ&>FKb7pgS+Ns!g0AOnR4(Mby>#}vwi+#}puMzQu(YTG`^x|zn5i>zT9k#6 zgJ9at$Z3&HdE@xHRULCgFBmeI(bFP>CLRQ7$;fGu79$72jGB?tB2T$BV3=7mdRk1a zIXxJ1YKlQHoHIc!BxudvsRRi&ognE*BuKF71W89CL4s|NAn8dYNU-SyNk<|=}9C=uniIJ>J%pVexq+CC0u= zp5L!)qrrB{Q_^4;(4 zVe*tT2$QFz!PdZ2hE9*uJFZT<^C|-toZg&rdRxlr%_*n1rJUZJa(dg!gwvbb5>9Vh znQ(e@Tf*sW(e2G{>g?{h(|`r1x1^ljk#c%V%IO^`r?;e>-jQ;8OUmgTDW|ujoZivU zX*`sp2teiuxzz_`ws7~Aj~=M#`oV;g1%(!?#6Rsog>ojH9rB9f1@a0bZH`hpN-I5{ zHo4wK@a~sC(Z(hJ+enlHdlh2=!TguAfZ;$5t7tP1Vij$sLg;C8&I3Jd_Csb82xgL9 z-9cQ3cMue3aOj}T>Vyv3EL6F7U^pv+gEo_sSp|YKBUIC7Witj4lv%DD425SbAXozn zo`Yf=Vmu%?qQVi{>`{yX1pBpu5>PC+xyfKS-$Dy*c2K4T1m|2Z&}LCHhHzmE2Ss7J zdy!511Aq(#d5efr|FBV&0e@wvrD1hH=#yj0S7r^s)9p~R40y4jmI4mny}aIl({ehQsELJz&9$R z_kqS+b7_E7y7zzvyLrz{gEID%G^k-Q8Sp;cj(a*9yX$^jp5*|t&6lT&YgW1MVSrev zv|a#>x8?!>873wG#Jlyh*9)M*_>iyxGBm_^z!&Sii}#!cBgM<3!7veKK#uf09gSP{ zZo^BY!PxSYzSau=gyU*rX|=oM=$LuT{9OYl+3v0F57nVY?FU)kY zuaGVeGIh+lXMt}^m0=32jG!1_tRjL6R}DcikSJ$CQJ$D2D2k4|SwT^9f7Fr_RA^^d zTk_>QO5^qVI6zQW@KY2_*Z-~WIw9x;!;jUcrrZ!IuCu_u)= zc-<-*6tYWve{{f@_%*>3bk*XPNX_L@0>g)-&-&q!pHW81B zOsoPUG{z|! z`ktXQyE{2&${QlxDuj*~7(wUL>j>zeSobIi!N``C9k~o*@=>S(>n_yDAs+-ATI-EYwp3LPLx^K_y7 zJ5}xTJ$~+)rQ_z#KC3i)PG@A$S9Db7bH<{0^wZ>b=3H;6*F~pR`y0 zqP)Y!jker*!j{rNV#vxr2t-RT+0C08LEe~7EbgMHxI`t z^-0E62ISQU3HgRqVaPwFC(gI)Rh;Dv_F${?G6i*4jza3e`qG3jhO$r6u2&mLv>!S| zyACyyGT=@_EyM3S?5t^7sB4{4*Nm$S$c!caZ;br!)5G;Q>Q&s}lzZ3n|; zrm>Uq87{QxeC@i_NFx7xwCjaNQU=^JcNV|%v$n|pc~<$?e?v=jg1lbliY`&Z`w z5a1qqyf|5(Cys1Arz8IJHwf5d0?L305XD>134;S4+qZUE!_SFcSIV#)VUn2)%aOp> zH7OaE*0(KH{eZu$BcydlP6*hh12mIJtHXwEKOnPfMo6F;t{Il3aSWef!*HCX!9L_~ z3w&2bHK2Q$`XK!u3detHSY%-jBX*j0t<>wTr4{^|3Dif8yFcWn>(O}o$PkV%e_|Gr{zB#Mt}mO(evp^g89;&xM^9qcxJ)O? zoNIB+745K4`&d<<9+pt;lN$*LW1=))9$1u?9a728FpLo-Z&QJj@%M2sWWdun9ea4fP=L42|K0 zlKc68dn2+VG{kO`dQM2hBK=G&i)c5H2%i}BQ1(xS@Fm>BloQO9KENf45*OA)E__Bi z8CV_i>Yh964_!_8vGJT5+?150R6uVk>dN3- z8jc_9br**|OFBT{snUlgC_1!3(V-D&D-_wLC__I|Q|{5Lyx$bcf#h#SG6Xm?uiuT4 zXextgt)oc=G)mz?5*lC1235mE?oK zQ=KzHR}~%l+F*u3TOs*7qGNz_)s%NyW~1ao@@^wp3iugC$;jUxiEh7lfsg-?UK1ya z9RZ1CEisfrswRSFslZ94bI)wJUxWz5(prxUY)fo+W0ECd#X<5E3qrIhWIr*iBx!R( zs>GuZNX}3}ljZfZm8C=UFD!yZ%@iM^yg#%Yy887pti80cZ&3^2mDq)hiEzU8A&XTM zPDtKnBvCl$>4d}*n6~g}rFxEDWzqxeRn#Q;#OcwGX>dOPKUxy*K=MRW5!niNlRX_I zi&cvJVayeJV+rb2rl1G#Ju!8d-}GF#R2i9xvgm~qQHBjunvOIQo9rba8P2U;60e>0M38#+sGdO68R*8)_>=C{azFm}oo!JoE7!GQUx%3OCQ01`2K z`WJTPo5RP9pC4ji@#f7%5Qui*>Ye(rUIbok!qx}wZ zzX$NY6tzofd8xzEqKNESN2u&O%J~BmS^?aqD5pR57j{Fs#1`)&t%bFE6%xQrhFTWd z^aoW;&?SK-9HHc1_4o-4c1U;hD9wyIm6Pzn3e~2Y)ykH zI%Q2yEoDtlEw-k`nKrb>uSj8Z#+8O(jk1ChC`JeiBiJu zk)Z~U97~7nP(lrY3grtblrE@Hb~w8x=21b#vZJ|Cs2a(Bk}Xq90gqFZJ4RY5vjAHA`)A&z|+hRb1_fKEAYibuCVrAmd~RGSuMN!^op!krhur zAmNf^thJiODUS?8=8>pTdjLSyX-S!KM0T!f^gLooZ)U07uq zHY~IBXYqDZNk{56@k-+Ut&=bdBp!i(NTuWxTK~0r-+#Sc6Yr@UuX9MnZWN*uCP?zS zizfCJ%6)-e_bzp77JI%zWM*0<+0^tDaO zjy6Q&a`&f)0nLqN7FNr!8hD3ZCBA@!epjhvS#8$0&#aLJY$|kt z!b}kqE;0q`&|0C!a7w&D-N@*-sL=#>KuZ1}o2y8$kOOfi40Q{-!g#m95(a&<2K$3H63djsH*-{I zo6Hqq83+n<$aP{y3CE+5gkw(*(n%Aw>K7u+q7*$cw8A4pJCf_zYt?a5Ea#noq)t90 zw!IX(?vZGqx;-)sZRqQi`iUo+bf(nTc`b~>q{38VW0q*tb%s%?>%?}lRM#2$KzWb} zS+v7E2rASks8CB{wf}@>;B$JFr5&(qayEI&LKwzhP{=0wiRBVO|3hVsx<}0JqbvXw zKvLj7Z#SMK5FL-CK}|?N!_0WPIDQpENl|#Zuo&?2mzpqt#HY(#yI$P`saXooHe7e@ zX$mhe+|jj8b#gazc46p(QxTf$aG_~EDq`zy6pp!my`8C5*6AvWmx;~fpbGkl39^|k z7gUY~MVk`@Jz^Gp*DUG*#7N23*72k|H?t=;Gul^T=7s9F3vNx}%MI66`+~w>G+cM> zXA1w^a7WiRsg+lmwYFivFsCB4ES1yWtB6!rAVx|~ZyX<+*%O-?L;8sEf7h2uDt9!aZ z8!I{mzr54kGlaI}J)uQ}q@z>(s z@_4<L3v~2+V5%}4Z?gIT;dZk6YP8PJ2>~xP@k4-D7^)92 zHzGPPh>fVEjHs9qRWqUjh-P_)19-U^Q31@2h@SLd#8D=@2M}Y%l?DL`hYL=R*|pp> z4gz9KMo25SV=x>^*nuYb2?StK>B5B4g(!WS+9A&zp973)XUM`Dt=y1>p&@8Ps2Bo> z%tu)MLl4?%nzR*`yd{7Xi0n}SasDCKN~~@Obqor*wFS)8J;d>W>?n|<%=kOt2Vp~Y%#3-}R3^#guXQ5kO-#n^|wln;I8!=U-l2ROPn zb}Res!@p3>|S z(TFbj5flByYu|`&WSS?0;0Y>lKSwwszk(U*l>fypQLChoJV{As$lFehXrXzs((d1g zc4Yc!J4EDhW<)$PHM7D|L&a!l=o$pi)q&B(HAr5dq|ID|=47R9>Ka50(&nx~#rSRL z8U&2qCayuksBS0|g7@nHZz2Q_*9F z07B}zg%CjcLtXjy&%<0ycjg8}+mv{W6hg-68Pk3U8AC1D4=F>bl@vnDh-xW?5HtK* zOJT!d_^-cnVMbv?DJL_4Y$)Z}5s(d~PytfjkNn;eJHD>-k~=H zJu=K|$k}ulVHPvitbEKxQ`_cYH+MqGgf+YA+?Me`%Y@aW%4Jg|Z__(P5Hn#Vs+@_} z&r=x})MWIKv8K0-4RWTfvEu+GIcX&h&@y>7O$|g$UH_;Vjw) z_YgDFwU&FRnHO8k{a6D?rVbKE01^)|Zr1_+Y`ec30A!YNf1ht0m4--pTNA05>)uOH zfZV`(*3i)LI&<%0Gk6*B@B45&x&(#XSjth`vvtwTZCHX0+$ttsz-JpYsOsFG-m^cCWRzO>zw8?V(YaM>e;GwL>zB0huv+(4tBH?s&P7L9GFvwdn->59k(81??Ir%=vLO_FiH4?jKtCg$wXV9 z%79oP=i)O%$pcEhM6Yg}3;Z&}b=81RGhBBK7^_k#yFEz1q-;l({4)?e@;sHCOPg;+ z!N`yi1W28D-q5R1Yj!>xs?D&W-V7U-V}@;{hq>p$+`xp^2ybYLps<5ELQsH2LT-@& z35v5ee*V8%Bm9&Z>^u4a&pQBAFW{RD)epEN4M#0^)5(NNbfj zx)czHMF}7-WV~Uu;lsX`zaGG+9+-vJG4|AG9j{o~Q~(w=#IIe<+CLNXRlOcm8W?St zlRS41_kB5QS|r!4a+MSO`879vLU_K$aIRy>>6Kh|A?w@cbO=Yr1@-Hd>oKLXx5@GM zn0(NalX>u+z1^LjI}00VgoC^tSQ?$&jQ)da*%rNy8Hjt8kkF6G@I1}Um3keM0m9!a z!=WzJGMVO4lOG2(a37unv|JBq#(S|y$?^pGY-MXLyHFpdTr-n3fexsSR_gsqqn*p@ zlL;v#jH!KsBtl33dmfb8P+z57Eyf7)GjxEDnMVkj2AYQlsyD0n)=~iVh;p?$z@TS7 zYJGsg!Td7T01M}~>vdjcQkF>(@Rd5W-+)NwJv18>aoL%=x=5@3^oLBcs&8y*zIyFegXnQ!9 zj%8jri;n4kIDaI`0?KqY2T)j^4#WS5|S<#(3t&C0s^Ahcuq&+ezj7|l4igxaicgmf%>=j2l={j9W z{9>0)SZ9^SPyUI1bY7{Ux?HJeoIw50=L=+Rg4NQq@W;{L2^G(JLBuw#RjR)wM;Kv>hP%^!@ zG9I8Ednth865R+q0`Wx@tIUZHCuNe~X?cF7Dv`><)Fd~MFrb@|GO$^gnvg=mYV_#Z z<>u6}>i*HStNy+#-;6!se^8=K?NC9FLkK;aS=ZZ+z+2(r6s zu+?DQ+St-kn35AMR)}KJdV(CKjcraO26h&-CaV)9$LLAczWFwfQ^stM0|7&*iA+Wo zD_;{*NFH4r-J-~%_YYefzpv+MW6u~+-l`mj<#iDahp7Hw;wG_40YJii;{TQV$g^|} z=9Is^H1V!+_9%k^oMByQ6#qL{c`wS>all+$^{(*=aSmG)<>1Ry42xW14{%|iaeV2_ z;Tk$G&dmTsRC}rVr&u8?9>&VAq?nU&Tu`BzaDBqmRRCEK<%dLotay{%&4k_Yg=*@1 zLyaF-O(47gQ{+);j<(O@K4BBjyaviYdY|+OO7et|FhX z4Alz?&1^RW1xQe&#CKL%fXS9ehAK~2mH)0+dGGD1EVwCczLr#GNV7TTY`e#l$bRkqi>fi)vxc z!Q&+31w0l|z2L(Fz9ve3AF9Z>yi@{6oMkL*PZe>;a}+0EtSA`7_O@Z{0GBz z*MNzId!q^q$h5-|0}|kHq5rA3vF_A7LYrE~p}AXW56E}n@(Z8n4G_rKR6oMczr*Zd z9UjQ4+lXWrTkFbz7@2w#?nEW;)vJpQ@FK%?)qr1Yxb7M-R;6OUuMYEva;6p5fE-|* zsp@l;^Q{CJZ!#_c85P+%R~RyEqwb{Iw^c)>9*J7yd1NDT%v_QrKqfKS8Udm-+Zu&h zg%_PnCl6<*wNb-%@WH%`)(`mab3pY1ZctQa8m;dgVuYdRi2tgpk(mVW(*;?5HCi{A z(mp`MJEidtz(1mr5&MtRKwKnMfW$>I07y_;4$(;(`HgygOi7-Y$aA`-qwOo|9tq0n zl^g>t;jJF%_nMtW|A3s&|FAP4hip^>yuUpE78Q{CmwQb%M3*S>j07=uK=Ur8-Pb)c zcR$1w1H4oSAs|BjbtOj_(9usxjN#bVRqs#rItDppht18xc2FF)`dC2t0cDs|8gcI_ z_?OEX1`y)wm8sRFLjP6e>&mT*ia`*C$8456SWnSWIi`aUGR~S-MmWI3>YdfZ;H-MX zy4^SDplmY2uwwU46rp$>*6Oqb*=T!MrF+Cx8rG+O##~pQVP!TnJ*>%wgkd%A7YBoC z4D0ZK=5-iW;Mj=?Ywxh!ma%Iuthy$Sg!R@SuVJMz(eKEY5@rZi3Yq%>$w2C@kE3*^ zeTH5q<&KU{1^I04+#~OlvXnV(w39Ase`Zb+1!}YtD(ZghL_o;^XjnZEFa#Pzl5WPp zqa(yGqyOKH5Gsvf;dbnd119&>9S(eKVaD9epg;&i`GN`u?s+9qTxjU;O_4=Cv67yIlK8X@QFtN|*jhzcAuL-X^E_o} z+3zCDDhR_IxhDlI$dAv^{!_R4TlFfH4){!~^a|jqhAIQ*F3HIl!L8b@TLth+MfsC4 z8dobaI~k+(shk_|Iz{Qp7|nbLM}JV37-1jhR0N8sfX8%1f)8W=Kot8vL=te>F94!N z(ik6M9i-7bOs{Sd1@1OnR}J_{hU=~YUgcO6ZPss1Kex4t{U(n!*$ny zu__h%?MmPI7|u2N0Uu+iA;2RPmEG{un*LaLqeUbZ--r19P5cr-1ZJlnv=-!qfV~Af z6(p!K zJpxWF#Mk=_0^br-De)z4ZcfB<401Sh{880A9Pr&{T^TUfxGan#QeSJ5x@!1I@ucp) z9|}#v;cKJ##V>|v9BVcg_XB=RQQ4$T>kBy{;Kak2$9n*~bVX(2Xbm4>ykYQW zMW*$IrpZoZTNTT`ZDNN-JMCP1BCd8f}M&l9I2R2E=?9(@{fZ_0`ZfD0b7~)x(l+%CBI>@mZ%{EL8s`lPMnn_eAYr65777KELKC6kOmVL!ddj?P-twg>H*!+q z#czp@PKID_C7j{X7Zdt#%tBL9TL1aJn092YJK2Wlol5LK-&Z7-H~6z}%zARjC=&mH z(u{oM%R=btzK#LH?;o0Zn0@IW;`Z2==A%&WK?a+lwN+KO$NGaG>A`I<2bZKl_A`o4s z#Dx;UqU=-1S)3s2)`ipiEDLI#3q=M98AtweRSl`3VyzmpI3PSz?P(=G&@!aQRMKc* zQ~t6A0Hh|eC6^Xi6hc7uFK#>XPh8oVqp)9I5(mWiY#T%i`-Kp&;L1O7Uxp`a>*XE; zkZrxZ=>dp4&bfZufl(2O`;$X;3MD%Kx`DzI7Kf(O14Jwhh3O$%!|E3V$cWOIQzC?{ z3ayM^XjvG>9KR5lp`>OoI)##Y;s@#sVqZ<7iuq7QZXbc%Ws8ozr|PoiRboo}!ho zK}%j+3LC`idz&T`B6hun)$d<%bKFvZ!ntkix*BV|$c#YL8E1(N)>%5hcj&#OJWQb| z;1(1D@|{}r4$0e8(1>2;DJ0+}CVNRNdkL~R2TXi^oc+!3RW`4E&_+T1kF4DD4D=nL zH;Zu6#>bd5G%KbE6L4h^N}1KRRxtoY9*Jzi>5-whFR8d+pRXwY z{s@g8MP~1h&>GCS0bisjeSd@|^&oo4(h56@m-ku15+s_`8!w3vQsjZB1Sm1^q1;It zy}Zi$5J_s}r4m4*Bn7o5m8JMCpM@G^>~^mwAO)Kpjv!!f>NSSXXeimUNJ2J@qk+R> z+LmZ6!Fjz(@Kd_}*eN`t%a&s(IlXCRIffRKJkRo5t{t$8knAu(!zRKSpbhJ1in(Eg-2dJ4SW)AK!TmAp;d7DwrIZnyH?t6 z8hrN|%FfZLH1}4t`wV4guk>_+d93#Asy$0aJ3$R;2nt=GUanme9oNYG(N19lcdGVH zDvivuPLPCmvUQFeGujDux`wTw5!3y^eVFsAQ6@?Ob`|7Oi-AO9qv06>H&HM)kqz(3s#3Z3SH87*|?``8(z_!7c;5$58zSYN&(}=Z?xz zfH)^f0mM&X#J#84M7Rv_I79UV;;Ar(4l@1Vja%!(dB)*FOC1g)m>L`|l>c)L+z!1? zC{^U(XD-d(E4tI6-DBQaVE&>#le9;6ZKE?mKA@db@=p1o>F9Lu`{>}`qpn|eW24jP zIzUOLg(QoGYAU@?RjU0HRPwMd`)imG5yRk5PZr*v;MIlEPGWql)<;tX5G_~Yna&#b z_lG4Y;ZMovqaJ`#RZ*HYA7aQ%ZiIk-G=Mv3Tg14_fgyte#eQ!s=>;C#*j9 zG-SrF)Px=fJTkP?v5-9Z+U9=+8BZBW4TRFo5Zb4~K%<|rv#*Vr zU_?T|pqXxj;oyRTs%-90SSZb@$qi9&P8}A`1v)WcS)=8EvP4U2Ip_{#H6DARwB*=B z9HZ;u3^Wunlv*2Au>}u(2*noI3QkNDEBG5Vq*g$sih(+_>{;6Nf%$UJ(=q>X^Tw6- z*=tt!jA%hm(O?kTQ#4_Z`@F`WneP`e^K`0A5?Lt5T!&IkT$*h^g6aaH!$ek>?NHT4Z6O+=~NZUuFa?OcX-E z=GJ~j=u(CY-}?dWuxrz^TsgD^S`%o*A#2cFbW=j|$%sT9hc!4$qjjuaWfnvL11zzK zkvK|Sb`XGA>&9^3n~X|irN5^9dgUeN@_G|sE^dWJ{jG%>yLI~L0+HYtaeyi68BaiNJ8jXfa0R$wd$qoXN zOH^jV9RyJr%yZU)zo{L>87g6=UUdgS^8!WZI|wkt+EcfK0Qq|D40jOV-=zJC9Ryu( zQIhlyg5Hfv)gVc9Z!%%2wUOS>D^+p_K_}%APWRxciv||aO;!9u%ZAZ^qt;Uco9L#5 zT3bZsYqXxHSJ^HgfB}}+L6A5~T{Z@QSexx2XdyGPgFqG}sT~9~R6>Ig&`l{cAfuaN zXh23cHPL{KZfYScDewd+xr3lHnllI>*==A1-2}0<OUz|` z2h7E}jRE35U<=M5V6GvxgZQE{K$X}*gtGk(0_h*k6^1J>6@aCQ9RwuT+Q1k9q%aaY zh^v))i*ZFei0x|Dq<>?j?gzZiPy>KpR8+R92>m!*d$$;4^f9ol+O^9urVj8)hUx`; zmZCCsv@TizZ$t6zCaVt+?rbKYHSS5s>H|c4Mi}~t1LI6X1@I|`8UjR(lU1qFT5W_C zK$IpP0N+bAb1pP?8ER<#lQTz7eF_dmPm&STZd?`|1Z6&lveJ=29Ql)1R{#+w@9c!K z60h;#4R-0nPMQey4?ErS%voS2sU!Yjr}W5APe*5h++92U!_KVJ-N8=>znAv=hn?2% zo`+7OYlf2ehn+DcfV@PgBSP zQ>UycAM}fr@3Ezkb&h9DMi?mZqe8}!%bc8z?w=31e>TZKMMuN29Ogs@A2CNqVmyXi zEu;mhq^AWINJ-A4nS3yrbh|c!PAXG7xiFWlO;{n9e=Hi0Wegyd4an#w)BD=+3&t!2 z47~k}FdA==FxF-?8Jo~ByyRO9jmINI1a(eGoEn4SxZ64QT6J4Ml1g4klUQ0pO(`)$ zIuE@}uZbIs6LpGbwH{Zpg_|VHL_tiNtmO#Gxdvjc=ocz6S8vP}?S=(b9HvPEeqamUcD)J>=4s)jF)~!~%Yc&bkZ;`Hf}F75&}3zbe;#dX*&|Fc-8m z78Jex$C<$MbRu_On{x1!vQyWr$)z(( zc`sO2sXEr!#6<}c@2O%h(|eteaB8pX45#Y4&M=6_MxOVO3CxmnIY5k(AYL>)wy4Fi zsKp6U@dp!M5Lh*#bhZ>)4RL?f+EFx3L{BQcwsc-;_W7mR!UGc1_i*{|kngc{%$Co* zcQ}&Jk%Khl&(y1oBfyP@ssMh)P-VcME6V?9lw94T>$B7(;32v)`yY+cI9!q0k49-7 ztytj(JXTTqqfweH;X)ZYMpzlY{N6Jhwcv?2V-kEQ_oYF)L!HdH_00naz8v5n|U8PR7(l+1`ez|&0QGC+*U4rFP4!3g^RzkC9wEd#t) zQ5i3-U7l@(fCnooBcydgP6&8v9U-keD$B->N+lJQibdrv%V7nOP{Yi1 zEh-hjYb+{5fW#r=r9~)22_WuegtT&Jqto`d4trA0_L#HxnX^5BzqP7Z1UOOMk~w2j zl?FaL<9aLtZ?_2a0Dk$IqN(3pXH+JNp*aXyLmM0^)c^NDJSE5D?GBo*h*h z#W2Jln~!T|FcpTuL>LAWVc47u!(bu|gQ+kKCc-e73d3M64#Qv~3_Fuyh(8~^K}Q0< zOHcusw6o~b!gnDA{QsaD;xPQ5u7+!EejEU#D%wgT&k1Y6;g}cSqcfqyg;UR$<6?u2 zK_O)Y_@2Tlnw%!$k??RPgxI(x7>6ZtM;y;qOE81^Ch={eSav_M+2P2R2rySS5UU%A z)kTNcKhhlhLa!6uTimO&(`9hK(!M>dPoAo*c89*B)9L%O()f3{JpagMIUZFidUw-> zZmvnnshmuA#w(9kR%a)}Q17l>vzv3tyZ)n{&`xVjll(Dy#|8EA%JrDi>TP05;77qi z<%9lR$~P~`CtHHtH|2zb3O;lU&XJAousO39o^y}SR`f&N*J7miv+f@k_5a7-nZU_a zRr$Yqq0=D&nh-+3HiT>_f+hh05j2VmAWBztRomU^PEZGBTtgrbFraIv!=Kqn8rhfO_v)^-< z_v-bl&EIhUIIHr;DEX%(i_Cc{D_`>_WD|l2%-5;5Xfi>OFI)|avh@T}yRXxe%#I`o z8MgNwNf0ux(Bo2Tac(-k7>c{}b4M)x`h-GuFTHSS&0@Gb^C)D73%aN{c5<>%T1le_H~ARoSnu<#q0FOF*R}FIo&&03orkvH%DvTuT<{cE(yF z-vGpl`eQ}Wm)*yxqG#x*+^Gb5-K%2Vt76^JmjLX&8$6RW*%Em6+e{ z-_CadP}#WM?RM*oz3s!>l53aS6(YOh1SR3-mpc{iY7->FJib24V+}XRl-%V9mOP8! z)fcN;8LL{EP!+$N@jZc8dOII=XX#F=WSXenP44o`Esrj}&kq^r+i84GiN$?> zT3_J*c_; zoqo1gmSF6U?%^&DCv>w!|`4t>P_PO@1lxM_aOA#8A{(nKl2M_ z>z6;i9u=-0dS@wDueaD!wcbcY-HRt^!f+UOqXk!fE=uNjyQE8l2<_X&I*c`DnK<_(#seI%h0U-0HkS zjzRg`7#5=@W$+gpLy98x3@qaXPh#Os#M--JU!SR;62;Uk!_8Ru4!7yoxmW95K)$dQMj)3I?z^CBmil5U z{sC#c_ja3nxsF}FK9`sN0LymkW|uF@oj3K27T=YNIFJKm!PqP@+=`ch#1-rcDGW9- z+{f$qJm1Et4|tl9Mq}NhvF_;kx*=6S7P~4NmJZeabq$B5$W28Sh>--VLdyUAw!k zHI=S~o7(e!=ymPiOP+r1TKG`g1SUH1Zf?Zngf0?|Z&o$uzt$e17BdI z0U)N^#WY6VV{Cmu>?Xz})BJta=MgyCM{4_lrx>XZc#V^>e!X z!j*ezxQq#;7i-tId`rm1Pa!lyNyXIr^ZdHT-B~QXJIapbb)=wD|;3^{x05R>x z1S9`sY<)oNCdTAs&EL4?u>pLSk@|sGDyf*jFIXF`;Dtj;Solv1HJdrQnXnujQ~a7qnQhRZ?{q zx`Pz1%k&1~PAUq-;7PcApeErU{oJJ@7s}&{Xg*QN-Hz;T?aha>Tzh6{kNX49UWkWl z=kC7Ky(n=m^rN(Y&k6nRrr>P4o}nzWLl(IzAMc^}c;)I%aLKJeeE|YopHuGr(%kY^ zgz;{8{!@7mkm~4KC|BijJ5LfIVM-+8skk8qauHa+t0IppiKwt}rzDYJfDUR!a`VIT zxXXAqGOSTc+>`urXYGgXpL^XuKa+n7PxD{CRk^UXNDg`ut352_=5P5cC6VNiztk-f zcx=q#8yB#U*QSM&(i*UowR-_1zDZOd+7bVa^*8O;F9$$U&i3MMR}#PE^6b| zEn>6K2{&A~EBCyBTYm737rx}$;d@B==QroqA9RJ2la~Ai_r%+t;#2U(6_pGNvZgkP zFrQm-7U(9leF7IP0Wjd#J>NDF`DF#w=~1 zpL3#S-u~LW9RcEB!AQ%y7TqFwirQaeehdJS-c=s0spy~JLh~ArYxgY(~*`Lzas{Qxtr=Cb? zZc}o8B5}us`B1Lbp8P~2J?`SI7vgs9%uggmr#rFCg-)pX#hvxbb~`?suFt57{LoTn zaZi`@(EG2-wU0Xr2VCwCHPx|4TIC)RUL?n@|V`Yc_sDrjMkIPk6|E|i&3FVTdM-506PFF3XwTq2G z4T$=JF&vpowOGjgW8_D1J6GlR>|{fg4O2D<+0B+_Zl8eZS(<}6o_W06?Ig-a|8dGU zx3bP%#TRNPaKdqda_(oG>n;_E3B2%}sJy)e@47b$W6TW~#XT?PUU!8vp+3XM@O8`H zTtp3pLTnZ_Rk=b3z8N3=&r-g<<*Cs`YUqBBvdt>k#%GoiP2X|Kvs-0+cBvaOmF8oD za)k;`I_xk}WC!?R_cwLLth`T$Y=&hv5R}>8UPZag!UF5@Y+4FoY0a{Nad)JOA`FXa zw(*q9BrK;nae?Hi3=2t%xmB59As9jPy5x1l3_8DKP~~XE~TCjxmh*bs-Nx`!R2&M^EM^BUj%nkXM8r4 z?`zMDQsMMK{E>F+mf0aoUMlo1(Q0p!WV)9s+x*Jf zN=N06XXW`^z+zCI7jPb{5>JRFL`An(Q5Ht1Uxg13ghjH`OE%YoJagIov_jbqbS80= z=;>xA$8&%l=JEfnInFk5kS?u%Y`T$W&6ra;DofP{{j3~uKIviU_`kHKyj3UMHTo&X zW7ZTFki?fMOhis-a*_wKNdJ+CHH;@7p~NUGsV{=yVF_q5N9bnWH({fj8Qz4A?h_|p zmgr`lH_=8n(|xLCAna%UGrOZ|q=OB5&tiw0Z0~lzE>zx-lN+}?XO)?n?Ann(;e>9I zmmQt29L}0<(X)%Vg(}<6xTfBqf2EU%kuH8qhWJ=W(!VhC2bG4{0)r7W1~sH_>wXzc0T0HUh1vf+PYgUl72ZL6#dJrz>u zCQ?r|b5$U-N`5^Wi0+wghqXO5G&8X`wD=UYbc|Vyp5M7tyHM;F8U~NDNG(3du@p|Z zDq=tw(a)uIxm0^bX%8>gDv&;-*tF@HyIfiGE7LfQPO!x%e%>=R|US z$lW%Xy&k>$viwEd32Ojf`C>@@z_X3i0G^|yqDpApZjAlFA5CDS^{tnfk+Ar2?U{Ci zs1C%8Gg{AR{j(XZ15sZvHnVVqS*QYEW4cCwL@c*mfmn9xtLDe+OtuRAzL7?NWT23x zg~N$8=wKZp#%DiwK_|xsxtc6?Tlp49zHxF7B6h>`Dz%$mTo%0Jy%0~=&fSXJ2y{bU zr`>y4cX6KSrGqT&8Qq5Hy+vj3NZE&Ymv-(d=K+jEyIcF@sF8o+GyjQNC>!;2vgqL< znTvd;ieD~pKg^In*C(%n7W5|hXS_SCj$Z{+BR@nHewOn+S*m*`Pj$~!sqUFN)jgA? zx@VG9_cTlOY-*7-HVK?=vi#V|vmBa7== z8idXe!PpAF@F?J3WW9*7rfXMXF&hJAb{^ex(--+HnB{MHLyUsRSR38ec!mCc== z%ZiHmOz(dym%DIqQB0WhbW*ea3Z_s3onmd`5#5ySgv3CkXeTZkJEfKxWokMM13G0r z!9R%9b!l1oOV>Juz1^yKX>@tllZh;lrK9Cd4L_I*O$w84CPNc8x|t45*yv_fG-0EgSx#? zGBGXRr!3p_Q~o4+ETgeqiE_3FqAKSLcS@?7=;Z1#_A>fH7^&}gYe4Eh@dVjz%6+>@ zMc4A2Ls>6&Wa%KQtQSp}4vu@QVTobDk8?_kD*9Q6+@Oc5QiXsnpW;V#8H5{L`^d#I z5YOBS*@%2gTvg**&92Z!!-x{E3O-)B*P2ux@KPluua=P`sp+d_xKLTTB!+~QNiT`F zsw>lV2or(asptLk|BegA2Jo6wAoT-pFj52f86_17r}fI0nk?{4B^8X}B*c$S^P>)Y zeo3&N(Q255IEq$z@lGC9DdD8H)B zFLy1c`PFWb)nV298vSgRRr*g;J~=kHr#U93p3c*ismGYcy9;yltWyqo>LY(BWM#cy~Cwa@VF9-f;L>;|}nvC`HR6Da#%Z zpHmn8C#i-}{oExV;kh(lqU3I^kMY@1UZy=WeC3SyKzxOE?(RF?NqQ#qwc5YugnmiU zcsE^C=U@bD4VMLTGH>+PkHbdNR4$@mTjoyKOP8$jZ5A+D$hykpEa>;d9McWim~yU z!j<8jYX0-)Mj!A8O3DwcH1On!_~!opr|@#pzkouH2o5Vn;8a*RlWJ+^Jt;^Zo5L7O z$L28B3CHFz=EL$od@6!%Eo}G4OJU?kmV1I~O|ab|aH;eS^s`%NZR{h`8LRZK3oLn$1%&ZPQ)C~B{qHupig zO;cZ+LXJU#Vn=u(BIp2%GB!XsQ-wm9{{4TxP+kP)kl9Q zhWAf3T~*-6j5Gp7p;KSYXnFO~bK=ia9oL$!QKXifO3b5AROl5&e-`=zRqzQ@vH@9Hq&-&F;_HbphyJe8J#+B)`0ohY~KXHIOU z?5C=%yK?$Xl`i+o08r2^r_g4?_n0TqJ-67|1`2ogQHP3A2&>%R1C~O(JQd>Q(U?9* z?VO;W)8xUP)0cMjW)SM;NG){aV(q%hScZVl_G#TVMvm_<6Pqb}vnuP7-?cqMULZ;P z9cE-0h_P<>$^sdd-VDlh%DijdBYzPi0W~v-j0_qU{b4MspvR>6#nnzOrdtvoEpWc z&v_E?-;C7wcZK&DX>^ZQ6a(Tj7afVNQf{xP2K<3`Nu6yUYtyk^q@Ou4EMk;huabVx zgo5tGo{8{3oBz>C0T!<%s^iZ_@l@d^9{DOOj3)8z)^AkN@AcC*M+m!Wa^;VH!aU5^ zt_}Jrc{s8w-&yZNNYm(hNbI3ePpJJxs^b)NRff+~c(IY{KrH41hH?rbIf|QA$z`Uj z4~R0!SEwqH!Y4;w4VnUtQnrFH@Wk$Llq&KOuLKgN9Li(eD`VYJO`+Q>dX!F5Tg7gH z2V^|GhdfsUtoxLQqaF`&^k(*$CE7z`mPb)2v9f476842Q-VNY0US+!l@C8aLwiQ~l zUt^5GB}yt7X??^qhS|f9e=)~l1 z>{)>;0#+gplmE!8i{hma!~r15=#qB?ySJdz9c`#)-I1m~;z_N91l}ea$=%j(PLl1M2 z(EE%gdUJY0Z%%KfmmzJoEQI#uTLD61*%cCy!YWqa&=+BB=7t}FWr-nJmKcI%(GU;{ zy5#JEtBb7&R>w}Q4xK{kR-MiMIChEZ*{YwuC)9DR%6`%oiN3EYJW=OIK5fxN9VF-S z*HrR*UKy~YZ0lG;S$rvehAJUCIa!5aB@SB~REoId9wpE#s>X_PR@@79{AIM_1fy)DN=lLFJwhnxPF(LG ze2#e%We^sxr2+haNkv7CLRe)0fu#^HPlXt7F1MI+;jlGI4o^RCRn7@V- zY8`mEC;hj=pL-JU0VSoLT0r?LOson-qg#Bd87-70zPv)$x71;|W(6YUbk(Aj_Cz(_ zr=PpXi}vMJ6*TKg?sjDPOPBH4P+q7#GqlJ3u|p5UmD;(x@04FG9G?k&wf65hp%2^YQZP%ig4w2QT~%tYt= zmC5}%Lb)IMXD0WE?tf9XIo|O#GfO_vf9K@t=%z&6pN$lMkW}_9oDAR4D!JXNr2(V@ z_pD5RWa(*@rH_AHTV>8(yFS>qUq%~X0HWEA-Id#;J-_hd)%a=TR@f5ip}a|{gk5#@RYShmzyp1 zjMl4^DvZFj6Bq*@Dp>fiS*QZhS?pZ2P?yLsy7n<$H6T*yxqYt+9HXu_nJsibP1#Bd z?pd;@*(MOiI8D{#jMOo3a=#EHBN8)^bhMltbe(C*siT9DN(@@+u<<6eV*QbRjww3O!t>fS!x@k_YUPs?(=j~ zUg7Erow{n`5#3Xj9sMUyUd41X`mOErvo+9O{ghP*NJ%C(Xu7bK+UMzGDkRi|6YBZu z#LLW?!B9(LpQiv|O|J!T;d;|K7dnL?1=M6T>8663u+dEoHDRNhQfk6RH>DI-B6Lc~ z?ei0sG&(4rCU)p1#jVxWwK|NXM%G6lj!(81>0zf;M#U`tqw#=?zhBmE89-o3ZhIv66SdQIdXeK@D(*US5zXC{Jj0Ru z@;`$b?}fOhcJAgnH*Xo!Zpd@Bd#`f0=x`U+v+3+rroGEd`A=WxY;-SFw*6CVqEr5Q zVOb`6yJ4JlMiI{eDS`aG?Ir^2eOi16hCfN;e~EjdAOh6sfs`OZxwdRGoO-uuNq@-JmsP;)jZ_8x!bq!t z3{HDxa|QeyTUu5DaoDke>&8|E5|6yrteK~*i1HlJh{Kav(&Msy`3l#z@ zbD-tXLV##?ph6H~CbZxttjvLy-9&(y(1I;knFB4`LVzOLfs`OZk+p0yEJaFynIhE+ zWTwa=1;|VhE+B*RC!aKU%1rr7O`5}W(hTUQ`)-@6IY^g*)c4(?oE$s0Z43o-VmL;j zWD7Nkb4=Gl}J+1K2pteZ6S-WGOR7nG6as$dYB^@M0xr&>6}{VXC+uzoFvi-M+j!t0Qs#@_fvXl zp;EHpXU$kNnVh9Yd% z`YF$E0M9maBeA&=%rOdyw|Tr>b$&-byDDeOgCGks+8F4mY^W?>w4t(KnfvEV;lSSB zdeTFZHett-p<6LWI3^!d?|gPwV;(c|MJIZXv+S;pvm1TI`&rffHT{(A0x>6x*4B*} zBtGE-;zQ1d7gS+R3kE4-jx3M zMWH{B{IACPs|o#gru5$#UA_IIUgdmNKPB0~=UB;Cfv+@DACL`QzBk|APX4;|3PU!0 z;Gs$qzWdv09Hqq0%9`8u%4j{qO9PKn(k_)XcaFDbG^q!%LjiJV&ix5Ti)>PFU6q*+ zKfAq>ky7+QrvxZ5=%L;}M59zkB|{_kw1H$v3hJiRDBTo|($SiC{BExtnxO@s9gQU5 zucL*&Zq-y0n|Kl541no_%A8*{%bPdl8}am>4OmyK5OIN3SIS}g#!!7u42vr#HdFVm3P|4Mn++gIbD0^;Xg$p`vkFMk3PxHumncs{!k@Ov z5D+P+YbaXg&eB-meiVX#HOU~Rw`goqJn>+{^ajrqR|-rQc&4}v*Jvlx>91&J-K?(N zs-H9Dr0yOPS}j*Ey%4{tox2s6Cv-!;UAy-xcXM}{&O4Q9?=q9WyG-{z%C>)st$cSG z#;=~uZQJ~ktMCoI6K@j3x>~2=;=)FLyIkUJc5fN>m#7%qtGmk{iyofNf zTZj%;W_63v!OqNX!4j;@>J}}*&YW&Jj|ecMcOZ|5Jjz9YV*5)^z9;IAcZ{7V2dEUr zQEtgC)>ZaM{S*?Ae8?NEkK{YanemNUN~ z3v&@04L7bAYf4!4WS1glmI@&O30IE84{anujON5p|2eAvBK?$+22!3Pw{0Uv&#>X| z1D1?#Gout&E*u)gf)A?3fDwt@iUgMQUo@5o>Y}l!^$*AThZFkmHUCRVyEnQ@+!hT= zHP&BE=--s`KYnm!KX)^S3*CiYn5{G}>MI?I+HFQQBmC!@I*GB_VMi@mg#$`ts7c)Ig5{GMjZ6|CPNZ&RKCJ8=8j~AJ@UBK%mVx&)+VTvHSE*s!uKeH7 zPq|e9JnDjCT6fByfnO6XS_s~0f^}d?aBFkHyG^hPED3IFF8E^;tN}lGUg4^|rX6^- zlH5(ET1E>CiTAk^5ISX_0)A{G>idBocpb@cH_mANhf-zyfmkeVoP~w6L^Y#U1cB%* z1Zkmmhb@qoN0TDF?bqDyA~!U)k7YDh+bA@EM6aA-_`@3hD*c>Mkw2)DuPRo#yFn07 z(azm{r`(tsp9}p}+P|mo-~6*{2=u?u&ibLgU+TNqGSDdd)ui~v>qx5}S5Rc&qgOIAJwvD2#c zi(0MEu)U<8RfITY>$^oKIWt#C;heln72Ttsa^3l~5mT40Yq(Qo?r7_EJb>YnAHwZBqD9JoYDdI3$-Pr~Q|nl#}d7sTX>0fQbE1@y2i zYEk;@s3hld0S&~8l-Tw-{a?@=|EGSsD=hGLjJ7NT-)*$z8JPIo6;>ES>Zri0`PIDDR0deInG!FR=kL}<|cySaNsM7nw<7wTW*!v z2gDT_qHXcJ&Bs;XO8s<01m146Wf}O>Mq8eN@hvqRE3~kWx0-JNPf(J(3Ts5qD#U{I zlwkamTCrd?B^bX8qEG!f&$|lT_z%T8BG0e_ws#i>C=iPUV>6>~HVaiCfyhmNAZDDdsu{&X;@bgYLZ@sPz>Ce*eqdRBEulU- zXbh;=XPfsUK+MP=Mno@(YfvkMQRuC6!@Wh2NH=Kk(D`mpy6 zPYPqc4F#SB$FKLKP^w+K7<(B(Ad^Gx5nsN6;C|&*5huare4+j$mH(w)j-))tmW-&7 zU#8q=dnv5VZ7$WPT_<12TGRlJ8)*P|g_5Maww>FxtT%(O(l(Mc`HO8I6OCxMnP)fqv)P zV6o|=J4PkZHPOVtyNPkusH@e2KvB?0&>C3RHRv}Jc|<6-@D zCn)gVHeve0fpu)*D)DkvaglzCNowdSPXc~eN#ey=M&mP{FVxPEKg*nI0H3WSR}Jw| zEEcS%1mmMvELcqm#z(POu$B^xUvn1=4yFX-itp15ywN8I_(dc21HU{WU$lDO#ERJu zoTH>7v9w;~8N)%sk5kNO9r$lX8USM1sjp|W_B8baKr9!Gw7ehDeHi>W-;7p)nURKo zHyLRJh%?S&)m+=lTpI!6Ou>V%hmo%V>E& zqL)T~P>o$@M*D!@ej}AI44h^b`(oYE`;@=0I)9~~ZUF}G`i4Szdu4eBo@cZ}Gw_i{ zJ1he?jP~RVyw+%kXW&bWc0>l=X0#(S@V^@EDH-@XMms74-)FR^X5il$?dWJFB>w-g z_?Kp2VwTkrNXT+W1xUog1tehM0urxWxBv-PxPU|}TtI>qE+DZA7m!ee3;3W$Eq(z> z#0qPsk7nZGQbxc%UMFyok*dI_8L1Dr(nwX{dLz|SC?8jzSJBS6BJjD$hcHS1zZerX0?qBPM2JV;3$mDAdy8Jya1(S_h1^3n-; z+u*KC+j{e7=`uQwRn0IggCX?zEMsp5{x8DklZFDOxR4*ZJe+nCY# zp63Jp$n%|>(Rj%70W0dQ*xiuPm}MbUf%|#B*Jd;h_I$v@Jl{DPjc0m3;0d1Z?2N{s z=L4SX`POGNUhVmSXL`P~G8z|nKH!@?-QE`q^38wXH3qb+t`^8t{50iHWxCxtA2hcD8=X-B;jimE>kL_*2GKTcI2@6FAV%*nUBUz zp12{SafcTM-dX0OahE5)Hly*77Y6>W%tzx7o_J11V-K4*b>J*p8qPi^qp`Oqo}JNH z;)Q{aFZ0nj*b~=hG>-AYz!haa8pnF#Ss9JG7Y4qt%tvF`6VJ?Oyuu3uPxE~1G8$(p z(QQ&-rl2d~t=iA@lk+nR__K9MDW*<1!Di}wd4f%)DsUGY%+;)zOk{ai2kKd962DV$ ztmT*NIV!l0k8DljdnskCLZ7MiXh z;FFXj6TB^>ahT@=KDEq8^2K^8>nM=KcT~=7%VZe<|zy~Q!u3%48*zZZe7bz*f zYfNK}=erbWLonK=a2NgTsGQqoXBx7Qc)}rcY^Gc~E?*?CR@&0Jur0$~e3jlY-y7EK z7IllIRf?U8@E%0>58tm+AJk8I(QaEtc2phmLwy;O{SJ{{D zHmT^_zW1m!UdYu5)6g$F!{M4JT%m_3V%3#xHdeGcRvKNlw302 zuUUG*rCc`-A=hWuO3^Prd_<-0(oeV0fICgos%Z-6deSn5TkW!O73<%t^^#e>VFaG5 zq>jqQv5dwWjIR#-WSNh~^_~yt2Yyu;r3-wk5z~zt@jES~0U#mGkiT=i^xWd_ zT7!!*CpB>1P;+w16k@Lwu2T&XfmPe9oa#0TV>S`AjWP`LZwo@6$4o z%s;HZ%r%;0Y;DK|^QS8XG#r^E%~RXTG!bg)A2|0?|VuC9t*jTZd4>_)z+-cPr&9Kk8fCSD{O zD;oH2q-X#|Yz?xU0g2Lojj}xdre#H2FJ0^kA_c^wgj6_t?^k!2L%YdSe#Q4Shq|x( zFoMRQyP38JRO>9++)CQEbS_(z(f5$@%#1u*w1{kLByxAj8gc2Z)wT%&u?1$NSE0N<=6cSRaM?Gg*tQ-blO zY%Ew!3C2&m#DcYyVEnYpRt@B4A0zP4x3g>y0FSx6Ajt!Y*NxGrDzV4~t&b^HY8`k> zS(?T@o-Z6ze7oIzs{@~IzKsBJr?^Q#>zSs01c-+PBQ5VobTt$EQN@o#RDFBptO^*@ zlBDg)BvrG}s1pzWi>=nMI>VwL1@2*_29StcoXrOwo$@UV@QIeQejpAzU4xPPi9hh# zrelUqxnBe%Pk(}(VfEjwVg`U%m#4jt3sW76@#2?IRvvIYPJWR*ezJm;Ki`i6@JC9L zZ)7?%8kC4^?qrtH5>hkwzN%3zku&j;i{&})h25oh^PEvBH4g56WWNl4LVsLE9`QbM zv3t&Y3dzs|62a0iOIFwMZu!5w)Vwj?E&m(urjNCv%rh|FE&m(urthtqlv$P0@ox8X zu=4ubG3BA}-;`$$$1~pT8R+=3GR!P7jQ2qOS~GDvzTJ$04hAdbBz^a(l3DVi#4r&0 zcGu~Y(nH^ZNqFcxQhD-(_@s^Z&_PwnuYZ+%8SkOQ{@#rE(PL!jdxi4!$j|e1%L_op zJK>;bQ&SFl&Q}ii^FuOnD1dySb=S+?kl$tW80ff38D=>RelT(+^s&XI2A)2qeQLHen}y70Ik^Mb1m$20 zq#9b@=E6xXD^WP7Wiw|Aqdq?hP+<(gquBI6t&1G`yDAU-{>rvW1rsC;dYB}_XZ9ub zCRjeNleb%*yeOX4no3zSqXVkxrKa(W_Ch0+joKr}Od#fGV}AJ}`6J|oDj2$giN9Ax z)8&-BrF~a|bAAZu zjcjO=6olVtN!dVBAidA@)`94qE*DT!qykonb3?jBaDi>ByK;^?3uUW$FbMQ&VQw}S zAF;^J!BZ5D-su(N4i!^tUES2H8BXJoXvqbkJ z@tZS5!vBy3|J%~jI%IF={~4NI&OY6h^*a5bc%P#Cu?VkMCDSYGndFdOYa?0*qOimr zjwdXfG7|ZK;@dxw7dY(Tlf;u!?u!9GPrMA7y zQf6jGftZ;SnrX5HT%#(!Y^tJF2Nn)|@-;e;Ov#_jUpLkjjt_!UotHwn@ci0r9PU@2 z_k0Jpa2mh`N^-}-MnoZ6LY%9X0QveI0s0x>St0K~Z51p%IJ`4|O~D(6}?qlLqX zXEfr()l%Np%-a*q+Zu2e^P?Y#*UpbxMhnMfg$Dk;mj&W=AxmqUX9N;f!8r9e45lI* zw8%)l+{OapxLgwfaa_27I4)ek*P6G3Kmu{z4km^sx@XAHOw)|-QaQb2`QnV`3?+A0 z&RXBK(DgteO!<@i6-5tol(>9P7?v8DI+>&BJjx1U6o?tQQ8u_iyRR^|mB3q-l;7o~ z@n6Q*2fW)zRp2bmZ!SuMSmaw;AVVaTPyVT%T$Bc-B%;kiC+}Ty>&(6Bi8fiECUhBz z1<^3L4jG6AR%oQjIv|iY8J@weTqKMGHvq(ryO9}M5Xq^Ob8j=W z0396A%#UJmAGvzjC4eOKg(k1i?v6`iFYgaja~`|#=YmE ztsf~j8IiL;-1wUKv$&rMrr^LFwy{Uk&;{mx@k`PAY+YQ5Q7^(GdHThvj6c{ZV29Pf;Hgp zjnoGuj$*r`MJSSeVA*K=tqn$hZ59TBSd#r2h#A@2fS3_3;DMH$5#TZmjfPE@)UZt5|gzU+B{wkaw~7? z6Cc-BnX^|mQ-dOD`5A*rM}WDu=^7=WB~K&apvu)xs(2s8^tdG`URjz+T$kmeA4p!L zoQpFrmFnxelBjWrFRX_jdk0G2czS8^wZv8L5%I>I!q29`9P8fuD1 zVrOaUs*K5*XMwFh#08x;UQQE!Ils%R>NEn}mbKy3uDoaGIe(_s^EvwIoO0)K+V50) z;@*Eet@`kk*&$_AVRA4J94-7WR8_25p*CF{x^00%rFDdQ&(aKj#A=}rcmCZ1>thZ3 zrYD7p-lqz_rJpjH`=1$`72LDEB{vrbkRqQV%sAux*dqQu0Pw2Dn zX~FYK!t+Y#xvktdKGRm75g@TBZ;wsJm)dGM0(^y%j2FT>&kH;^ki+x&+rcaut)xeUb=t3=A=;egAlQeNbhC1ScP8%=nLJb0L%UapwLhX-t{wke0 z&$Gi+A63l8oA@q)+0HsD6b;;_sbs8hq5n8g2(42LI zHjmL2X}zsreT4L4TfO^O{9ofqp`y!G!B_QDCVqTT2NP?Y%t2N*7BwNUOc9-sSZn;~ zMCUqm<3Qrl?|$`d58E62fOwRzSNL>=4;P;pf}9Vp80efBramWz*?No~zK*a%dUTNp z3(A`gVaS-c5Q|x&tFD7m+s{sp8XJ9TOH#9Gj7HrCB}xTy43ecH^v^oNk+F_&bgUyB zA#8HlCRqjIMsXR14_vkhBai|v81abBbf83JIg=5>lw> zAA*sWFFTRKxx^pqgi>xn3&m24TtcoVAr~F#k-J^Jb$|GX+%{dt9qkK>@3N&mx?{Sd zs=jQhs=)6kDSwR+4R11fRpc%ISX|5d!D_%gNsdD)m(NfvmE%y(iaZpyj<6=s@H_R} zWftLkGAKB)a5-X5ywALkH{gVv zU4oY6qS{1m=9R@LyUbvZleD9qrMA(Pm{vQg>P@Orp11?PLrHmU((opu+FYaAz04Bd z0De$OZUt&&wD!M>dsPi!--jTr1`aBzkfn9IXAEmD_M@Kiqi%k*nICl^#$^=+V!2pF zX?Z`Q>WTfRru?XyA3f$r6^L>9G8u?v=SMZ8<^71hOpg7irTnOwAG?_!HK11?otI+u zgDLfcrhb8`9|WS_#T4HMi!BVLEDV{2UChD|uxuf|4>lf@u1xf9s&sEv=4jwvrRn__ z`7#*_qvY?*gd^m8!op2>QAv1F3B4%3n8AaaE%*^2PU$k;V^wpy&7u+DdLjE+k~8AgF&ghvV(NCp4cbT2 z@^d$XNPxMvk7|^JmUB)W*3O=ou zXcTK2K-1?;(*UrfDY{1yYg&z_drZ@6U`bPSk0P92a)jMUnz$E%I^u#(8!zo11q9q~ zevjghcYcXh=|TGGoN`|#(|)ea)#$~i@wDp0Q)Y)d5-~ZL2aXp07pf}OtWcYsBSV$e z5$ZiqXX~hb%Iy|pUt@vwtyg%iCxwdoRl$Y&DU&(A?*J3g%T!}UAb257r8;p(VVhe0 z4<7~)TlwmOwQW9-mR6yER{pP>Ty(zvk(!=huO;Q7qw0`@~#u$1Z z`%zE%Q8zzsHb3gXr+x(W13)Z0Kk6B+zH5yU_zNW!jI>sm!vnx;l~gd&y2CR93ASLI zI^Jq3-m1m>b&IzOB-v7gKq7QTs}?V=B&s#edoATh&HT8<{HOuF`e=!W)eolB51RU~ zn)*Q?>Rn9nAv(4&l(H~n7S@}EAz;};e2DH+0o2MiFrV3QzLt%nQ8W~QqusA{@eBQQ zw6GKs`sXI}&rRr`o6sM9nVitylhEIj(BG5LA6?xi^mn8GV$Gez-3@*R`4n35#3~f3 z*k4Le&*a(ngx>m|&}ZM%Lgp(8&nuzlwsPb6)mO8ii~xy6d3$U!c74ovfisk3yb$*G zyuinXybxA+Uf{DsUI>i482eLw(CBXA3`WLGMZ~6WwSM9Igtj*OPQdL#O5YQ z>MzUMXpr8NHu^|?t}RS^PMkXBJ9`PfKRQG7GpI{!tonhBnJft_LSJJ|tI#xTnpOcz znogct(*T;5o2CI^NmF#87W=dsO)oS}tAQm=(S=&LP|AhcPSV5$8S01&I&HkP3pEJ1 zEo)_yFO&ax=a1J~ztKB}0FH}PdMvz>KR%B8{tWlA|3GYXZC7S0Pt3;zrC>yp)N z>oi8JBeY~4p*iaaZQh~vGS{|?K0<1@)w_?ye-BRz6&?;qkO%>r=xwi_{0$8e0aq`=fp7eIWf%E zb$a;vh8@zQi$qvZ-gF2<#>9nK%o1I7-K;s8kcgceH8$?|HL_`p#)CE}Q7VwbH%mq6 zpLK*IV;$k>SVuTQ*yOTJvI@kF;xY;!xNH+fAO&17;u9BW1(g=VEaE_ly;#;p!$}MQ2SZa|=$n_-Tq9Z+W8`N9(ytO9}Ct~g7LegW5H@lF#f}K1ZSD6HDKA*_>a@6?>P!2M&K*k z3!3~M0`Lt=D#E7q4bO0Q{noZVPh6kTSmTKsG8!LNqFl&)N#PGYi2-1|WkiAH zG_^GJ3pL6+EZPwub(bH4X<)nC)s~F^0xN*mS>i{46v8yeEH9SMAmFCg0>zIN6{wC? zDu+d`4kTQ?c5Ie9a+EnV0L1%tdCMVqz_YDXxXF`ZzoH!rzkZ}6i_f;V1-)V$3m_Hak#UxUj7K5)+wG5h_?kJEi&f@e{i8_fItEgf0m0k=#c9` zAkwO?C!sF-+tXOxa1(WN6Y8R$1dCUO{>yaCpJG{wu5GtzhFRNpY?)VK`=H`~r4+fo z2L8rKjlI>arM76)o~>}yNd2!+c!7~>zzdBOy*3^XSpTH7fvkSALjzgKpihOdy6(c;)!h;jgNX_dsIaK zshwZfPq)y5J8Y&mx)idA$-){&FqtMV7F}In?4FVX;C;-2=!f+V2^?~MW2J6lTpgOP zAHel?BUsO;EvFWuTIrSuj{jnB@LNif?RhMt@k7rS zdi4(N%8aE3e6Nujz>g~F36+!9kIkC$XLIEW;iTTEZbFk?lCQOuMOkRq(bow*H2y@P zw{Ct>WdD2(;*I(#Kd3sE(b#N3*MJ0_^M(1lU->(CDSD*3fx9WmEfw+onpm)&5{&QH z#Ddk7V0^zO7ObTNQBqn}Ad|bK4mwG!FpWiq z=p^XbZhRvydKu)TJvvEJT2+{^4H_Y9l{@i;Q!4cvBFGWtRF#DOqAo)V>1xFWwNfe@ z)cHyXRr;Q(LY9Rf6i%!SIKzS{cc_3k} zbEX~nnBs?K+}~MDmAtNeSC?DJnoI_!ORf{$Zzs~mi)liap;{1)m34#`FAJuLlsj?Y zw2(rni&cv3C$3jw#Y3^;=;mcC)rh5{Ya`rQWP?^`yx;0cz{`v@z-a%v%BF z1~&R@7(5^$?Ulna@b_#c4F6c+er9hNh{GL~b9$^=w|hR|4MwT~KdYpU%G!+?jqiFs z;Jro~2L8c|wr4b`rnbs3q5E2hZV<cmvZzM>fj3XYBzb+xlGR$CoX!t`_<3TouE538gz%VW<-sg<5K`+bmakI zAhzZCnm7d`aSEb%uhZCWwn69z-lC+A%IRYn4RRoN*RhRZA%(FDDU9#as+FL`^H|XU zp8JuCzFcCcxuz&~(eSV!>k>x;E+FzV9t{i?(%~t&poSrYmi=anb&hPM(;J37b7;)> zG&A^ED|owZC`Z2o*A1cV#FigMS?uIQ$OVra1b{?XFoqTiYO}})!(G&2f@?t8H-Lfn zTAJu66H=%uaZJHcTuFTS$~ZHq^5rX1-quiTD>}Kvis~qO!17f8l|rw(9_x-y7wCSn zDQckmu{P`tD(20eL~;Dulc)plK_m7cI$_~~Pt!0SkX(1<0e{IH9sY*Gz08GSAP(n8 z9vWA9KHv>TssTT%q~ge_?&pWILxCQW8SNI`jm}i z4M^&QD@>P=Vsld-c4YjoR5MSp93)C28vivK__!sY0sOs@27sNr7USbK??y&zSEY*U zz{gBr3luM%qw(#1jK0x~R)Oy|(g+YUF4C$+I@64f z0Lw<>uLdwmPUQdy>@Z!cfMuidZILqU=NW-zqw&`YV;TsNY1R_@vz_R){ zrciyC{GR7pxvzi?0_>7scXo8z?^fH6zESA3?>3)mKoXKa4(E6w;Iw}n4hp`boi-^m zRS*c$s|w5X_3GHe=3u`;9g}wXb-J)lJ6xUO)^r=ZXzHP!pxyVDV8rEg1;k&+7JB*v z^$_v2s_1Ff`Rd#W=41o-JS7z~oz`u}*bn^P1V&o-c}CzbCoqN)#IDp+uGBMJX_$pN z@D3x50`aG~+MxBIX9SkLi`S0WyK2h2s(H8CEL4H#7-;~AKgA+L>weD&EPEGE$k@AD z%Db9*_cF6k17blgk&c^sYz9;M2NU`SL;Wt)_+0`=t3#-DCv_z;-mMBY2|bJ{w%BAu zrxt8ds8b6znQWopux5fnlI0voA<51Qq>3J8O<|O2E(Z)?*+L_u#WWX2AT{M&iNB}E z02CXl9M&PXz7-{{)GXn3Qz!>Hy_O6b1hnucZApSV->KUzP8)F@K zi;+fwxLC;2y3aEL%MQmoPV8_MhnJhK0U$0qqg8YGBx9@sUv8uUATAcNw0`Otfn|r| z`4&4|!{M@bHS_L;W}ybeg52P_X>~Y+bCiFPe##HK-?BQRg)P}RfM_ciLv{P_s5&2o zv)fD^#^s<7^n&iMfTk8~eW0h-)q0grt*iC<9O^njb4?Y>QV*mGWnBkSf{(HwG0I9J z8xyc>p^?%0iFzrFKwm)`QBo&Ob2)U)QuchLxS-huM zyfxt4jMNV#Xy-@G2L5?A@HOBiM(PKWheDPXsgOk)NFEACS_CbOKp%hf#Cts6gQ@rj zE&huw*g;@9*g*@Hxh)41Ah9@y2Qykfvh`pP*j^fM(WtE%Hr|84{U?v1?H=$9aFWi&yMNHTEOgOJhI&6x&JYn=7e$hI{M*1(gL3nzAzBS7V1x3P<>C;n#!d5 zXfq;{9DZbPPn>wz;5dH#F((m%5ZN#_IKgpKW<8MOraLEUl5o{+1hvHZY{yTY-nFLr z;bK6R2VmKcYQm45bzR`Si?(nQ$F*3#kZ>C0sw=s=Aj##fyB{3nS5}v}6jsKiu#&34 zup5|0M$5Y&?Swc%Cf(m|geDygo2W|tjR(5kZd0iW{GyTifcGm&9?R>@Xgp*@c>xZM z-y2b0fJ5U6dX*^R>oXcPPu!5v7`N-mzN-~}+mpgzrMkF4I^E9lRUjF4`K>1M8+tynM43J-MW6wtzB_i<5P#8b)BU}PvR|gWV-Y~XSFT?Oi zZr8=G48#VbM|3cFqAGN2D%dOU3za)I0wtW(a*TSf>MyXa?xW+r-mYDTf%sLt>L!d* z{z`x#ncPAGQit|J90(*tc*F7K7Sg!6QU{h@>51mQJl}~c>ugHYfw<(t=*ei|kG*0C zf_E@_9&@qg3UB10(KstZxPm3&0%Ajr&P`0v=mFDsh3Y>`C+3@Mm5829d4syj>}21y;<;HP-9u|Vc^o`WJ6Utm z>ra%b9=$z0Pd_Cuzyp<3Y`@_s$Aa~gVEk2QELcqm#$S-cg0+-j{GHp$8pzo`M&Rov zjS)dUW$gf-r-J*+-`LG6{@(uSb7mLdkjJ%E=IoWhf{p!To2@tD4YP|G{5)9xxPRHA zWzB3Y>s_>9VOwE+`JxOdJb02jKrP$>1qkB$Qd7`0gusO`pi4DhTl^RRFPn4>wrcOU zyddxoEeXEQ1na=EU|gVB{D28ofn~wCKyOlkyG*bF{9hyW0UuP-ee(C4>)X5MpCqYW z);smfKzi4=7X!TmB89-gd0non6sj=*UJjGnC1947nY+N)(!+Tnv+S6((kJSwSKVk{50==`tK%90N zt7f!tQm%o3xL7dK+Rais4D>NXwH0T(mWrWfF?`Hor~z?YE?R(i?Odx_3@@-@7y{yA z!AQ$HT$?)SgDHmx&0$=U`-8x;cZ25L6Rjc!frCntnLn1%Akog{!Hm{-yfpB8Woa6n zIy{9mEgx`n6Z1+s8d{&cpi+*;=U%9QR^k%fN z{CJ_Cbao4`VC%JVp@Mfnw3&DR>b*PQH1Uph$-4tiT3>15Oqd};i62zyK0r>CgRRIS zL-k8!t(rxIIg*R6Z7H9KFp+k|V`P|dJK!-Q%*!3{7!fA#j(Cg=6J-ZHo~nB0%Wz5c z=#^z&=fRaIbw%|b+wuTUq7-FR-b;5}X=^ht2!?899{#!PHX8=w|G&OIU!{vtN9F9X zjK-&wD5q&4@#cKtVC2LwQ~T)gZK-}vm*-cDvwH?Wh;U4=&uL?#Ul`-=ok+y_%?5XcB)0 z9x9iMICOhOBPjZqMI3!y^-(qd9rLIkc)yWW0UuJ5`(AC?qIH!8;lO@8P&d!LvS{g| z?xjnXFWOvL(0y|6md;`ZO38H1i^h$dv%SmO24vg*lOK(#+{BR*_wdENkV9tb;&hi04Z8^Ox*E-N!O~Hg{)cG1Pt?k! z)ZIh^f6}V94|t=J{pW9qZ!L+`0=j3 zOZypbrw+`(i`55}ja6J05g-Xk+<2iZNI^Lr-iLn10=x1|38o*hi2B0;`Am>6`SO}H zdEF!Eev2s@K{o*;zV*iONmd$$`cvwLEbl-@QeLkF#IwZWi-$yC+A|^HacR%Qd9q*X z3KR9#V2H)jQ1NcE>m8$8qTE3Zl?UCS0cr?0lf}{3OzQJCGAqBP&?|~onj2%5(A&iN zgcP=BR$gldq{Y!Jh8s6#R?aSQhvp{shEaY={r!r5y7w%+RxVL#V?&BCRGg^GU#J#X z6Ki_R65X>}s<%*HG`H|pmPO$CN-CB`T0irQVSMNsG?qFLX{W26(ek?LQ|a2JB45-k z%0hiz?xz{*iz1E6K9$yKxmKz2P;YmkS9(HUO{AG_(wBd$O#?0NHgZFL^nx}S$eEjI z#JSmpw~LEY#=^Gc?wDe(V`^`8Jz%37%X4Bjacx25S=wmm>X4Sv8}!qy?%;PAZCM7s z%4o|o@byMJGy{LhXoqFsZyW8&8TiLWJ3Isb#%M=m;I4fN&)X|UX5bk{drEe<;{Vl8 z*~owo8>t5Dv_c;S?q;MKaBm|G1Lqm323%sKVc?UDR0AGiq+#H(Mydf%Fw*d_!k6f$ z8yN6;#?}WsUrBNjZMRamtRxH0jCN!OzRGCJGVsTZwmbuW#%PCT;D0yTVHx;VqdhqT zf7fV-XW+Yyc0>kdV!3$3VkIX8n|M9l(tQh|4f-j|BJgec>lt$Ow|$JpHA)md;CTxR z6**t%0{Sm7{dM3a`xVS`VgO>LSUqWd(rng&pYy(5K1SnKB^J`OX3aPEE+3Da+ywE{!(4H zDG0aCggMA*x=>;{wH=|0IUomlAcfj=iib9ZF-%Tkj>Z;kxqPi(E`iK4d1>{=j0SCEpyvSPNp1;G8(Mh1Pf5(5h^CafYKGQ6R${8k(IS zLha)I_OVcZPMj(x{%DHSF;MafY`Goj{$dLXNu8eexWzIWJuejcQWDxxkN!#Lj8_ij zv08NUbAE-da@_^ZPk}1%N@bT@+G81wJ3O%?qp|yA3MYlLJ)`k#Pi)I*{8))ne2mG4 z`Gs)VD`;N}`(z40V^wpiv^r1L&)j1gU+~0@84V_j2p87PXoOhnsMa;(tBBRr|4om3P+B4xUXK6G^=I$7c1ZFu;%lbLGNM8bwJ2< zh8Cifzq}vF4HB$Js7O#0xmx;|^ifgdT&x4f>XeVT^3tJqn{p+l8(kkyvPjL2chmI= zWtn`s!SyNSo@%3+@OS#DaY({64mzf5%_fx7Maiaoq+l>dEODB+Qb*H5g>YT7VWxvKN5V(C{4mEpGR##Q=9q^*jx7mObopV9)%XD} z5T_Z=t8x<#$T_d+9WZtVLTl|zjz!K$JO50M1%|w<^1#!7-d3rE{u*7=PTru0x%z}w z{N4KLexA(THKEP2*DZe?v8B!RKw-_27vHrNJAs$SVKbYOrzG86(0d@NZE-mcb z(kVBO9Bb$aA-_y@cgf$@xdNe$O7~{Obb+7&&P0KPEK(rHr3xhEOAb9dF+saxcSBDb zKc?k$8Tv5fkTH?`Q#2c=`{VeiAX>F4B0hl`V~6cF6GpCNxZ0yuzt;=Kdhm?4a2vmwEk9Hf0U*8TO<&ju(5Y0q5w za>5GRpi4CzrQss;Blrsqs%aZU3mVLs{OwI~aV%5EAeb+~1 z$qz?{Lo9}xZvk+kve_(3@FfR`n(~%6$a1wo(jNKjXST1Woyq0UK;xZBfuE{G|ii;d`+paia!F#zlo#6c> z7pGX+svc37v{C*4*gNw8yNW9Rzh2VmECdjcT_AuPGVHrb1O!~b_g=r(bSK@3in1vh zLVzIIlo3Gz!DdE8^zsvR*kppZpbm-xg5tuAI)ei!iXb{JaY0d0f2VGp&#Bv0^}0I{ z73cTMA9>$8b?WR@w{E@rZcAOR^fyvaP$qTR;amgRAs|Uz#w=X6YSF)#FI#aO*y+**PHtNv+@u{+ao)+jb}B1QSm?S{S)}4%35ObV5*n$vh-rMgL&c5TV@k!1 zYv*o0a4ZCPxEB5UgxdXm{gg8fAPHQDqm206y3-{Dz72Qq=FD(M9_4q81th*QlI=DS zH!9qbH#G!5TLsJd1Z1@Qp%nT0sMEk_*uWtm1Er?ctcu5RgiEkG@6h2yT&E>A!#cCN^<1HUrM+E3Jwr(1BF zz~>ohk#{Hj&nS#r(s9+Aw^O>FZR78d(++W!U^;0Hh>@zG4V_6Wh z@I3^^N&xk`(TNk2I?Wd(R`{|3jKkDlBY?4HIFXnM#!~e#uq45Nh40`pJMQ?qw@%9? ze0&pJ*>OYSOWz4^FOhSz<);O_MM=4P8iv7>xxHlX{BO3aALI!~9#fS=X&_&=_$O9b zIG72-NLYCaFms@dMxNZPJQEy@7{CY~o0MnXfK?-!MQX%AY=pBmeajwppj56B=dMC6 zI!a=!!gFfe@w6ny1)0%tX->zbt1>!-Gu4qk|DDmXCZ}UfPKR)&I?{_}ti#=n#i4Sv z*o%@cQjoioL;loi;t$*UI7>|csg3eEN*W}Qha+R_WPI#Btep~xf&Mu=_veG5P)6Yi zJOYF{p$7^ERv!F|Z485AQ!9hkr?o%Ug&-(~a=LJrKJJ8iU(W6%B&;5#6IyEru2rtp<^ONGr9pipGxiQxo}( ze;nu!Yp_Uur^l!$g|!-{PJHw~rhJ=}4^oAPcghEI=%n_eN2f|2`VQ4dC65aEhoZE1 zFAZJhp(wfzRhABdwD;xunKOyLvnP~9?}eSX%IDbe%b&AbPpH zP2~x($8|RBtkr9e+(6h@JCY}>==g7C;5nGe6Jm5dN7IyhvWmXTCg8C?O=LzDOTFNB#pQ9DA)G%x&Ea7QUS5_hRmIZXpT+ej@S(xeE|r^Q&iVdk`-_AV;VpS`xmdLgi;Z%iok>{VFk zVzJpIu;^izFwsc>=O#=VVmme;>xGwKKataZ93v{3LxjE6=R+)(O(0fJuPybb{SYv1 z%gBw&5k4$fzYeGUXj!5bQJ1t4Ep@4wE3>%{WK_2t z<;DLP22bYQE9@h1z4O^UoCxc*U}pHr>qunsA9slk^_-wb0Kn8=zdHJ)NHn zg`<+5b-tQn2YzMJ^IID`5gBh7eSkTfylD>> z&P+33w=K}o7wadJj&|jKHG5;n_zGXv*O)K6(T0h$lFUketlBx5z7@*1T0bkXg~Z9O z#1?o-${p`T5d3as6&r!qDJl2OC&REilZvT^Jj0g_dCZp$IZT{FX4d8Rt6P}fshqD? zzJJxvO2{F7Ub`}zb4K53EXg;Cn#%XURPThw24H zJFc%afxpV}Lqw@O8fqu>MO=$V>T%y0`pNITXy2gpaQFj-Q$Shl{q?M)Y&bKAKd6U= zwIeo@bIoJa9Qoc2xX?&VAkwns%1P+zipTE0gZAZ0m(vcp4U(toAmDiQWXFCgNas&= zboQr$bYkBVxv|DfWMr;mZ&w>G)laz{0%C#vR1gB<*5Qu(BP#xP`bp`~Mr7yR021KL z6iCb>1(Lmp#C+M*^Hs)A&0_ykLE6~S@uz|@Vk%I{Ms{3L%bMb*L7xB3y6L4o_rvdIJ;VA*J{(a2#7JBk=8*wW5*)kqe}9Ov`#XE z7Xk6XGj?+0f6R>raGF^+2&}r%FgMVUSviQ(=S_SQNND6?8DLd>)5QP4oF4$8REkIhVuxp(+>~KVc%CU81|F&;Cu_K% zb+l4teZA?_)RN?o_v0FUe3k_RINfyq4dzhBE@Htv6;pvs5t;6(&o*Vxw`v&y-mj#7 zd3E>2YfhCH1wna0d-~)jbaJ9!(EgDL8Kt_;x0y8pB+)@`ob1>^_p|s_q9>02bg@hp zhPeglAPnwLy|>FAC`1EMw26*;Ke*@ zA1uVHq7M?DPatKoSZrvqaDS*oN;Zzjb{!+40|*^KNF>sM@6YHI9Tf;40hrVR7zs;a z0Ug9^Qi+pC?c)AEifGTBw#h_I7&?`b8<0jVl9)zK7&?@d6PQMA5)0C(MHZw{n?zz9 zwLzUJED>^AkIXCWa!_Y_Kg|}t`W6az4y0K2RVrX3TS14OtK2WLQA3?ZaebFP0T46F zOFg0}7~$(A>R$6?7&u#PDleOe!m5(kNf<_S*Y+$Jhz*VwkGoZsKW;`efYZ&0B`XUW zIQ#6{QLi|4)l~jeAy#0%+=~M-Z0f=~ng58HKMee}k?O#Z>iNL^%}LpyVCw#s|2pvb zMw$=AiwEUzc1-oJFD~htDyRPm&=UO%@&4~CpQ#s}smgeA+Pnq&7r=4$B#py_MfkB0 zdLVTplVepLz7BuyGEU4o42`WLn%@rr1~Rj^miKEc z>3Wk+hVfIRfQbYg|5uuuwn7rV8>g&zs+BR-Y40cQRHb@uzrfhcq}={$(TmGqV_lbb z(oebaW7BJFDNeqAdYN(`A4UPWFv`NU=2TX(bAqV&R;lERt-4s{eksUj+%-W|JnGBJ z{csqyM!E0oJnC-k+Ui+ep4?vopQj{uuiY+aEeVXk8;vvq{Pei7v~GJ2N=9PqQ#)EY zJ6dMPai+8dM7f+Z0nzMyXce^PKN~w10XJ0BJR_~+%-}^peDI8&-1vn>yaC+N3?2j$ zQzxroo}nRgd_KN%t-c>n1Ld>LISHghfL=` zn(sqE#5>=Ik_}_3VK}E@*fb2AhGAe;!*HTuwhnleeom7a`_5Gb&2vp&9XO_eQ%;)K zSUlUnyT^0Ve9$a4 z>A?JFIEfP+qQOT(C$%8<@jNXj_DQnw{j`#9)=wd^6?l$Ktom*W_YS03_7_yZZaPh6 zNk2fj53^Cpkr$)blgK^>h#BSf1W~x2Ir2i(qvpvla2K^nE=h9XLKbFuYA1g%!yRxj z5-W0tUiYi|tt}1>;4_sZH{B}>8n`?4b1JOBemV34V%XG$#cpH%;~|Q`klFdbW3qug z*z!Feh=JX;eF`vp>xtw4H8Z3RBxL2$4$;8%?0FAfVfWOElTqk<62*zA2i2SM+=a%5 zRw{KM)lu@%z>ak|pCLNLT&6dWfUSeHPQ`*Foc7Qr-0N^;6C1n^Lu2cHrm?1e<$(<# zVV?RC#NsKRxcIC6Vm(39vkw%K%Bd@Ua_6kJ%#ez|%afzHD5f60Cz-amq;}VtdT5xj zOs7}cqL*y2Pf~7f6lD?uxw^<2D7#%&#N|QOTNE-|J!5RwwW@&|OwqtV?j(g2kNSa* zss(i)R>)mtX4Kuv@_>zM1HY#vcRSQhw#MneR&HQx3IiLtfsH8)Y~}{0FE1ie&r=}h zV?eerau*0(q+H%@TL0SR$(aKk#P4h3Tfk$CGyp`P6W=Om1@Xy24dV6jznW|ifQ<4t ztT4u(`r`u`H~>7->>9y>*O`MOz-x?@6dO9Qq)+~m=qYk*MSCx0m@ag=nVwrfc#(F< zJ=c-)rM2`RiqAt71RLwv_>bC^spA9H))D;_79dfQKhA;NsS5izOS@L+XYNmZ(sy6*N8&_&%cW2L0@l z=bz+evkVk-G-1FbNzT0kb4XQcI*&A=A$9X4f__7^nHRpb0< zTQ?M>_6_Ik8!oW#I-7yRK*HcO3>UOkSx|<7Ra4UQ{n!-uSf9JfpqX$$ClD+)&@tdN zH2`|-`e!wOU$sCEqU!!YLJPKMwgk5-_g8Gx01(;J-L6XhaE3W^U*H9z)Ob_X$#wzN znLwOg(6}oQ z`wJTP2I7o@#{UH3%!0-r0&!MBg9TblnqAOfH5TFq1r3&7A;V%Md z`ZPbTp`Q{o;Cv;?9${HQW2q9|m8o6OdYe+qN7Ktklc9V|d%A09)WDA^txsruJN6eB z_jwyG(@!bNu7bvuN|g8luMd3P1&vPzKHwJuUr#~fuD}QU&%oDP(D-iP1O7PhO)F^p zD)0gS6!`iI8q=qHC&dro)08CsPA_O|9{7NB17CkZ<5__ZxM$#-QP9{g@Bt4Ad@~Cg zM+81#Gw{tSXdDyxfUgdGvkMx>2R`7Ez_&p`|T%&$`TtBC|ztKFl1{?UWhNRcFGHFOOGHj z^U9IZIphM##G@p}D-CnRBW4PDm4z2bMqITw3R=&wYHt8vs3iA@;wl<%3w%H(LCHts zD;ii)3?wimAB~31h6a%JpyZ?RwZI1?5+&c{0chp|(6j(hi}K58AQj{y)huYe$Rf}L z;*4jcHP1|L0?Djw<7;5t@PjA!V<_jxkoiGYY|Z+HWoQV<+9eUaY84IK z@uO)SYR(M-amd}HfVl+t!L&$fsfWgY1c|_EXwpXn(;6Sng=RPtn&Dh%hBKiVCN%i% z92qWX9d0%c1Mysuv@OVjh_n>agUb!M3dA zq`{V~#9%*#(U=0&#L+G7@X78d_|Ks{>C@I4ZL;B_-S$*q!& zg4RcsDvZG2jbo&>=`527JYaTJBCSJ}DiVRO-k{3parpneYU?NTQ~p}ZM)IdR#`+}Tvv!k*P+Ml&=R&B66SSmXq_NyneL#O;@4X9UUd!c?vdpD2=HjpfXArrqjW%pvS zRy0fh`!!6T(@%2-!t*pd;|!jm0oqMJ$90BCdDN#a|42XOx{5X5vLr1)52A$&>7*lb zLEc}-!qmn0!cF|E|W>K3_-!wsODv1!u#RVgOV>Q!GjZBs~)cwrVEkfN- zEdYy<9qdk4uzgj)8WS`CyirMWAnRh}v0nB6*+AG2T62=aJuMV_2 z_}V}_5=>e|-aIg?p6uCx6sj{1yrpHb31q=@Y~WW|Avb|!#IeOw`FFZpAFna$uZ?vt z>mKVG>w>~0mvy9jf$NCv4ntz|Z3sThQy;<*D6AVcggV48>wtiFK^7G1)rQ2SX&rGi zW@;4h-N`<4Tr`+OIdL=R%$fxS@sI-0;uoG zhUC*am4yVnSxN5Cwd6E`*N0fX61*m`0l!~i zi^GR>hPzuu@Mmn?9!br9OZ$FoJZJEe2vc#C?wRZQDHn1ebF;g)bR*+e5Z91YO7b{- z%#+Dy%#zL0?K<`D)la# z0p8eXX$rT~KGxpMO~*4dbI%GH1Y&b8gO6%7XW5!R3fw|T{k79;OV_&BI>zyaBV2Bh zc*iVq-@bh!haJgIs&Phb;h0?3m-UQwL3*)vJpZj*Y|Ib?`>*YV|kV;xmNC8NufcXC;YavzILMirU)~R$4A6)$RrOw@Q;sBai^ej0RFT zP8yi_3ER3DLG?OOVARM{c0lUVv4KBs)i(^J3LV?JSiwYbElP)VMV)=vT>kl1;~ zIDAB9$V_SdEtIxVinW;&=+x9>81;Q~`xqb*k(b1*Syj*q;(4Qp1q4|HShhbvMnB0| z-=;BsrditnVpn&KA6S}QwSU)P59%=8w_+4SJ8WmaucLopN(bYSne8m2X4$A=Ao@CO zXLr;g1aJMv5e=letMFX^87=giF-Qm?N_1tVQ{Aizy5@4FY^ z%`ERt;FcA(trb6`LIYDLq5((*a(f)6V9Vm$0)`?zE!pE7s_d__VND>zBuEqNaUM$c zBz8^SMwcVMl(M8Br&OtX;K@e%f9{%`hR?Qi)$p}#OLL;dstpX``}^C{{c)=0Go*e~ zGk*`|6R)PJvmKI+vt(FyNWiCTvSVl8pb0zG3O4yB^Fr;qv@^@~+Qp77bMm~qE}r}9 zrvwZ5aw81_2W_dnxDLKLu&q&eYardHuy1qkqHLiyQuu$iG*aw7R-r@c6?m1&9zfkU z0||9O_CPFqU*$%iyB>f)VvAdCW^ zZ={955hINP-)N+Tz_W}r3Ve@|76J)P?mCF=FEB#~UaSy<`f8__zi*>`yRl|+VS&HI0-rn)Gl3aF z_Ih*C5_pW+JpjDkNDbf(=JhG%X$$ROStAX8gWz9mUMHtcptsYB;9t(!vNiMOGy)%%oxQEr;p%iUQ5Y||`kYdG|%{abL zZE^KoWhoo|R1{Hp!Xwq0R3C!zati#v>tJxJFAo38{*7F={9 zTu5<j#B}dJyqED46w(^Yzo$`AW=85 zcc{y_64XuQa=%=?yWGMwu#IMVzmReEHS98EB0XQHHq$OMw^G@IsqDd2_FyV|FfM|B zNQ_MO{(sW)3`Qp`M<-4sEa?{4a*Z>R8HA$daxj?lnP*-XAEe#(PMRfjupj^<>K(z@J^X>n& zsrSvac3z}dWx-`v0F#0?Z=o8$wlW4+Qmwg z*(P47EG#$~7N$jJ{a~g%$wdndh?9wjJ*;*@iUW|2Iwl=;Ogw6Hy(Xw%hM`rd?j5^jl z8UkX5do8uQ<%y+1j}19PMg8THhXyR1S}CTZrc#QGBF{4SfT({GDM$L*8Z-B$Gx#>6 z?Q60Bm{yeI`=;XitjbzIN+t8eEgCVNX9Lpg<{!#qJVBT#_tShi4Gr1jznn$`Q=vK% z8itP_*{Q$~dk0YY{8q4YK1sHqe91LM0&AS(1%*!0v zA^m}h^qi=;JSP7B9I5?QbGsMdU~?TspRKTgZ?)2_1MjG?eMRxtD>N|GE%OaX1ajXS zPQMxl|5Lc^l)9g9K(h~G{_qoDOZi$Vj4-=2}yt`>y`5WhVmEnJr(2I9A8q;;l6 zp#iK$A^m%AE7d9TD$^H;J5EE(oFlKo2*e%FNQ=A*BM^5yBP~1EqNxb%CY`*VEe+Uqsy$f1qlJZwk%{D`0SyNvGj z(*9?od%d*r9{F*Je@6YrU-<(NZ_$$bu$A7(bd^3^KQn(O;$Kw|0=sK_6yS%nPp*2$ ztI)UUr~6~S;4xEJe?%b)yK8$DvFYztsUOwPnesK&o;&YV{_X@Cg(^?d*W7c6z85NX zZQ#l`bjLPU1!L=F7&uOFY<6{IY<|ouB^$FGrNwlTe!3`tFHo91z(7mIQZ#b+c;HT_)ehfzuKrDVI-AP0S^3MPgpF=|-=xZh zEINsX-196^u|z{G1riO(Y8r(kThG#LQ73Na3BJtgV`#SO!xYDc4J))Sb1Sx+N;p<}Deo;peb_*jA^PlAEkG>6cHdp5>G_FkJmk_9X z>&R#c;ljmIQWea)`pJw^Xtp?vTVCQwZmSZ|CbQy|N6&?ct^)$f<-Pjz3CH2X(i|zq6+J_ixo!L#*WFD2eL*4nFxm5?^sr>e{hO3*HX?5UMO6sjGUfosD z+D57UQX}OOTtL|+h#)X82qYk}2;-qpp|TnSSuLEUV3x!lw>4%lyhat3M51$?2CO8; zOI=B}!mdtu<`bu_HRi!cZ)5U*jSeWOKT2*}kR)>`xr+c8uY9&E%Q zAz=&BGRqT!$f^9rjf+;&z;HM1!34BqjI6354YR@=#JVRlZ)53C)qPoN=OO|^802r5 zY;RTd@7w!0%KNr}WUl0+5z@E_7_uSq;%pqRbv0)7RR>aflErgZ#^bh8{>mz_-X24kY}UW1#!C_u3!RPr1@CohsL(O;l#( zdK4RtAl#LDz2X;I%SgofYF{Tg!Jr}FpVdYx4zXXeDLn+lU|4A%V!;^#5?WY08H9Ez zOcXOTb18|#(OuiWU>IR=v`*RxqmeN+Jzhc7+K>L&+GcU|D@!!&%+Ap$X5b zeVyb4Q%~P0@lgtVqr^!pV2VgJ0}&?Mn-2zQn3I_ao-wxV>sk_ju+pdlDFum5y4tVR z`gp^RUZPx8f$ufaBH*`_L_-G95ZhXKCl;joDdbd(bEi`TnqrI5$KjAycV$}jNN<^(* z9!7Y?xLG!?0bF6EK_H@>t?Bn~$m+F$O(1!bbr49>oUEp$aIK9U0aleZlZr+u#Ud9t zAev;609KU_nNo^H7=cuZvu>!MMX?AYkV^54v^KW783j@*o{<*SA`*erif5!XV7VU! zQWl<(7U2?!ak9HKxMW2R?|~PDxC4oqTb|u153&2HvO=Fsi#v9gw@VbLB0X-rqtmYH zbAN`AHhSIPg=|VTf+bXlzktjwXp=g~uR#*eq z>0105Tc!tqpHq@MYFsWVXy0b6bs!mo_5He*f5F!D0U()@sqiITpns*GrOPykwHy=@ z-34K~0SRyBYJ9Pp@74!17Ca6JgAR+jN(c6-FmJole5y)hsKF zIrS^u24$J(oTam^lScZPompYb&q9S|ndoF9bkZ0n`t_Ox0`0zv;Ji>!{8uZkrvkB0FeM?6+Sy2^hE6-zil%ji@qrzY@iO6r!u@vz0wFcimQ zJ1c&>O1RAw&j(Vk-L+LU%98SPGzxD+-Lg0um}2az&8H5-a4G??RrWZy)xJ&+|DENSJYjbEmmkOmDbD^mh)-%}8Tofo)Vt|Q zaF~9|^$d8pk($8yMoNB77z=V2c_h7Dhq))C;zFVCl_t6g3@cKy>Tj;A5alXYlvwaL zG$C_rD>Wa;l#|72Ba+KhbF_Gh<(I^bP|o@%Z1EIv(mEOTBrAov?hB1raZUDAT3nW& z$Q7DJ!b`+w=7OhA!Ns*+AuAWm3F4Cu%oupwtv-4sl^b+I4H&2bd*ip}7_6lb*zyp*tt#+n+r6J818D|4{n30mT9{G%RQ35kO z7p1RhB|c==i25NKREkqh%qi+{=2(v%Y(bQ3L(7VB`dSc9X*uEp}b6|sgV@5d%;|MvPR zf8KM8p|O6NxToaU%5Q2E?s|^zk;fV^@pX38JG3^6i*UtZ5fV|k(DstadioPp_j^+} zfVvNux`9;PK&oy4bre|U`_ezFx<#h04qT%q-%3hGlUXaRO&`CCq%ZdtI=NN3Jc5Z9&Yyzn>ndUbw$9Te8 z5^4O(V%!80z>+U^vn0ljmc-cA%#@R3SByc4OgW*(T0AvYjdhwh$OdY5O>H}IO){zU8mz4`tDJwSJWX0B7N9+LWRV7ez z;*<%LP-ATp5~0SbJT=y{HWR2*#s!Kw>rAb?KtW&<&y)_gl)1iKth_j4#X@J|VvCN} z+QxJFS{JvGE+oYL!RGV`@J~we-%HLdXz!t$1X+h(qL6|Ufdui=zy)M?AzTD?rCn+2 z)I0?#T(L`At4l<#<4V|RAfc8V-jbd>A|9^n&DTqws0~lofb3|2X#y+ZTc!bdi*34= ztSV@n7or6uYME$Vpq5^*pP6&cm?a;Vm@6BXTUE<1O*VX?mUA(XO{m z;2uiquf56rdAq*!F*G=s8F@Q0T$!%`U!dCN>F12vBKHT~PwN@$g0xV(HuYU{?dsce zr;8t1**AAb;iC6sGrJTpJk|wevGzP-9@q$1cN_ zIchppWv$T98MU^H-BO7Pq>Hp`Q{N>QITyQgC95uW^nO^m`aPG#4hm+<=DA16abU0j zwoL3|mb}9;S2lK6s+nlZi^_0iz9YfGkBpDqDH0V(pO0d!*c4HEL; zlwT{yag15=p2b|*Af2gK9j;Mc6w4eJhrEIh>)6lR#cT+8tC5<(uNo=2%ZUZK+e0LM zNr&amLmtpRjL$p>_#s`x@g(yzp_)oLOh09-oIaTd)0A(#k(LkL@APo!L_42P?w^(@ z%R7}NYZgqLlzDBmp3J!!YV-DEH(XicVB*OoFNziqFIApnLRwxY=UFH1?&pK667ZQu z+oJ%VYqUKJFz0`{r`>K*{_pCid>I0KOiAvE{&qoY&K}+nVFVt#52QuFNB8q23DN46 zGz;iK_sC3*vC&3ZwFmBv;|6p_G}R{3cB!wbzs;6f#p zuF&|mzz5v+;EG-v`_40Yz}G0Llt<&Czz00=MSf_>N8=nNW@dDn$yS_65p~lF61mh8 z*#f>_NhKeRIeX$l3-}Htm3%aA4tzioU-Ho)^)dy3n}!hIw1Nil7d{}-FXho7{=x^m zKcwQO6*P#y@BxWtDbITu{!ioCh~w!BuVFKz&jQ*24(&zk76J)_A4`i!$fL4AW`k#> z#e@(>AOZ7?w1}B70*RPsq(#t#5lF~9BQ2sPjKG7n96ci~!YYg*oF1h?ZRUd7w4g4w zpf-UNrX1w~|0Qq%DNEr3-ewir1b*L013*&Zd}0cmgESo{?6Fd-6w{(zp+?4y?4;4FL(O{8R`_){Vk{bg#8^(2)4&dGs|x|4lz~7(DFf3*c$D?_6}H}v0*8z= z1Pp;6jSHTtI?q;ufz;;#eCFDeIi>zea-kl|&72{dIV4jY1d`14k=KVT)uTZ2>#A}n zGjoP=GiRvCq>`+XQZJDf(?vW3l2y-0tCC|NbHy{#BDWG4Ah{JKzzNcvW@mu;HQVOb z0FZpR_g#^bBQ&0l7EdQfBe=xS(TLn!y?#1c93)4JEM;rcmKeyPZNz@RbBX3ICW;oZbgv1@kuf;h&;CRXKQDogBmxv|yNk#R0!URe`j zmIu|bKkBCo0(c)SOgR~&mNuT}1-Q2qgSpPRw;5AFO)a3zp&HUDf64P&-7_IkNPDOv zgbRtf7x%UFHvWhGqml)v!QscrE2Mg1=a7qzjdQI(hKe{a6KfJi| znqoS8lxeftsqzOv6E?c1o2&{OuPLUJSZy9;$r}iJ!@H5w2-edG)=3yK5|`50)Z;+h zqSZq6xFaI)ZdNr-s)i&wHWV(flqEHJr%Hc7Ki!`uc55%zxkM7APzM9CAOu{iu2$zDG;a6C&t9n9s*GP1Sov8Us&ZN7kJnYJ+6RsKK3->R z{ioFG?^>cp`gO%bvDE(SRL@fVw3iW|61|KV29$4ETf)nTW7gP`tJDH4a_c#m2uVT4 zVR7p@;XrF{@s4>i%inL*g32u;H%1fR3~rm==1p)?!<+eJFf@>~!fS#H^$6;eZ+&nMWW2TCyGGXw{Q#(0_P35UqY*I;#Eh>qzK_xM^ zrzFPal*CTfaAGVcn!t*It5;?WT)h$lISCdGt5?$CAlOBrUC<(rB`b}P)QOLU*#(G^ znJ}zgNf;=^QV9)mC4-3!)mie<;NVyIfE+`Vd^E_i@O296348e-SwVs4D9OF4WU0Ij zg!bunKGTY$0&9Bw(n@`6rEauRi(8#6X2v<4?j{}Rdg6&Hn`!Bh$vQWgt(7k7kGiOI#r09?qDm-;N*9GeZYy1#B1Yn*Hf~x$gQ+96 z5l6 z+}n$2FIT!;V1cAk?Ac7SK?^zK$ zxPv_>8DKe8fY_9Hp7A4$yKM{Rq4FdK-W zVz#_xh*rhywK=ob=FDE3Gka~$?6olx$Pb-q4oyI*^qfRu+O^y?_Z>2hP-OFs!tx6bVOMpBThKDfgHt z2&2~DWes&?!+*IS&tgxd&e^|sYP^JcYCM%aHJ;0!8qW$(jc2o`#^L&o=8fdYxBW58 zPnG2{{ge$oaE4~LyLRfTc-W)SFy3MkkCk`ucX)lu!&(r(u)=5pDTJKjpGCt^9FOg+ z_!2e!TK#l)&|nhoZim1m-`x(ysduzE@s1X!-O=KtJ6fD_M~f5gXmPq7El$bPG+i&# zPkD$F$d+7w5L*K?g`{#;uk7QZyF%~594DB|hzR7Gd+t21T}VK#yl!NCF?F=~ALbHX zU%_#JJv9!or^Z3{RKm=<;Hhz-Jv9!sr^dnd)G6364}E6r7iz|Sp=RtCYQ}z{X6zSg z#(tq@>=$ZmKMVEFRxSg;Bb3xD8^-MmS`^a^`P0|>ni@#dRP}{0(5SQ7Wu5VYo4DSp zBdNRhXf1j(Xi<$7GvccB1LDH0s)!4;+Se&if3Jl~qOrexmNKr>k{Fk1NsOzsB*sNr z65|>ziE)XR#JECBVjTAuY87I)pN6sFo*G;3sj=yv8r$xvvGJZ7Tkol{`JNivf4P?L z4f-ipR^W|BssryZQgbpv?t+D+PwBASBc}8*V>0`x#q+!}=1OLRFkjPfiGIqG4`hnV z^1cz6>E2gP_Z>HiCwk?Iah)m`60mX+*(jc)PD>mbjC(572+P*^8ewhnRN~A{SiW*H zUdeOanaUSxrt*cFseGYkDqpCX$`>k?U-3$)RDQ)Pp=P`iYMiUrXaSvMg){&>UrB!9 zrbU@mR(eQOo2=U#aY=zj1lCAc23djGh z(k01wRj%xD<(9;_a7$ubwI z*d0%e{qfY;Ay17x^3>QRPmO)@)YvIcjlFul)_+4k<+=nMFj5_OypfXYQY^?_myk59 z!*bUp`j~Q=>(V9jywyy$%ysD^P0p5f@;x8ON+4H^elV+oUYGije2;39chCTt*=~3| zqcg9n!H}mym9WN5R3&&$>F28?p;;ajtLET>tM%(T~-uE?f47fjvfQ0yi*H@=MQHkXyZxG((5wR_{Hu@9!n7 z59k8Vq|PsQA5^bCWdUmgKVzg8@HQj0cR)kf)$O&isb^Wfc zB&M@WIGH=s^}1VfrgKL)e;dW4!@RbPn4x)H%&YJXl zl8iOsd!>pskL0Wg-z!zD3EwMKta&77P5Nb1#+rNV+n9nUg)9r%F5fv>lq@#??_JUQ@9D`=b^_<(N?e0>Ftw+BAp zKLx(&1&zxBAMitgufL#iP2dB5Jn+pZXnZ>G0dEg{GYcAD4t&6G1-@AYjsFgOz=s0g z?1ILR10V3wz_&p`-npAN(s z1&uEU;>?1^e+S~Mg2s;nadtuD_kp-UL4(CijGt4`m}4v3;@rwsr!vTMB}WwULGygh z!zWZr)Z@DfYeJ*ZRxV)aDj1~+UmTBxf&L1Tb+ZU zi!d}SSc_ttQUgv&4Tzb<@L0~P)HfWF$N~RRyL#lLAWE`eRpnzeVSQ+d4PxXrBL7pNfy_8<& z2lcSb(5iHIjo>h>5i8qL*PbJ64N6!J)~>^hC5a*|NRh2+Z{pYSI-V`9tj8Cv>{}vh zX>1gnMQ4q$AWXI+oumYoI`ADPxB+~pl6q<~EocM-lBggmu(V?XU_piK&4G+6wZ-FD zn^=%~YNw`E0@FeB{P4^Xc*>H*2$q)xDZp1KshmkPFe#%ALjp_7TNeI@1r;I%$f#0V z7~v6}VZlo(Fl9-6{rx^(8s{T&LwQT~J5ix>b!YqJd0C5jwANGOlSohH+OzG!UaO}P zZ#?l-<}RzZr^Zp(dS6fJ*K;0ue;9CIBQ4;pbuaz#kZ=J{HqwHd70%SPMz~@(}EI zQ#-4+=T1GVA3dxL@>~=JC+4S4%uk(|A7`zh#{bfs7yuHpP2~yj*?*RQTZsi%{8}x* zXe`Nd(Hra>i0#Y|Oidr?Wcs?re@GoY(kdc(lZ{-yVk3o=TC^~>C^LoSxilh03}pHU zH8X{Unwi2vjjiz#GE-Q1GgDZonJFyP*#16^%{lt%Uf;W{d~vH$G-~cmtMU%Pv95TE zyN(5}1FxCgHUa{rZ6m0aTXh-)?NErI$%+GGvf^afh}dOTt;(t^->P9e3MJq|&7@4I znUo1Nw#-Y&q)d1-DHCcYWkQYZ-!79f|Mq55#x!;z7_4T7cv}Yq!X4h;425{ukl0P@ z$R(iE9y`DaT}gmET9ZkDP%{Y-Y9;|fjV+S|2sM)cp=J^w)Jy_|8r#2LCIS9U&Ln{K z$;C!qDLK}ai?bg>oVkA3kXY+QsukJt8_~}yi9ia8D56kEX!x34Yes=2SFU3O`XIZ} z8;zqZvf>a6iTIL!ApvLVPD8}U@fNN)*g}eZer4iwa&e7WB6`Z1D&$r0W2fout!@&0#De)&3p>o;}vJ7O2z?B@RZDf21js8?quvjn{R_a%-F=mDZC;(|96^k8k>E(F!^s& z9I;L15)hkI664Y+i3FC5M@fteqa?<4Q4-@wuhIxVl_Hd}N%nVfh$NRnjbr4gagaPU zj*_RwVe-^CPG@VJ-lm^3YV&!6JEZq%*QUNJdJ8$d*D6=P=W362K_Lax%AU-`avHmF zEljMEoKCEg9J_(7=*&D^2@B>(JZ{M@UP8@8U#OT!iFyf{=nHQq`aj?hv0-ROO6Hpi zSWqAyN?m5+Db!3ng_?<{P&4rqY9^jS?PNAKmBksGR1#y0N@8qKNsR3&iLp5)F}8-Q z=P`OG?@@GbtBuqIDy054aPJbZM; zs=k>QwYDiBtu^iu;A2Kw6vr|3t(EhwRp8rynQtv14#|Bc5O17st%4Q~2_q2GWT$@B zs>zLOWalbL+8bB0U-`TD{Lp@pO{pgE1|tmuzi6c9eF}eVq(R{0Mrs1j zwAnQX{IZdnz#l8AtR)&uP;m@M0=sI1ac!kR8zpGpuyPv(5=VK-?@$~_NBhf4|FY)d zB8$W*kcCQSR+|Lrb5J<7~)K>Ns zG%g8zRL*||5|#5vAW=Dw1rm@7DisWCEJAn|0tu6t3;d$Rbs>-_2^a8Jfh$h;{aP_0 zI#Zo)og_VEHP`|^Vx&6oCq^0qVnVk(N*aX4%KlRUo+1T&~>woY690P$xZ$EAHyby9xV9&R|>;GT!;{Ty@m*jXkK17O4j`6H6OvI!dN9T@9{cBSdjzOi0BeZacR1V}86 zSSJD@5tMD!aC?ubyw|yVrhI8WSN;pTiOXe6>urdl4kQGt5fE3?m znT0}@MY`u`A&M{{UUnEsflv)oTM*(5%czW?xPH=6gXySb8DZ3e8rJEuG3awo=0hMx z;xyu@aUlV5(1uTL-6-N1oB1w%GmSW5mdzrNOS@dsV_l4Trsb;vWVGykdpZrgbX#cDE0)~kl~9i`NV^_^)nP>gbv1FS}AN_B}W!n~em z)nZNP2d$c0!0VLcr%*f{8F-NmYycUUStHgc_cv@*8~DA7x^_}{47|ezwt&@vt&Rh~ zYy%rWG|QI)z^bf9M_F*o?gGd_Iao{nR@7QmJ@arML`flV8zYSXw^fo~7-@|fV;gv^ zkw$>$7^w|>!x1PQ0lra5UIVSS7-JiFxsgVIR~V@cyx&M8z=w_02L9YgBfvixsU25W z8nPB4BO<;mX%Q63C6J)V>;qnI@o53q8fg$XU+wZ5XuaO79Rw0O&q#~ti7CK$gt0&Z z>c`R|rZN_Ibr=gI%zi8_A}wQqL|Q@s3_)+jL0`96F1Nt7rsnz7eEbV~GttZm-n3J1 zY(kJc+0Lcj*n}WW!j*dS1SdqVMw?k6YX~MY(}e{5spVz__(vu6)s~kp_NPsG%k~|} zIwAH0iHWPYc0m~4ZKXMY&-VmUTwWO$@Mhy+?*v|9%%paa8)3vQ5+S6??P_7f>b43Zv{0|us4pQUl=Cvf~8JO1Co6| zmKKIj*#|!yHwe3@#-9A$7|6V;OtHUEF}_7B?mVl!Ch&YC)qxiosSUheN!_*Oy>Yx_ z;nU=M(#-cJV=4ks?iSPi^{-gmyOr^pWoXY>mBQ5 zd0|>uM_g6b5nC45bYJb4OOEQU)ndueW5Lin$z|rttN=3U<3#}Pgf-SKq*$?#I_YG= zju-G3*-9~RbQDop)6tn(sy`P7#yWeo3+MhCexj3rkpn!S8lcCagWk!PN}Le6Zb8{p zgR+NyPLp?7$hjL$jsDkOqNw z8>tN(dWCU$5dKf|*dmXl$?tw>z0Z=?0v=+)83hs~KbBUoBl*%SwWE=I-Zf%(=9L!AffP#w9W{OKnln+ z(pqFy(Ew6Io{`q2fe}dbJ!2>VkJ1D*a|vi>63`?8k7`cj80knbX>eMbmXzmMteZg6 z;2CLMYq4$uiLPg)MS$g03rKW5BdygI>n4!sdPZ6&S_qp!qU#xHeIPIb&(%rj8EG*e z#YSK?Dd~}Tnv|hjQid`~86qjw1PoaMmG{MavZg;t>B5%6R#L}LZwWroxbLp83n1@Q)h*eaR)1vjHQ^RWbDEMB~NC8Y#U*`E~OqP6Q#(KFgm|8L3iYM^iDb} z87=n!@kk-XqxRHMYxQ$lZJ9g5h}}{$0w2;@tmIua!Tac^l4D(mB^C}Nkja*LOAe!6 zucPH}001{uGx}<$l?N)cvEIGs(4Un+Avlac|IZ>xw7b=#APEWP)fX1yk4vbD=>!k2^6@}m+-L_Ixhm#K8+OBUyz@3!j*uWH$Y|MZxn2xQJ zVN3$Y0}h7Sw*8MbF?KHun_*3IH?v~chKiqFp@DHkZVZ3~F=y@-3M|Q2`j6;nZfi43-88*Z;)cjAnp^?7U2wEG;_*01N<1u+rx~hk zW4k|V0XH>L9k_*&hJYhV%Kbzo7T)0}DqE|F*XU=buUfX#ej+1tGW9U|Mb}n(pUCZ? zeEE5m-cxX!=DrWrl4d(6Bub8ra^_ppN7W*9Ou)ivg5&_mD0eHE-n1hdZdnF_)q&{) zIt+v>^B^NE2$W4eFpNs>Tewf=HaO!gEZjefw?Ibay=5TBQQ|GIIxzK?fpBHKeUqk} zw9Dh#@#KN!{dNX30^|-~mcj?FO73162$whvL|xut26C(>4g;$LQ->J{SH@vVeJk}w zz7=}od*z!y>YZYT9!Db{@BVahJT~{|lo`i?m;?ieDLxpnuzZNl)tSp!2~j*yNU`y+ z)T_uv`YE*y+*bFM@|7TP2PHWR+6Apg0%IrfTNz6W$Y>|4WwPEH7=c$9X%L7;KX!7Z zjhxa(Mri}3SDMlhU{z^aXuInG1W2rbSS>365Si|q*;C|EY})9^EEcJP77G5uj2#5t zXQU<&+45H=?@XuW3w6M$HZZA`lqI>>PFa#)1)%`r<>lDD1r6dOKP2fbXkd7stCStn zH%FhRCSPQx*MSVpRQ{)xn~2HA21vwYMgkF(58hgpb(F?TW)W~34S-}6xTBK%ETXly zF}8s>YJ6oZ@E#+zfzQxOPo`$D>qN7w1#IgkR2D=amiSpk3u9zg2Yi8~;zo1uaaMLI7gAT;PD%?Z?u>bQue*dXcUL zc(JjCy$QrhsWu=cI;BlB_#lh%AP{>zBP~o4iNLCL=~ij6TD5@%v<}?Zln%rpk*h!~ zUP$rM+ov)SC_8rGla@`q5ah!U+v%yXsh%2J8%vZK6pIs5EGsk5VnOcyF_z?Lv8ddA zW-Lo4Xe>%dv82qt20;&7o|8J4(nrvpdiN!ydlgbfnH?aq$%T|cYUET3<~3^E2hGUV zM-&FTTgX1l>a%sT!Y>BWw-nwNNO<>CTQ*z2QP^t^w{ZA}ffScYYG*sOvyGicm<4U% z%al~ERTzJ3-~--O#Pa1@5{nm7EL}*ka3OWF{ry%5qp9rCRQ6~pJ2^wb?)RGPg{ka?sqBTR z?Bpi*G?lqRKV{-Rwla3^W!kya45$M`;V+6u9iyeaNI$cp_t4Hkn+J8^8;q2QzFNC( z(9f*sv$gXolT!!YVx&a$f9mAU6Ihf->A<^=@%Bgo1HY*xw|8h4v}Wl$j5C#v_}xr=3;1;tKMF*k z6W_{+Pl_G!m@L~-AVxV^>2@?7*vt(~cZ%u2q1?dqd5d)5Fa!VHBC!xyjl^&!5=k|z zPgyTcYC_99)aU*Qiv|9x=FJFRkc-SVF%>k5D_<}{0|>fyn35d$tim8$i+v&^*K2KK zC)F$aO(6D8xscD)cwQFf$!U5Jy<9)#hViWm|0$5(p<#HBMMjt15QbECJC)r|Ww%q= zx)KK2A5nYH)m+IFceVGcbqsg3FROXK)=x)!TrnARwD?~vU-sRxcp=5og%k_VY}k=K z-Rf*U@MtB;rQ)JhD7?<1J|FmVC6#>r>dy}PSrTdNrbKy;0l1{fhYQ~ed_ZDW%1ix9 zULcQOr<-3RxIju|BBBuKLP}*Pzw$!%sU~|Ml|7Kk9!O;;)AewbIWG8dLCjI6XPN+}ybPvi0xi@s5r?yx-9fqybmBKjmv*tl%QqW4v%ms)n~z&9Hy5j~tP3e2;x zHnv!69yIq`BWmt(W_Bx`@l#w)SZB$pEJLw(sh#cA&Ng;FY<9MRKUR`d_!X-#{#Y}< z4Sb!FO1{*iDXw9uN6E65der*7CW9qHCNgkb!KR+C*9y$_-5S(a%W&{e%Zj(bQDq6oa#!no6c*=>(D~ zDK;Q!nsU;%Od_#fOK#6Zd~z2sL1aE)RXKoTR8r8FrJ6`mYOdubS6WRXcO(y8orD!ZM^Zl|)7 z%jNsj-eWaa?mX>uwT|JAwnWW)mwr0hWs1pQ?kp~rFPr&TypUq)LW+fF?gEf~wAI-_ zDtjQ6J&?*yE-lD@naLhWWsjt?M^f1u&9G$es4{oaPr1}yx+-*g^gB(=Tr;2!JkUtV zzVkX&_A&j;iatj>uhmbvJ_BzxQX+bD&GvTsnHBvD9shJ?64AhaFj6A=@O1J}k6{gI zv4(ui+;8ow>h?3UpZKg}JtC)>-EwMYJGHZoou4#2+rZB#Nn&@!DvS^7!xZOmYH2d9 zQ%hSnXplqoEKH4GuR%^H%tDrw%4(N-(;@}$vXYs~a`1Ky?wG~BPKa49Gdt0LQ|_;Z zQSns#qIPX>Ri+>?!h*ugK5!Eq6j+j-=o$~#?w?^TlDiPE}IXO}Pn z->D?eNb4$NYy&@MBn81iOxwZiXaO%Z(n27{_ze-Q^G*ChAjWt`TG%5~5Ex8TZ4;T+ zz_gXBQRds_t7xFb>1-6VP$c_jAUey@L7zuo(HwnKKd043#`?!LqIsW^`yF{uditUK zOnYW%&*)e`q+e>+Y&lLF>%UkAUA)m)Kh$@r=aqqwF43-;PNo|tLZSV$_RTV((g~5& zjMoVD91UMZBcyY+tEv&&+qG|;Mu^0Fyhf;(rWzr=zp7Cp>OuqU%5fUC6C&vxuMsLq zuJ{)6Vf~a<1bDHL27n(mQXNR;$@V!`o7qvnP6Z%PSb!u(Ecm^0ZKtkJP4;HW z9Y)nti;{H-?us9AjP6l$ejL1yS#D9?Up9-|z;j>endKxCc%G8{qDAYBR~e%R;eW)> z)`7Bk0M9YfAP|9m;h`18C->Q3Q>(wBpYrF7`*$zg#D>D)zS`JEt|yr05fZR!JsWuY zUpBm(Xbg|CSk{4WHPQfqyv#^-AmPbW{37LEZKDS6Q}~BKdX?sW0zr=^qw49Xq`c8F z--;`_9r&~g{)(BCNLj6;-)>Ucz(ZbLEB{G(0}oY_FK=4k4UB$7_#g3KG4UzaI2JlZt8U}7kSrl)WLanil= zCHfaBNekH#+P)_iv}`L9SIHX<41O~0X=f5 zLpTC5S3fLgRA22)9X$8+$+Y^Pp~s5zKTb#Y=dFr%FTmI%wTv)o!X0FTgJ;Pl6X_J8 z8wL*2TN6p=`XFlwkdkxK;#zaGwAhj_3^0r-;D&*NVNDEMrgiVwz{~|Hc;Lzk+iJyU zS7=}yk)c3>=rqLneV^v$Q#OSgzx5H22->j>M0r`+aYSGAcHq_VDKvs4o0V3Wr2H75^8k#u`&5VX- zPDA=EK&oLVr(r0gVJN3zD4Bk#hG8_EYN;9ok{lP0;Y=kY_q}%{h4vQ_|KF;{Er6JR zmYKf@v&pm^-UEq}8wn=s!WO3-e{u>DDd*X%SJ6ObPJYSQNds=Ux1(s?BP+B2`9c!s zW;!Z)O5qd9D4QZ@TJ#pC(Oa0xPNq!e3uV-?7RV9~4Bn0cPcjQeQVT{>3v5j|%NC}Q z3l)ZpCSNUK6l&M=Qb@tzXvJsNKlw5Xh@RCYU+-A-jEcT`8Hag3G;`35zP(L%aRjr+VAx#&v@L*A2_9Lp{z zV=TQS#_~&ItRXv>(eXhG@_gWrlqBQMU4`1^7Ww(Wo0U}ZeOnv9s1j-XISdBQ(=e2S zapMLH(|jP2EBRv2e z%=PbT(8mQosP^bX%JdyeS{+ymaldF)JnlyA9G9_}X@xR<)V!+$Nv1^E7?&Nh87l{ zXvVi#y-Bucj9pGmZ>OfWQ`6g+exsS*27X3K600j#rT*mBGW;1Zf7-xflvGMfoouI0 zCd**zWGi*@iRWj57CK9lOnBh9iNY#exjx``C67#27BGV{=bqybrzO0DV2k*s+0bv_(xSI{~YM1{y*TU?#zDR-DMgK%?=%sf+_ zjv7cuC10VXEXig(Wl5x9>6JDul0_62{Lh?8{j-h^EJztSFNCG(j zKcA1j->F-tPSvmKoO|!Nw{H|)5Jlc6Z(XieQ`(vBeF!N06DAh()3jrgW#wo0y zXb`x{L`~o~O*A;LG<0~j{$@fuSBKfhdNqL<%!Ib5YNINKfm`Txin{Szb7u(n%u_8i z;0Y=U^MuwnPBTT|cU2S=Y5gH8h92?%@nf_3(E%QBek=kqa=6u@^&K<62*}7mkyi90 z+4xEQXyM1R&5uzaBbVk|1+8zHAEQ7<4vMs*AIU=bI^|(+8*~`+?}(zvUqHiph+*Za zJCHG_>;bl*QtmW?bzAZWqNGS8434$BxxsS@?j@i`@pmFOF)!m)6w4r$;ql&4~f;#d)fJy>%M` zK1T2MWC8)7prSxBt&!IUtC<&+G5-lO-vRDp=0|}TEX{WcTG4#6+Fs}cKsjjGL{8oiBGH~}-1-)R9)(ccBiog#} zP^5M1zgjH7172MlnZ^bcib3EVuc_(Lc=v0~9`M5|3Lerr>TFX4Ua6ugWEwNhF+Ctz ztMtOr`2W{vzBb0Z0c-s>j>jGWeO2|pYyFx)k}TtG8uz)agyiwdn#W4$lU50Rf-0d; zMkVyjS3*y7C4|!5B7d94AF2(=tu#HUm9QHV&?XmFzz3fhl%?OsaT?GgH!hHKnaz+v z;FTuoT%~YdO@ed;o};2_02)_EJzygey>T4&*V`-_9MXxhT7L&hhx^Zx=cP#AW>aYp z_!SjZgOjL7+e{h+GNURza`)4yN9NA9IWxFkA#*2lllNOKJZP135)gJ|!0X3p9B!-( z0c!<*y&jFp#Xjj0V|V1L`3~drZIR?V#xvD?$2cU_7`bB{Mp#d(f#3uJDq*IY?--|= z`(icUk*nrA#xvENek6*p;WAI_wOgW{~F7?0^X{iV1gjMQ+sm4+d z%rq`0^KAue0>5jbF`W2>ts_m~)h1dD9I(^&*ipK)G;OsR z1I7ht@e|zXAoDo!-RewqYB9E-ZMGK!8Kb{3F7I~DE@*dDwO{5!Dfa~3K`<7?~%)X|F^JQ2}tDtglELVzL~ zRQU2Jx>OR&$SrK@!D#BiXzIae>cMF0L2@yiaxmyU=xrQQco%vb2NvFk@?}Px z7$Ww@s;hcvL~aJXXT6PslX^;B8^X2US&0qd+4**PI^-Rb69axl7lv@*e=K7|K0TS? zyvj!PBzn~LD%T|{9k25Hr0kMgZQ`U1%M5rkUMEv=sY+53sMHws1D&O0N0lV%S0*Y+ z)NfN2Ldxt8W>V>WnXUPYlQHpSuS6yBWeU~sQ?(KcCK>3aL?s#MZ>kbkg(TI!@T(P# z^TK^#m0aw7kyi`+dMzTqT8oHV7uISK`ITBkKEs|5xx~hUCAJU@;=!)A5DenM3C#sZ5`Jlnk99M2 zs2LiWh+c_(T7T=ukX_5wTB8Z1{OamH|Mg?O3hKw;{3vC!e#}=y{g|iRhbe2}#&P1q z;Y#Q|t%TmiO6Yy7gx;x2=)HNKUi-gbc{m~)C|9ovaJJpFw}9Nw4>V5My?^H12d0f7s-j8*j&SUN}GY?BQ>FwX6R}#b`Z{_KF=uA^t?6*TQCuVvLYOjID zs_Ps5r^#L%k_Bcjxe%iBv8qzuZk{f;m9(E@UB`e}lRdvX#`{ZUBrj5BSxV%H?AypH z*eyo-NaBEWmr0-ACNg^DW-R3t;@K)?;lkwm*}rB zAgleUqKV{68>bE2lqh#Ii)Q9L!^pKbx>>ne$D7_*b*N-#PJ!T?=36pV@HMKmflNn9 z%9_+6o|3y9ZvGJXGNfkeMrkde>3e|%OlP>GckXClA@D8xnFJ)t~_CqP- z;#s;z@p7`L0C{2{FIWP(G7vAxl^MLn6MBWb(rCtnrFGS1xAfj#_z{-7;pjxsvj(duP+KuC$q)M35)bun} zdY-6C*o~^IG-+~S28`^Dr9+N+L+?@p7n`9b5JTd?wc|8u!Eeq5zd0BD=5g;yWiu20 z=J8DUn^|VfdA!D}p2kApfp&2{>~kYaug{E#ylZ01J0@D*t;{n1*?WBg;la1nlk%|^ zywnnRb1rc==Mr~wlDN{Or$>@`b1rW;Cwap`-lXW_;HI2|n{p0rnyQ1Fat?0l>L9Gi z5|q|Q^1eG^(W{!WX}{!dk0qC>o1ev4C4-`sR`%kwvKOZwjQP}wt+cWi+dNZ%pR5dW zG6UjNHKDy*mC!p@3B7BT&^uQNy?d3gn}bkwpIQ%as`8vhjkkMo9sPqY6o;<~jLPMh z&!LFOU_)OC6VWDxH$@TId#df7b}m-<=_q=rZi>&f&GF7L3ZD~2N9c=z&$1T;ldXf) z^`X@DAzWW#eh%UK&!Y(V8x={WHmpwL89F|RXNY+AG3SSXk5f@)Ee&!g4Klf=PlN0{ zS5x)hma?%l?roYfW+0);(*)p_vpAvz5U6wn))EWk;xzNH;{FA}zHube=EKQ_?EvrYeaQJiQ>EbFLkyajBh~TELBUy%=~v z7=x+I-O1;*e^g0Ge7vUW2uoR_!oa&Lko|MuDJGf^)J@ffXaQf^6V!Sd^MR+TsIRfS zr=YPm3Re|3XyY@z)>#Kje(*-V_XNZnxt0gwjc5U*?d0;FK_0C@z9<5Gyo#nZ&TKrQ zpv5rdJMkk0tvJF67WutE!KL~ue-Ht@!bF3>Z<}a%S7l_jF|!ExSQ90+=Z7nA!x#_a z4Gd4F zC7GXGCAl&wDxoUDnqTa@GW{bJ$EoT|P_G{IUJY!Y{kgT^+n`T-8RdkH0bSR=YqmpoT zQti3cYZMp_4W6%6d4bK9QQ*ZUN>+DoyPOu@csYHrj+?pZbBOvEooatvM~>ansq!q< z88k1cC8O{8dI&Evm6NI@m6NI@R4+{iN*ME2vQ>heWoj>036gadZBiwiFSamGQk5jy zR3(Y_gPO3%UZ*0NC;AxClF^{Rx8R>w$&$qIAJwr28cTI5A-TdFOG@I?dT0wy zq$&wXsY*i9H*}z%nJ){0e^rr?G#=>*=+epk^D0@A82;O8FOr1h=jK>K5*Hg=ASqQz zNJ>=_lK!MVOtY$92xM#_Nq1->RX>@WB(IVsiQ#WgNJ4@ane*^f>hqkw;Fw(D0S`6N zFz_f9m1parg4SiGI0W2z8s>(9Cz&X@y~c@IW~c)^+(e^5hAo}wm=mL>*a0%Kyoe86 zXijv1(XY{%n}G1^QD(aZtUJ*%CoYPLz%Qvt{xRcY!FW?f&?;y}M_Q9R(#|>3Hb)pq zG7j8mz6}Byx%8u*@gtd1DIp{Haip0a1J*ej$+$LxYxp6%3cyH4a*e(5zEH9_F-yZP zzu#KED@B`0RLWP1oHjG5zp<>6)26tmmouyEfB+H)$^%mHbBsvJoS|tm2N{|+)G{<} zCIWV7lLGAcqJ#n}ps#V}w1NgT(AOBBR?r}peu-sduW1$ubx>KLfqRt&8u(OcdWU|d zh0&)uxqst~#=4%hpIE)FXOaAVQhqt~WL25oSl27Z*7e}p@!k7!2w$u(g57m-$&AY( z{I|Z{V!*-v#;WSO*7%13%)dF6lqM%4C4w_uNFep*TBnL8#?CStq7El`)Z6{af z*rvB!z5?kj%mAaI`Z5w6ghi=B}~s*}#|Q@f)! zo04n&R3)K4RY`b8<*OD*!XGO5lBMbr?SHxc4m6f4RY=4O540lct8bTvOFBA z{hykIhUA$x!hGOyiL&P*y-LU}svFf#Jh>-v_URSTUUp@)=wk|FuSI@%dd*hV@)S+C z&(&mLOyy47`>Otd`kOf?zef*!&{P%z8MU{OK0r7^)lai-$s(GnB*l`dBqXBpLK`S4 z7F6&hTP!W@Kdis;Y>#AtjW8cLmMHsdZB&g%4}>FfB?vsyM8m+Rs;In(9V%#j!4!vp zkDGzHVc;4QC0CI+@gOtQ0Um9lQ6R&XPISzP7nouP$jDMPz!m012N?ZIuJ`ck@n*XP ztUJ*%CoYYOz>O*@H!{I^6Yd3QMMsiTe(Fd&=SbTeVI;X}0RF&y8w4_P=|{Vu75zvy zGE+ZB@Z%UWzX(|8Xe8s>2(B@*+ynrl6Un^%xt1&=ON_w!h})fv)Q@|qI&kEVnu4$= zLTjg3@<7N9ihfDhMa?h;h#* zu(Mi4B~Krd<1<$4L%5a7?81FxLc7cGngIQXSH5OXHV^ma`rSbfz2ZXgsj|QyS@CaZd3?B&$D5ayF zqoZ9w$8@8k4Xlrl{Mr;HKl^Y!3M|KL{N=7dkrvZ~?U?T-B_>kVfkHzY` zWSqvV2L_X&H?7C+RvU=M4G*fT(rP}Wrb^?zDhyU>y-F*&a-T+>we+8&dySU$XRN&c zF{kNU)&C3hcR-$e%B<{9|LsdODURx49u2CGXN0sVkV=^b1tL@7lJS0dqKJSR!3u2( zrLsbULNO}{s1>Zxrd+H80xE_Mv?&_vfPlK818qvjIv}8y=nyJ1{{P36edv^T9bxba zo6m!uj?5d87bk{>G4ukRqjEmR5R8e)lO!T&`1pf^qas?ndw`$OUlHA^5N<>S9lzGe zCZgx+41*96#pnaNEOC7l*Z*p+kEX7Vrml~EO_zxq^jB`dh$i|uil5P`Rsm4;?$|dsZ)zn4;H5$B!5KVJ&=FZzQ3SO@TIJ))?WR#L~|mnlxZ;i!DPyzyv)3C(&a8IEC!-;nn>RN{#QiLmd2_Q=cmtk{Sv3QBI+7tg zA$|x?NjiilB^hGsv>dR7KZ3fiw-dvEpoTZ z`H+Any8-6Hv%DP`vdd6>mEJ&{lXKNY<21N@3cYD@=9k_;E;NH8t=;vuMHKy|K~UsU zg{uTn1aip|6enY5;-vK?O*{&9tP8QeCssoL&S6wHr4c{-e&MiOc>!nWZiC$U0jV0% z0>b!2uaH8$_X;^la9$xMAu=A2Y`7kXxssF~KrSj>51&JKI8A19{pos4met5XSRH*{ z{c)e!bY&vvBr#c;I52M6T?{FK=SB)Lrxx3-#9`l6W=&MVr5hy$xQli zfYeGKAX!fO_&g^HX{M9w{G`v+a~+n`?z!ID6`hnumvp)1z(*|C!@z&BTz7yZSZ?fq zHCD|)qX}4_bM94mx+_D{*Zq;L&lHgvgGEIY9w8}IM?#n7o8mv^6 zCJjzF<^%$6W^lsSir<5ip{8{qXIhS&X+?&d@h;@--;JDU$qjDhM~0m7B16u!B16u2 zktau-$dfZ|(#1+H2L0`Ac}@HOpEgXWDxT!9mmxd9ls*~BYzEVda*_h*PipHQdN+^= zceE!!Yp)tFXvG_>4v@Jk&t64w0>{#LqVBaHZhP)c;4@V;J?t^1f01#B>&v5jZ(wH( z*OP5Ie0?)6!Rrf1YDt_t|YK#(4oJvr24ZI+8CIG1j@Yj z$bvHGawT)cCn8Sv_s}B?{A4Z)$D86BxmPSok{YHWOnU<*)0KgI*?3xWW#e_EY%;-I ztuF(7Lw_Y0;4f5EmQ6O8ue!bj;|)v_%r(kM?XlnQ)G~LLvU`vgL~`#hT0U3Ut`)?F z)dkha&5ckaQU;wL%S5EKUKq;+SSu4z^h5XRo#D>*G_whOxQa?<(`5A5u}*VAi_61mg`%63kPSlUgd}LL-w( zkrFW1D-x7QX>rLSpZf$-T|tqPTrI)U-E;C_eF*;;dl=sa?rWk2!2MKIp18|@1cUYg zrrPc{04B=xf$3hpTn}!Vsirs8PRZMpB~PS~$9Yva5h*QBtwJ8KMxH1_9*adt+h&@LMJt1b$aVUdKxOX**eU4}#o{hXhK6Oq#56eaWnYxIjE^siNKLiY^)qRVR}HTp%$ z_F9d8k@}GM3f)N-^#b5On`jVtvWlRe_DfAQp`QUTQPK~ldr3c-faMJ4eO;B(zadM% zNTHu~U-%O#EtYre&@Z~XX`Mese}isITeYN!ZYC)r^@09Om)hUk zGQ9wJtBD4Ix2uSx(7w}DlcX>JCd#CM>0TxU3=Nt0qS*yMNjG67dYR>!^<1PRrKxWn z1>GguVzW6&;rh9{wgl-aZB1N%Jbz!`HArsJUwPLUxY>6SIb^pmM9U{B(?;H^`Ye%n{@;z$*ax57LFRehA~ThI%i8)bkglp1>gW3nW#~KyjS3c;AtG_mTSukozDq>6LS@_d=2*Qtv~MdJlrsj~}FdbjEqT5jC}QFa2O? zd!+aI>eGvo^debmUQao_1d8LN#Z!yilZ)Kbi`)~ONw1uHy%&-kk$NA3)O!%5e*7Ty zqchGGM%2{Kz4U{n?UCNMs84T8(u?H%=Jk}*OQ1MTT0FJLJ-Ns|y~sVmne@uJ*Lxw! z5vlhfNWBL^>c?vk z(~H~_oJp^od%YKu9Fck-g4BBuq<;J$^`kS+OndEfYUf`1!P53f@9mnrJCpPxnQmTB zIlTmm6Npr z_adA*{R$GK{`3%}{^Ss({?x!YUooPlcJ8GgENzeUenx$|EJ-httIg{vFU|yt( z8Rz9j zRM*Fz6E-F3w>{H)y!zC%^zuntBqy5JE#O$9JXyJhv(~?8KLQHUZGSd=tB!Mpxt>%e zDxXl5@{u+8GLy|Kz_=`*8TGM#N#C6Ovi_Fetyt1{=qZhdCQsa7q$;&{YUCmz^X`jC z{V>bZzH+Xi?|b7rHX`+X-;nl|g^a%IV_%W_zVA)@$}I6(SE-HL^;dWTKI~yZSsr-< z4^dJ0BXn9njEcafzB*d{^*oJR9~lgW-n7o2W5WR7yjKuay=naLarNE-$Nwjen=OtG zkd9(4{rDXHh)`Y#0uo;7ZmXbmnFZSd5_I+THyT6AK=@-a`S=_H;tpnMJ;PXN0moES znWeEJ>H%S`(xU-uHar5@2*XDL14`*==jdn~9nUm6+Q9k<>Br~N5k_(&jMxYeDt|Wy z{N2NYx02CI#%aK08Q4fc3qGVb@EftZE*YnBz}~@R=uPW{N7V+R(evoKDy=gfTT`X+ zH5CS{v_7PjT)9u9&RY8Mxi9`NnWooi9KX=t?jN5+lj5kp1V@9($n^(p3Zzn|L4n9r z2p^wAK#gF9Hic4Ip+TXT6$I1@R%lZ$)&T((LkHRvjdegk-OzzHrDGisP)l?Ol^Os4 zcgj9=%3l(~;NNUM4|+N>e{cB5#LzH?F3>qD59%=lVVHsl};Ni&GC4ryeZ!w0IAyzaYgOd@1Xy{8St5%hc^k znFiBO{@fJEDNq)^4~)|w8jRBBG*}s>!DPyzzazjf8qBePcYfSdrU!D-ZIO77!x zo({Q-l0R)JKUL;qk4HQKeEtmA*N1C_?l05 zTJ#e+-DwF1y3@j!;kwhJLv*Jl4AGqyhUiX<4lx-ma$D>J6_Jlk&Ndf~6Al-i{L!d2v=#rNq=Dx zr0E6Ee1EwS6#W%PkT%?Z{wzCj)_ayG9`%IB2{LiPfn~QQ!1&iJ38TQ3mV}i(1&t>d zwJUoI8Xt(lzJkVYqcBav#29=&d<;G{E(V_?ci1P(9rlTHhkXX!VVB8L@nzrXnZ6J`wAMn=>}F`V_BL~ABj(D z8i`M95{XZ2qTo}TDEQDw!qgKSR^ILj-rB`KO|Y)a93av1=SaZ0nqc{pd*Jsh z=c{`P8n3hTuI?>pEVT5l?ki|;rjbE_WHx4eYfn0?v5{>JjRRy}Mg_8cko4hz6+@$A zP;W?+KHcj=9pS!}`VdEe+`lG9eW`GQ+f^rG`I)<=@|ipG_ZF3xd-&keo8zsHT4n(DV znclo{9<4rbM}sXF*)R-W*+W1HV45~JrIl$K+>2JGX>bEtY0_Y!tTbtG)-xv%aMy+t zzE=EZqYO2z3pvwrPiA@1|9gy9&HRvLIP!8dp1LvbGafd%meO4?=kKOMD&uj~=gM$Lb&-n>rW!dLD~^qKZD{gq&VXRD~|SN2YIeF?@Jm?W52Dkrt4 zXPmrZac$S9XLwL68+k7VCteWCMx?Zs#Iga_$|mz#&1W^tFY2!_4g8*px_+4MKV4s# z_6AC(xA2ucB7)Xj*?1i(n@lh_>01wf&|e7#I9;DPug^7W&o}R{ zLvv3jH}fn&+_#Cy-(_UJvWM+jL2Ou^+&&65B4yCku}nlt>y5EYfVDCaML!{r(XdXi z$F@!2A{BN0Fx|6VUzqg0Z(gCSbYl_ohCc(!VCV42cx_IX?*}BBjMCO6WHj=@&)l=gpbW zJww0f^1e)sevz`fSEFB~KBTjhxU(aHz;jJB2s}?k&`*24sV4L@047TM!E`U_2NSTI z!Mv|mru1*f(l1izXWbY6L`sXLU+6a&=@&)lKUM<^-81xy?vYyOFV^3ndzQ9pNfF&l zQbg(louQ*XIMNE7ZK6ToBUD6EXzyjJNm3X96J=7sbT5+vhK9_XU=Nu6U5S3uCFxYX zE(p5owZ&d{kizxrb!`dK3$-}qHca90y81U}S6izYKv@`R-aX}6Lm zFn3&;Z_2$)cbIEDt%=2Zje0CmiZsKBNHdIx^wN|Ok@`T+Qu}Y$U-{A;@LeWa2)w{V z2_xQ6$q1G5Nt5;<& z97O8J3c7xzAob&1sX5)Kzp}srH>;?(vE27+c=xJ5lk;3uLhoE9^uAR>KM1Z}t$qwN zmiC0NS|hp6oKDXCILPdUXI?S5r}Ky(k_B09y%$01`3q7{V32wSgVa-alM=MUBf=3m z9RMGxqF%Y*_5XSDs`OA6Fjon^%azc3TnW9ymCz52zyD<^U!o~T(r1w*m+jtbQqC&~ zIF8)=j@-MC+(&@i2jTsY)U1)07BnQh2vYAuka`b-)Q=yee)I<_K?fO8ZD7k%zRdTs z2~C|=>ABPSTnW9)mC$=!3BALW(2tD2yQ_2ZKCPx4$pIG0lvB@Yb z)v;pAku1_9nU6$!s!0j&>ftzYPd9Q;IC4)pa!;T4LsBEBV(&$edLM$+dl004{2=wC zzg!7=oe|XrZcve&wtIXx?{t;j5$ynT+au-pyHv-DDM#`ei)6~PoHqk;oK$(bk$b|C zd&-ed#-5}`&gI^VAoV^3srMjA{rExZNB?&v=+j128~AM%$+>)LDPN-;vkNbDo;0ZPdIW|$JV^2~e=W_2wka{12)O!%5e*7TyqaUjTooqz4 zfv;APoXe+{^8K{~%x#a9bQ$CO?<73(ZHn&F^ z5W@-`E0zJ0f7c^hG4o&N94CD~1;~98kozyg1zIlr%VKIhNx*Q$-}nR5KSQOAlYNAhPq zvK6!Zx6W}=bq}Uo&hOp}Nx4Y94?*gSE=YaJ1!=?m*ROefl>SP}fyY?NS57VEn;mo8 zV`0bNnd)3jIg+^+$&_a~f3rue-U}(0Gr9Ld61H8=?*}PCi;So?aJh=)Tt2mw&(;nw zw>?shzlZ5qG37``ERrdw-2ds`y^wM_mwPWHN2H(sg)z21mw z1AnTb-o|OGd3oYh`bfN#@sb3uN0c&e-k-IqD)6ywkJ2EnQ#6QJ8c1Ga5l>ngLcW|; zS84|;I!H4q5#3BmM4Cy7NIhqt)Ny{;-9c2ot}5kA0pROQHm?BVa^^dR=kji9d=*K2 z%Dgg^@w7<$Zi;=s8edHksgLp#Y2P*RRi)3xz9RK~znu2Xl=ddIvFqbQe+eFVxQPaV z$EhfMV}RC|_oeS3@F7nKqQ1siD+?NrRAK%L1$gpb>XE#l0NlqsnGdYn>LjsatKT}d zfEeg)oY4T+jkOYEPf#QKo3SSFSQCvx#t9~B0WqKXSL^puZ3ymboI0(bfyr`l9g-k$s&mbbPQ`CF}_yhDG>_iex*H`)FL_-d0KP=K#9*?|T4 zcP2Zi0Pmm`*V{O_0M9nrAqDsVlO0-spJB4Y3h;cB9bSN6YO*I4;MFF3ashs`$&M(% z|6#Ht3-E_bc2rVgaJSB4-LC+{mt2Dbp-Zkdfv_c7K**Ab1cWQ;2!txp0>YGN0U=7X zfbb+*Kxh&zAS{U%5Rya-2uGp?yinHrTiQ5$%yi8{cACTas;ZlVtGl_qKf*PEyV zge;*F2yYU$UjjJ9yf4p@;90s{_cRvzZa>g*|E#~|JNE;3J+z=rSNVQ{{FCc*X71NB zJG{`v0hTBf=AkgzNHZJrs-fo2dsH&ou8nyUhbxu*h^EAlrIH`@7L5xxGtB)=Rer6% zJ2d96>si-Blb-#hcw}9VA7MWwcgRQ{0=~;ci+~$c6waKqu8E4>%pYPZ9bjRiF(3v@ z^PPfLG(XnO{4>l}3-}HbjRG-Pnr~&yk9IRZ-)yykZ!^&_ux`GcF+c3hzgTUY8*O3Z zgHZ&$%0z7-%=F7&9-b71-VT-boca4E{TSFNV zlUHuBwchAC5s0lF1EhCYH)t)>5#;<1{HBQ(0dF=@2l!9s#3EqJL>*vsHL1ze z)mF|`xy0AO@lV(W;$N7cv<@>rT0mUvZLB>O9DPg9w5f0HoNsOO?Nvrl8;CchZ|#Cs z95i`3IvsQ*H|R)a&=Egq`BB%^>6ppCwsMQs2?ISqKF_y$UC+?Ep0yXt@7hnw|H1zg zHO8C{$NoumrZ*}h>v|E)-$G}OE%%)f5Z!=SJyXux>v{?8$6GMVF(@~m{G7brks#j+ zfe&IYlekTZpP`Nunv4k~wyEYYHTCz`Ahwub_rP3Z$X%2XInM#37x(nyfc{p*n+Nmpu8CJ-5Q4MabC2#pmmWIT?YsmLD92upmsRQIyQmFt7y7>6MJc6 z&OH6NLA3B>*Ox#2_6BywK)4Ny1_EM9t&upHN_nR^nNB|5{q$UVh@DfLK+dfeFM?;O zu6)q@*#0aN2Rf@T;Y~Xx`~dR7K^-8^)F+g1>O3FfDeBByEyN+pM{q_`?Z`wEhwmf&KGBYUC6EoUWoUh|e4juQ6L4;FTsC1L9(7t78sd7ZriH7!+y! zPgDfrrEEU~>kg+oyf}QO*=hklZ=%J(y2CAV_yNAet%6ps(YY9imvWH{tUH|U zBI7W1Ap0}G%gv8Pz`Db2bNG8v5r~VW!)lT%)40MurMn?~01u%s<#vI>6N?8Utc5 zOmbS$e6o9l_m47LE#L|hjRG+krX;OsKG{9O{9$IR4P0)bVIT&>1fmtqC%Z?_RvRyi zw&-=PiYRQ_Z#C66kO5HjDPR;>kS!!2qfMC*Y>_b8LIT!pbuy)qKu z**X!3U%G`v6V@_YNEnN08Ku|jEG=ORi8i^AOGY3AblYngQd9HI6t6-*v@rKPbz80kz7Z)33mT|%#kBc z2~JwqBOB>!pC|u&e!81}mFpnaJyRPg?Ks8Q(vDMPGwnFV{G}bISTP%woZI!c{PzbH zK5YNMysR$3IVuX(N9*ZP5jbR`5#aEIzO-Iqile}#6BJ$c8#w$+bGQS1xY=3+#KqF# zPC@I0s0hSMi64lIp)akM#lFD0!|Bw*;UAmBE#M=}))){M%RyV_@N=Rf5HIB#6NrnU zFRfR^zQDS}>6FG{>Od|=fsZs>i-C2A+vad9DgyDR^sZgdS`rn3b??%3BlT_s?{+m? z!+5n{6aisN!orDviCVzA6X_b1I$;z=Cl=$xk!rqQ{`|UpNz<@^ z{7tK>*FtClcQ?@>o*WZJjCgVs0U0s#Ehk81p3B+*jL~$N@_r^`D|T#mQ-QH9Fxagt z^F^*}fN{VsQ>U9P77N)j1V&q3)(vlKjq?35<9!IY(*Z$QTm|l-qH@v<6|@!~YKp+4 z4htgL#65p0jmJMZ2t#jLgDRC_!r}P;IJbj&(g8AdX{%$N92FIT&oI$^AR~spakzle zL0h>&TQ=x1HbM)?Nac%L;QFZPD)(YFvf7Ld0oR&nKJYCn>T8@nUeI`d)B}Dv>a8kh zTpsm+pO1Pg3mV^xdcYq8L;9TNZUeMT8h4RlQ0UsRo z&Mau`6ZL?Pi+X1iG@caofJa5WWd)5Bq8_jn^-eEnERK4>=SRKM3L49z-f>#v)PS7U zec65Updk6a{>o|xyjevunbx01gXtr}O-sD7KBbz%6=Ogqj`aJ&5*pN~=mBxH()0G) zYJb>L+5~>hM2mq$FQ(33N(1|u@3i7Qt4?2Id|E*Rmw8c;HY-o1OoNq2w*1duT5kCx zfIY0x#;eMTkNQeoBXQ;4TNo*=4O(K=x6%|}VzT`U@b^u2Kmq=($qp>QyJ)5LHV!Jl zb4_+|0e*(b4k^GRCOfnMpJlSc3h*0Ec6b54&}2_4z+W)glZ%6N{v4^ly^SXoU<#__ zz-KJwX}P#{`B7!WNW1c(-p_~lL?2o=(i zLWL0#0U<;>0wXu+1CUp1&=*=A>{oz4sxq+-yuw5s;Ezny1m0$%4)B3Wh}Z%?!9*S4 zu_kH)pJ$>DaHWZwz<)PU2Y8W*TEH5EDP0|3Hmt?0wdM08@T`M_wuIxm-J#w~%jb|0KcCrhuA>bob6clM49TkBmOi-k?$P@>GR9dj; z9o$bHe7ic>*EnT$LF3&joGBkS-cwF9>!-~rkg+t&KODDV>H29LM$D-4FXEN`)}ALi z=}T*j-c&Ej1c6_nGGP_C-b5YXr%ki~c%6wlz`vMi0q~(3o6K0?p(a`Y95PV{c&dpO z0AFvS4)8n^wSaF`kx;Vc)b-O&*>P^}UE-9Ic?l)Gjr|MHNpItT!jsb5II!@nl&1tw zOZiH>=cTuCNa2a;Z5W_lZk6`q>j#^HtMrd;YF2PS$OPfnKl!!=b4^tUVt@N!FI z3;1>ub%38U(E{L)P1FJ2WugVZ*_N{o@Ng3?0G?=~4)AmnEdY{qDK8*VXO@YNs)5g& zp(b!foY6q$m~1rwnQn4v_KV~ya_g*fX(iVEV!>!XHlI3wQW$$Bt1!JzRn_tY|1QO! z)L(hVezPnS^JW)FzN$wgqMMh{`1(Zky?P2H-;3&Nmg3Zd^zJoxNrA)XJSjvekT!p!6#Zx*Tv>LF+D+%IOYR z_iWiz#{5E3S&WG(F2g+;$LZ*U(F-75C8o`Zefrgu((=AkjS^SxEs?&%gG=i)vHh}R zR{rt8W!nC;llugt?*p5zb|0m`LNBlp3+CD--kwBzZ4&MM)d)T1mdf|b+-2>dz20Os zy8!qB70EgJ+9mz_&s{5j7I8}dY=Zt`>;b$f-vco;$sc^-**+Skh=63v4ePnvUmbsP z3_)Ngr# z@Kek7c@7R7$a8biM)nq$e5ovO_5=3p)2ZYub)tAxALUGm>U2M*k$ ztg4H8(~CQXrcO*j8}qboi;BJ&F#jS`=>WfJqA?%_!+txhXg=BZ$NVz0)dGIiM590qmgZXpt!RF< zoA>9Ktv2w>CK?7}ur%M!m`_%ox2uiIqb+RwAc}ySP1FYBdbjs&7yts~DiX**Q|1C& zaUzx zfP1T`thV&+pR{8q*D-xXA??`8bxa?kr5)S3j_G6TwBtywWBTNMM`hqa`WxQ4=VqPhxdax!hnehUuXNNsT1{A5&~RdqEXnCaj|r`Q_y;AR0QH;P^5KvR0QIs5CyC|oE9JqA8NK* zz*m}RF%TC^KU(JS+oB>67lR_L|BQ-2yp$_6VBO)g@Nt+rkUbdSDl@+bSa-N>4!@Ar=ymit{e5xpG{>5c*K!G zSV?Yd+<~telW8KW(WaeAICz<(CVBLHxV}7)o z`M;R?Ht;YrKMbs!Z)eOW*N!vQ#=g-OHcn6xg-!cJQ*8rbA5|X&qrie(Kmr-98(*w2ad0nU%RxH@t(|SqML@ii_<^_>`qJ7r_661*PNxgi#coSd0^YQzv-cn0EA}g(q+z zd*;BpCn+6QsO|r>K%2npO*Duny%y*oBR)EcfQ*=VkOGO!^R1ckekNlp(va-Pq5@-E zV6dBY=8Ig{0ONpNrcO6oEEaNI1B|x1tQ+3e(aQG%<9!JDR}&2b`;HF%%Skg-&^lSA z@&z5>D<&w?I{TQ=J@cs@{P?*I+5!H}M591PF8%13AI~vAI=~m3XcUMup)akEM@8T- zOwp7}0tiR>=e!#F4I@3DL;%osSs`^-v9{yld1Hx3P28JrR=mkDqgAc>f zit#6RyTt!rRz+=KJ^r@EPk^!@1BtF2v|Z4`B~b+8Pf(;4y-QY<)VmS9yVD#V1lAoM zF^6BI*^w)E;45mG9VuunLaRLN>Emcth7csy(|3k)qbafb4qhG_RU&AS@$%dY1wcL^`znDcO;87`GDRSCe zI8B$CJX9V)fa?S00lAo(WA4K^rf3ruLlYa^Bd1Nw*ztiQCwi`YutS?!i+;EKu%1<) z-=bkFTKSQ2YiQw=uM3v{&OK#Egw;pJ36C%;p^u>w5&(`?Lht1%y6`n@>{ZnpfPU=$ zMv(jAk$Vr2dnb^0^8u%Q{9Pnkvh@F z39f0o-btmdwNuxUD<)ifj`Ac62@r4VceHhPTN!s-xXUf8*aLEDzg2gyX;iH=s^r2d zjcV$<9cd0a1jXisEJ;9ulB+0S%#`h8#0nZs1Nu8REr^lrq^>(lX+XmS6#W`b{Tfb* zPOi0ZBGyGY3H^v+^t-1u{X*1q7F>ZuhnJ0%sNHylLm0A7HYzHZ5)2 zY`RlDC|A++!2})A%AKpcPCkWLpeQo?qo_>RQsOF@HgzRy5)f;s`mABHS4~#et28ho zOEM69+3Q-2p(G~{Ki)n%4k~#r&Qq<=>aRQj174=0G95z&tuI7H;1?$-dOy+;I=K-# znGrg<5jx3Jb9gpte$Be@aMr&jOK zEPX_OlQ$JP@KIGrUv}gOo+T!Hkt7PfI0_x>efldKWZs5|rWRmotDV}KVmeY=t<=^O zizl_!No`HB)KgnSsjZnl+a_?-!gw0;^DfwxRhoZROQJ|AYb27x$UhS4c# zT^1FAgd?>JB$m*Z7T#qpWl}m@xW>q(t(N&QW`4AQ?=evah(D4*AkGAHw63wfEg&=p zMOxpFia^Mh_<=+k`c9sQb}rbq1&i-;?Fhv2a)fq4i-HkFAdUw`S~x0-Kzx=v?vuU8 z^EGNl@9Idmj?{{z9vE$PDI0I=rRu~=OFb7>D z*?sM%>q<4zZSAg1__do(MEA^z)aI6Pu`_{w)0uEmz##lQtD-C4v38D{ifK&3@v|z* zgK15<`IMUW8WX(6Bx(DnkTzE?S?k?&CfN6N_FBzZVRhC7zE4H{@@L4W$k~eahodSG zLYmz;K46an=Jx$Ip73)lS2{OGojJ>8P%Ohsvm4c2Mwmc$b_6WL_=Q_wJQ*a*c`{5j z^SZ0ndFifIXXb#t=Oh8&n!sS*&uHS$UR8hG$4;8dlMn&M8Of(F9vx$a)zcSG< z@b@Mf0d7`N7?jqnrZ@zA`qM*E%H|311QU$_|4Bu`9IX*k90oquL_@%36Z+Cx85M!& zOi=W5C1tpiW4L1sA7R0EfWJ1;C=kLVVL;d_gY6Wwo@ow`0%1q`0-+}Kr3E{(B@7%h za~&Ymh!${p)B?hiXiZL03yOBJ$OnN`V(DtjSUcPlTfm!4Gzf$$xf%k(Q!q#CX%@gB z5UQjv5S~I`T2LjI6TpRLt_6fD(E_fCTENI!vcZ-XRvXsp6t#^aW};jl0l%jdm7kgg zLxEgs9SvsINmt+j>TKB+3LCT ztXQ)90E8U5E(gLx`M?JZ5uyepzETZLfHDA(=t@e!gePjidZg)o+Oe8Cyx!&kcxoAt zxg>=PWG>xP)5^PcoO1C3oAFH`^z70&bHc?pax%P&+*{W#&XE(MY$YDsUsvD&v6}*y z?;N0BWIr=lkYy`ezH@}~@jf7D1+pQygWv$`ktyH<2bheh4;-Of%>-j~u_q-f`94fcBhlsXuY!agQ!46O3b&>Md`*q?)xWQ+-Rm zCpr`CWA|1q)NM#QCYfQ|lyr34jQzh>)o-!B9oBeKcyqnJxXL;iEHaA$_?Bg#JUCIT>^t|tv3!XCP@B8I~ zr>r9XDCL5`ta|tT+VD&1mvp`SsV$`)AahvmBz-TgYgm}J=EkO*Ew*W7w*weG=(1ze z%@#|%Y}WwUL6hAtU^L%lGtHa7Qc3!Oon)5ETj+BN+Re6ZjhwE%F1B z9X!4}&SMK_%ZH;=be^u;BHh>`$9E^)d+Vmi5wWm1Y47<+Zuu<{AQ z7W;$}4KUj3vSrxK7AG7jULgCmTeEfR%|Bd8Iz)fTKkevmQoMI+h2CYA+5|q!s85QQ z>}(6?v4yi+;XGZ*@!d%`w#f0_N%wim+y{-pA>gM}RBqS}6}0wzMo6jD1#o{84FeB0 z(Gc(rCK?8=Gtmfey^6x1wBBioL%>^1Gz|Q+iAI38O)y7m;F-jbyiJ_)(aG`AF+P52 z#C3q@njfP;!V}hj#8yVyDQNvZ_5~7)^aT<~=u0bxmF(1~VYLYBV-~<5kX)DMTNddJ zrq}|$-n<{CF?=chUb)`g zxb6ULWFKa(9LdwVh2T zsM}`S*%+k0x8Zg+CJ)xkZ^NBcOj0u2bpH^e6y7%7S;Z*TyRCOtvCCSp4N1o&Gi;lZ zj#1|6HYFXSO!jR}I(At^wkb1yLA}3Dr$^a>+5|pYMQmr&X1d7!B#WCxG$j`P^US#|_zZ^3!GZi{qdiyYsbbbno$ z`=v410kWcGikCH>Wm1Y47<+Zu)$tX9EjGX;8ep{5WiPCoElxO6yue#5z^U&bdh=gY z8$Z=w@=rVZn-uSRwafd>jVAC9CQ6Ez>}(6?v4yi+;XGZ*@!d%`w#f0_N%t(>*?o-d z@D2eFR8hH;I8@MjXH*1!%tXV$i%m2Hywya*z&_nClPxgdG!=zGY3*c+L%?U7Xc#!( zL?gfj6U@;%*%XI?OH4EbeC>q3wB8sMfp43j=qo;DxRYbJV+=1cqB_9aOf(9FFqxk~ z*ebc}6ts>vhev_1BYlBT6Z+DE9jP$j%gtN|2sNSwJSS=aVM(+mr>F%*Pq4@bfmCAY zYRgz#V2Ul^f0<|y2vxG-352I$j@Ho@z#tH+q%RPjLSI@?C37735;NBVLX~I%&x=~X z$Xc>-pB7dd*6I|sjUr~EoQ;4#SBlDAbubi^yYJwg%(FJ|Fcp2IYRk(ACR*G*%0jUb6~#C2~)ra4lo&0A2>p}nhD0} z#`SvRx&wp~-Mgm=5&8sxCeiZ*fHpy?6`F*iR%qfsR%jDXm*)sn%rcGQ?|6bBW)1PoH&+weI8CYc)Ba7!DLl*~5WxWy=iw@tUSF-rAr>n&~UvKDMZ(lN;l z+oq&rlzF;MNyjLYeOr@`T^5mT%8Xx7&($U9J+`1WffuNV=Loc!F0wHUq{#34n}2x9 z%(?Gx4C5(t@gHSl7=M}QQZ7KI=Y79i@RT`!-!B(DWfl2HDHr@@)w}Q4hF?m5s_W$~ zwr-v&FH6rZX#dINr&MyZHZ6;eLbu53ZAd)&@?A2v2$Mb=$Xkog^#9$U z>Oa-r9U2SP^{(rsd83MFmSXvL(wFwF>p^y_9@|xqo!;o*f9~l6{d4vbB@X^gm1d~Y z{B=Fvg~q2s+8(LDvUvlXqaxX$It9E}l${LTJIYQ1?-OMs;KxK+6a2U+I~M%-C_5T_ zgv!J{;KiQ~o|o=T9#xB|uo!l0X?4tg3yA%oIQa-|Mu01E>|`)piR>gWT#0N13|As+ zg5gSJ$AaNXWJiOErLS?uC97#X)dFh+2~7H)Jb73ngoVYDkCB4bdb2+Qtl2-)+uu#q zAECcFAH2?V@w5L-`BSZ_7&}=F<4idK_+|emRE#W5z|&RK*I3?L(0I63mgvnJkU-(|HV!Pn8;zJY95J9gp8zs;Ut{U= zf)OL_$q9s-XR?(iJwL@t8D6~Kych;U7G_qOYE=Jcv zAlZ`Z6tWaUO#Vh_sv-Jh9H4Q%!s1J24fXb+=r%A$vJfAiA0tUKv(OKJnGSn{4L=Or z&xTu^vNQ}!``U1ey=@ApVTMM5vsE-*zJxGu?&*y=2#9s(#tQv@Z*}-~)n`<>GzF3- z*W+aM3l7|2@9T9MKMu+OKwO)uuiMtKC(k^(^7ZBVD;Lth^Gq}he7lPJ8q0bL8lQ>6 zRmJ~k<9#~@yHeeri*(2iLtbFL27w<_QF-D_KSZ6{YGLdBW@{8!x0QZ4Keg4y)<@0O zVqo1?`icD1)(E!f*w-5+bJxrAbj!LduaNy%=aiDQZ&-xxYa~4Q0H`kJTngf)e z=9Y-^N*SV3m4v9LxuJw8Pl~tF*T}?+onzHrj28*PWbWUYn_HyaD`nzMRg!qWp{(C% z?6-mcts*ImRi1;->ai_)m?~w6!hvtA#YhyAP3Brs7(bkf8(t|xRH~8?^#>hl2TiCH z2Jqo35~9XEQ9quFhpAGAC>;2+T8u;?=`+_7qFCU!m@luCAu3f#hu zEGt*()z6RY>Z}F)jf(O&{&f7Zb!-E_W}*>bv~_0V5y^?*8ts4qDGwk9On=|vbluAhn@W(qHU1?`dq-n1Xy?$c7hJn9QQT{S0wbjDb zO=fE`ux=~83`%Xav9-x;jRNbo(#xRK)(Ez4G+T>+bzAAnTB$8#`A2F?HWM0Pm>1cZ zV8|5NDPRH-*~wrW64^=MU&Jv+zz`~WO)x=->{wqYZ%`ZGjj>OzlDz7fT1hUkQL z@haIx;=}6b$MrW(3nZViP?AOQb9(5iq%)Flb?>}cC*oe8Vb`exJV-@hD$v^P5>o`e zKt=gElaA2JjnFF65n8zs(wUTw&}M}FEvzvh!<62(ZO{`KqU2dY_=M_3eDFgr9ydAko;jWgKM; zuW`!ig2qf6BPl{ZM#;$P?u^7h=UJe5Q$98U#~dV&a-OChqFV9;KHUOp1D9JsOM8k{ z+8{7Fr2#Lf>%sROW^WkySJRVQbYCCz`-&=+zJW2&(@1u4W+}gvec27XhjHHmK2b$| zjU}rK8j-Q&{E4YK+OM~IH;sD8>4r5S zpJ&N!0SR6192#I!T)NUS8VIx01QTS?q_srjD>cCc6f|k!eyItrhmv025(>VS_um>- z-R-uyy;%RLPEiEny*wrW;(F;`+j6(Ta@PiuF6j#-n9!HjQp;Q$NQBZCSPwS6lTCvi z$pt%N!9K-;9Rb!O8?nf~Zjp@ual8y;q@ZY+h zQ5nf%ogShLq?D#NmX++!DG(4_w1Omtwg^aKqOuQqg$DZ;{jJuJDJeZ!oo1b`deUH$ zm(%zV{kL`J>#v-}Ht4$i(YRE18}nRsOwJ6OSM;CJ=uclEd9Kz6M9T?|V)(9=PI6j2 zM-7upc@|1KqoGk!1pT4~+;mkqsYu}Gul%Jp-thUFkjC#+DF2eYoZXg5O${cTEP$S~ z^-yK@-P$0>m5>Hm7N39+TIsz)&wBr=zj634sf7Bd)O`jDM^JxbWn;DY?pZ<(RsXDo zf!w=`+>iGM9q(`YTRw@dt_2aWTFj7%sh5d}R_QV~L#B%!LPRoWHrC#}LXY9|wtbD>I4vgxsKy+US=bj45B&de>juMydEW$ylXp08P%@my3hp08P%@my3hp07z>(zr@J{!;WDSo8b?S{UrtF~rsgH0mO=~Z^JZ}LxrS;1nO)Ql^o1jezmGWgGr%g_R zVOlKb;sKBo7=~$c$%$dwWN_l;KPLmM(54o#LYuQyu;Po9RKt>7sRN;??D)Uq=FJBy z3x}*YK3-e7Ab>>6+!u9PB-+#wuyO`byzDeT{?r$BI~9+1;H@UwLT zG;a3>5d1Y705UTx515VrrkB<74tZ}ugB4jWsDT&TmG~I&(f_AhGny6b4ho^73ZxQut=O{eV%i2jr5lvHifRg4Z%nl~zp%^+d)`p0lDy01RmAOSi9^R|&R92l8wXFSlX-2?o7v)6+`CCw zK!)F9zo;~D^rTK^-(2&n)>|sW-Hq7H0|&7s^sf3zd#& zNx+jU^){Z9MB|6ckmAoZ`QpxKWr`cES82dvOSEXcInn^+1Q!(jM8F(cLc^LQ#EjDF zMCm*onsLk50>D_$aKnQa*4=51wO6f5cC>=jJO6!MxPPO+ay|s!VxkuCHWMX(GvW=E zzGLZ6+O2$3%{#EumEj`d!TOtt?|z9&vKYNYAB*s;B(q*EOH`88|3DpxCoh3AY(792 zsvzaTO5DyUN%zrZEN)m;WGsp_V^O5u;@RrsdyKE)Pbs|XKLeHWD3cL>twD=uXLX-2 zM8t@9*$Hu&M?{4_5E3o_Y)C}jsduRdPtu28a^baE;hjaD4M76k5L3XP!l6p zN1Imi1n)vi&B#WD(fN@dX-e?8>Pq7c=KctIBBDyq$MiH!PhmCF1YV>fd2h9+pz&$T zSrZ8RV*mVc8W1D14fp_!zOqLHYD5nRHI+Rm`LI#a1YWM9${r2)65*Ry8nALEGsF?|eL+=1&ky0i zPruxWmYXNEzitIpPVul^;8Q3M((M>b-l2KK9&MHm`7@J7u!aQR>(+dd>&JW~57CwO z4YuMYH&I?iZcx%IM;TnZSIiE-vnRq>e_d)9Y&5W~oRgoD-q~y0Yjz~#xZ5)yM zahRHa(M* z;q>bHyI9K${^WZ6?3LU~9He)Fm5{i|ZY3nH+D%gWszzpvHMud?tjvtDW@ToK^hK7; z7}SjjPE?E` zV+9ijXH^Ffuw_si3=vxh(L*H6m;2jDM*n10P`>9)`j5A};Uhnp z>H$xWdeaIT<53TIPSo3>pz-FY2Ygr5>n~_r81;akhH+@{^>!?1+!pl=+V;)Tor)e#EBY&UyZs99AN6)FXgnzD z0UsIlb}48)D(V6Eje7U9Yd20PGA>j5V%s)tGpCQRow>H3c~X$fs|%wRkg1lriT}1c z`H3NB7W3rpO_>$C6pEGq%ify@*i}?{{I9!{&cY%H3WTtQ#8D9?1P~P^jwo>(`@MeM zbRZ$PZrtbVd8_yslAcw^D!7d9XL*_AEj?@q9XRc)56M)G5hU@8=*)tkX#?v|Ka zD)lCJ)?)S>HSBf0y-b(0(CxX?#_hS&#_hS&M*5Y#4^R9_3BgH!>w*vI&ro`K5FQa4 zmq%UEeVOhL@RZpMH^_)hx$`>|n8!F$A($mU0y0%9-4b zKU&jcwH-i~0Z&$Hpt0uUf&%FyUHVK!)k;$}1f;bDD5^R$%5_tPDbgDd14^s9wWD7N zAducl6MVZbs3|Nv(hkTxkbwTj8b*TwC`m$3luOe0@Lb(MPO^<8`D*o8ty*m&$!H;h z7TIRfr$yFjdp0OpZ*D(rFgzCTjY9IO?mZ%cGT8vTX^A33z>7_B6ZjFO%KmgLJ?9Cg_3}o63G)_J_i4j8V?2;PY^hA*)MrgU-&O<}M&nQ(U zCIw8&^hPgHB#9X!bRk-Rw8~BcDoOK9@2N=QYfBx%=lt!=V^pfJh%|vTl*GMq&ET3$ z;;z@N{eaK%?e+-pSw@WlU!YX@Rm4a^={OS{0j@J@6nK+SBf#mOqtj8~9HT~nJ1bRn zO6fgO5csIin>yeVjT!;I$f!}^QlmzIuQF;Bc&t$)z}Fd-e6N)TwoPE4ZIM_8B$UC& zb|$dxG_cDE?59m#8@P=HXBm)a%9|U2gtF?C(!W^SHjrq_<^h~*p0|N97|Di~2BS$Z z4mInS1M9(PW`fa7gRz`oTyN@{z(<-x%Yj5d);A#ThhQ`dN@rQyCXfiolj6XCuwXQS zq?SklN23&w)DkJ+F;NN_1Dm{OD-CRm!2Y8}Vi;HtY|8?>+yq-dqAA}g0k4fxz-bno zVIXk}QEXWh4>Z9RkkCmFz&*|L7BG6Ad{dcvK8ELYH^B1Mw|bJ62HovUxfN zyho{U2MN5jIu&k7uauRQG6^O95RD(0|6@R$mLtwZ6BPEf2#f(qp3pORc@zVZJ*7Gd z$45CJ0WIbH?n_7qEdawnLL1(@O?Zi7;C$=?N1z%OsN*G2@&W}d$3V3R)LvQ@nv*F+ zEO0(hf%8!boR2`@x_OIhGG0JDtnz{q{%)=W!s)6PO1LY%0P*^7I9KrgFOdNKmhPiK z5<&82i68#JWzu~~#)T!B9A3g`GQ)$uOb#!}7yqMNl z{Pg(+w%)CCuwp5pR@9-HwWJdCogA*F^_(^IOSX4Qe7a@H6FDHE+N9MFgV_V+`V2^* zWV6EB7>{8LBOaLtG0=@7`Fd;oeEFKpYlHVH;kyMHz4B5HuTm(lE%WM*m+NRnMMtKJ zl9A8k7K#}5ZRIA~mJdGoU}Mhy3&N*-cZ=dfG|@65RFM1`>@@jsl}2T>zO)rChgg(P?H?%H$?q##S3LHQuT2|EGmv2#D`nOOo$5 z|K7)qH|~Ah&i6J7coxrX!#9fi&#jAe^XMJg%XGQ*eBgqDA{zS|hXe6KB?&okQ@UF} z*JwZCmlNO{l-og$Y5OgZqgpj#*Ihbs&a}P5Qx^ze!W%pNUb5@JmVY>;>~Y>X(@t&d z+7T70of&KA`MM2nx!;1sFbpE#?}W&mdrR$DA=W7H@RUxNC!d3LA?wt;Uq zY7~fjRa;7H&4)JdO-79ZFE*+T{J2r0!0U`^1Me_u6c~e%+}TZo(Igmk&zt7?v&_vV z5XXX>&4SVgq9Bk_tOP0jGztOT0jC6dT3d=NCi1S0P(vLq=f4t2*mG7kkU=&TnmWbl^`Wt7LC3A9n1MU zX8yk0IvoREs8r~5te}MLa$E;ucqK>)+eHv~i)xfhq%m-GB6;x9TP&LXM>Wt%<~}^} z#tmr&#D|SC)Q1-RD3j=84Sos*#La5tdh2SUiwPh{GGJX-`sUiyt~R=^v%$6123?uo z6-d!E-BIIl0aH=>KVDG>)^b`^2oLB;wm@}t8# z`D$AWn!wYP8VG9veVk!oZUPyUQjT7K9OZz-u9WkE!?UBc(;elQCdan}=dTVQ^Hw#U z9Nzx1c>cTs3C!Fjn73#3m*g_yh~drd8F5i?PM>w)UF~j)kHF5IMt7?9k?F412W(2U zK9FovciMmsaqif|#w7U?lU5vDrZ`({GJ5RE{BQ3+By zNCPHV)+dxxM&2$q^>qxkxhZwAiqNU#>)8*CC%r1|yH(ovyIc=sr`XxG4IDSB1$>85?N{lT z9;2U2rC9XoT2J+-;1232GLF?=SeC~Ga11P9!UN~ z;#-r3ty5h?(dDDHLUa3{EytU{V~t9#B2$q>@eTR0Jys{gS^DY5_IsVQxAtT;i9mKq zfsEK)_owdxm{e{5p`ZF50E+Kay8a#jxQn%Jd)@;e`sGCcr#9y8IWXBr?ftHPcs?--C8n9plXcx#5#O!h(GbZD}2O{nqV!E#h(Zet>XQdxz7=WaCk;>F?5jus3PX8@Q%lqifseY_)CyZ!)S4 z{E||G^2+USwNLpgCYmg0bbyMmGJ|QJJM*nqmA9MfWFh=fB9ip}K@HAT`k7g$KbD9j zT6WR3^JP=7p&7BHlX8V*@HVo>$car&Ur=c3Y7{! zpAQ;jt%JgYEm7Xstw^T{rqD{3G9U`3$?f&sk|l$NNM&9-ZtH zyjwHJO(n8~tyFDeM|CsIF9EZ3UUpGROg33**l@A5rZmtQOLN=v9JO>eTS>|cq_~ID z^$di2g4S6Ec5KhoE>S5mP?m707@f%hp>gC(Ityzpp!TXyh?5(WebNk$xm23Ea=fgA zBR1tuQB@J254jm48lf2EQls|fN18z4i|i5jL8n5|85)J^rikFkd_fW33kAgKe)+`! znEqr>8mB{Ck)}Xw2u=Mz?~NuXhS0;t<*ip*&R?&xB(WM3vorlknoye@@uxdQo5k6$ zX!3sFc9}NthenM6f20(fDCM7+X!7(89iSp?qF|cmHqm=jvN^v1HV1yh`R7!SXr~~TC!;&R+aClcX3w7z1=kFGzgv=gt;=n!l4Qbhp-__EYu`s$ zczhRu^F0L4cM!O4`Wt#rL*lX7z8xYis?(dTw6|PoZ#;%M2Z#^1@MUQPE>@TL`jY;fh39y zj68=z9C5T_@fGVT76Y-SVtb8Wbwixdq<47{L3j#g(Og@FNX<%~5{W7aBhSGM?kRWC zp)fhB!fevB)(8rZVk7TWtnr@xR6|T~Gq>ya(s<*htTy3xy^pfe*o#q0dyACz<}B2t z{zpshHt=~yjR0R{R2z7dQ6tRSA6tgDfepKFlf0G`3+N+k4B(?xc{t6Y9aXsvWVy=I zP9DsRa&Gi17i-iyh(>fq$ubCJ5( zLsv#DcRt3~FRIq*;qKmKKgF*oUEb1IG1<3LE@$9=pmpW_9I5NPqe$&tDplT{5h)oj zs6T2|7?G+5WOwWV#Ha z7t0Q#-=0(#ixFFm()7r>rt}7`+n4Fd<3>Ayw1A&8stNq6QOO$R6@_ILO*d<^+|7g^ zYTeIFb#gP|wnQZ9{Ue$coMbYut-c}=Nje(Uj^3f4T?R2DmYYAvsFu~HZAF+r6i-!p zc0*{%;xK>UFkqI}$pYus4b&i-S&>qaIO1ESt6*X0gcS=Q}n2N;6U}ZM~Zr zrD4k<>1#>@ol%YPFGpV!UVSpiTC{5Wesg zv5fR&9|WN)SFG_+;~lH@1LhLOySI;drKmcXgN?jauFWambDokE?#`5b5ZSD-=j)7 z|8ADlx>FA&WRoiOCS}BOix8&@&LX-9SyBD3FGACmo2#GO>q^skN~ZQ7Rf=^&zK()A zw61)WEId*Nhj(SOP?zpK*hrJiod87WQ5Fs$9fXC$$Bt&mbP2H+%TC=r)yW>ZGGe*i z9P9CUdgI8jXHa~hMO0{sX4uW)+FG}LXd-o;r(tR_bCybF_Ix@+G_zo*BFWc-UZhKm zy;wHJcj@h+eQ)6g)(CJvqeg+xP%8WuWu%}qW`ZNYe>Q3q_(r2ffaiRRPDg?NrBu}e zrOQol1bFMWtu62lqeg&3-!Unlsd&D#+0_OvGHMuzuhp+`C{3DO!$3@~1Sv&3k~hv_ z$8_to2|UoKYFwKh1szjNY9lZrD#X8jAIAGFMAsB0JCEmux>}o>|pq%Es)`t zB?frDscQkxH3yafF*!KUDkw4h(gTp;mmYv~ZP;7D=y@`qQqRZmyzb^$#?3MBX0Udw zz|-%Wr(-}IsRSuqYW|M_AEVhUFTI^8DBxh#ni4+BgDybaEn9a>eY}(}3xRcqmu4JZ znsaz*#^I$ohnLzoK41#NcZD~R2@ivr8fd?s72~s*Z3IYj2CFrBDNKI(zXE=L- z-)80wJz2-$VEqhY;A3pXNseD_t@?!#m8B8wt=L!r0iQ*__nTv9NeT`D|C$BvMvY1qDDZSSQ z?{LPEoRYTQj;RuAl_k_t;2X@_Qzi-uG4Ym?9vBlM1)gPwO_`q5YY}8uODw{n1V=ldi`*5z%y9Mw1tl7pkYfd|*sYvL#>yc!^QVfY;QPkCB4X zOTW)aVHxmPrNZ(tQc(KT4@?kv-mMi?2~zr?va&A$Z}?GN(B}_sjGN7EU~IVV_K#0= zYk0IZYyxpwzV`&;aj>?T@qakyf7ARw)dDpPjMfE>-K=|^S=Rz$lAMFFYPqQ&1!9k^ zXbi&tV^Ybg`Y!Dz>K;b-Vb*FGi0;8g^%hw=I8H_3y;Er~qjX4bA-=6ld`Ib%zA8aV zYt6_}U_HLAB)+LBf3cz7=DSsXKS4jktqbs>#_d~xUt-*%0=(9^{R;5AjN88e|G&l^ zP=LQ_+>;CN{~32+0p9L=m7V>Krxf6Y#yzzFzsR_!72sDHcTfR-y>U-3z~>wHi~{^I z;|?yspD=E50lv++eG2eXEH9F+pRn&?uI^KS3AIcfAd!|cG>|}x6p%QJ6p%2>54C_q zS)_mjS)_o(Sfqf2SfqeNSfqdiSfqf&SEPW1SEPVMSEPVfYEZ>TAo1Q(e&BVrypHhT z`K!X4q}{*w>5B4IRUp^fAN#~aLGd{@Fiqf9x<{xa#dB;&7zKV)DV3ynqHYwjYy(*Y zRMOelz(`ghSyF+2?gc3|#835=(~>;QE*jK_|sKO|lKFr9+VHmJT;Y$FCJJig+t?0eF3M zJV;W!DLM{(Wpq49QhZHx9QaSsoghhZ)Z7^X-q$HT6<3W<*cyt{2vgl5br1zYqgue%8`T6}XjBXM38UJ;?-|tsK44TExRY%>E#Q-l zY6F`_wSccSsttUDQ7vF$R2#_tCc7Vytxh801NduI@Mz8W@R#VqdwD5y@_0mJom|Gx zn?1D0-0E7M3bs82nh5mjX?D2ildk43d`!~Yc8!Id-31}vqY2Sv5c0ilp?+be%zZEN zr(J6JR$MG_l<##B;oHlV;!D+5R8*Igl(^s&K_FL~l^`W9I5+s(6H`(9(DyK_u}1#x z*~cg3FI>_o*3S+X+*${RN3~{9tPk~Oet_|hqnIqW;S5KSkCk|M{k4L_Ov zn-1w)^>cE~B&Lect+t7H|GXwJlq)+|2r-Qfy8_{Zc8GZK1YQ-FwyOi+1g0 zRi$DIt5iiVt7;3#Y3r-wuRPG+S5TOyXJR4y3kvHZIZ#kIKavy42unA;EOAM|VD_n1 z^2IjrJTr8>ub{BNx*zW^D6Ea-KtbWMNKO@->aYUbHN69nho3#kYtqSL>l1t>k!S67yGQc=qkDAGV|c@+Ti)ehjXAT ztMa#}k-^YiO1lTUi6eH4s@_>tcNO+7-K4JIdG;X*{JuwpXEs(=FEcJ+?$;XmRtWf@ zQoD*d@`mKZOMM@5fgE1Xn<6r_K^E5+OdTPbAx@SxGD@IS1Kp~(Fc$rd01R!MKlQ#D%8)6afcbHO7P+!m07O253)2K;NwmX&hS z%V!HRd+3rc*&=oib`$qjlB_tB%@$laRL!0#--=e3PyIZZR~JhO7+vxSzcHR_TBJ?=jdV3qWyyj-KTQa?ksfX6MkEg+}G z{>DTDd`Qdyv1M@GmEkLfbR+OR;|d05GXV1#te62XC!3K+05M2LXDWlbjSjP5<41>< zY(Nt83pVK_@^9+gWNmc7g*n~=e%!=X^?}(#50Xu=`&mR|QJp>JrGtkSJo`u8iN(FH zXOLbx(Zx3XjId2lai-~|&nFn8F=c`g;j=Y#F~MNoZE+q05{*MPDW>Ef+K9*VHbZRD zjZR|xoK1`|dVQ=}3pRev(30bk%$aL9>9ll-j@RWj8IEg!ud}gi0lCtW(^4OpL%~6E zfarOIs7@G}IfE9C7BXkVv%w8HUv$;kE$&n1I1%}%dLMHf<{FF37!d0Z*`yedFK8nc zlH;2k3v~OKu<~tuF4B_2m*nF5O*)HxQRDgz{S3P+_;!nV3ph>BU2+!d19Rj!NRB5x z4=C07C6kM2;ou^&$myl4&Tes^G8c)+SJnHNi!eX5xQqd@{*X z$+1AUj|nTv#^)j}S8uM9-v4P_|Dc~C7s0bN@#Kp%Ah!VJq}K=LU?tg89;`OtbXA=O zGa1<}Qd4FRKHQ&V56lenWDJPaGR~W9b+>V5wr>15(~_Nflh2GhXash%#YEm!sdz7o zP7BBlQn`E859UO2`bL~@Hsy4wQO#rtmJpq@Hu`K7PT+&pe_igwY42UnzTK_EDo$5t zY}Dy$(@wkHt@1OA44ZOhku`HQL*lv&^H|HBF(9LJ=3hKdG#UjKF*4^otnvjbv5ves z>tT^MSm*PH)qmsHHd=1Bv;BdZgNyYu>|x*+T7I^Gt4wUXAN(ohPTz=g?WSA>bf2eK z!kJjswre-;`e0+_)9d=~)^*E*DOVB3=jj^cxQf6$&qiqsNF2`mi;oC10E-wISw$X3 zBZHNGgxI||exA~Dvt9QbuA}@C{S5OIyxKCL1w6yV*7k$XkK8zzll$o#aUHNJS54h# z2bORRA#-;luOT*8epBWy<2&WtWz1G+;N#qdd5w+S7?3!e`4=A%W(5{8GBS4`Mk9lj ze(tW)&~E(PrRD0)b)EPYHRU}0409KJfn`7o_&%j}s;;hb-?$*cg_um1jl2-sSouwv zECh7Q$wH9MRqx|uf#F$$%x?l&&NhCqX~_X{lP?x|duGSj4?cjT$@y5r9mYeO+xn8Y|Bbq!{OP^E513kAT@qoFbdpu~o@$sPLX4~9( zxM(L`ab)y>Pf<#4U?w?(9aF{#-7{2Cd_Dl?k=ASSgeQ$RK1Q@$t;%Iy$qF^qWBquc zl0JPGs-l=PNruxWnzG-2qG$gk9x$6%}|23bmfG&b1aBZ%?S)FyeF9aw94W%q`&tYH&~ zpW^kUd9Rx}uSGWJb<@0l_MP|_^isml@W{IS09n2LY4loTDMgoZKP@H}TbWBOATG%h zevH@q@2IR13iu5htaWGpa-7ng#)e<;Qo>bvs+v)KuKFhwZrm8%5RH^@N4oaoxtH3G z!POXG^H0r`5J5_qBCjMCb}&5ji%A5zR)uL8QT5Qv4lxyzFsAyUG9_ZzGo18aHK#;q z#F~7dF!zrZ(J&gl%yfvQ*^ql(2K`m0#q&mlTYZ~1AJM6$m9=Uw9k#T%5C2LOTC zt&(10p$f0D++7O%mQiEC?11^tprxp}Wj&i^?QSOw2!rP-9@VqE@azWw4Q4aXAD0fmp;f5#&{A`pv zv7qqXCP2h(ppl5N}knn>#FQeJ{59JpZhVh(z)1Uyc_PUu(-v3s{SHknE=LxTx>~5u=C}`As6Q zrZ7l$Q%EG_@B<_qV)Uibk|K&l5?JdaNOtSv)M)fwveHvTi(C-`U#Z6c={iVKTxF+& z5#andJAx#|$15#5f$!;*q0IxM}Q30MbZodNj z-^T4|-&(XE~(;aak5wARdbp5Qjwyh`%BQ#9fgB;;l#laaN>&_!@qkt?uHsG-OG~ zd!c{?BJ*43y>uL2XydU@0d6ZNnt(4iY8d!>qlSRz8#N64xKTsEuNyTC{FPBdz**|4 z!~wXcQA5C|8#N3(%%~=?7OgZi$!{w;a7?PAa83bVVak(>HAlNu#o`@KNb+>|6M-^fwk2J}&)@{R$tL{>J`=k4%5#fWn8SzwzY4$ELp_ zE2+k5H+3kyuL*pxaZfG4D~x+u0VWvzjf0B+()tgY+`;u|o;wS*DxdzfDsM3Tj$6O$?)vR`L6PIT z9Grny)Q&JBNs&H963FZehaHjh(@!M5H9ytqzLkzx>!#{@AD{aC#hP-~ywXz-JpZ1bn$sO(2Pp$(jFB=`UKV-4^E+v}Ck6gY*+92XjTv`=9P%jQiyVc4zwMNyxU(IFM}V)5 zhvuM@;@eHK4ZOQk(u(mYJYC^Y@hBX0Qamn70&nV+r1*m<34D4y3J0AOpKnLu5#Z-K zB`JQ>B-_B+ybL=1yp-u0Xg@a@4YEs?wuj_K0{yIxHxlmp&3HkPyqAdtYdB2HtH@NTob@JAx)0fP}@AG04 z;MHB4z!NIDzm09z+l{)zfh3IY*xUXiwf)W*Kp-|s&fR?HI3>)KNCSyhC5Ta2AcDTr zv(-gh+B~kLTAQgY$t$-8e_oyTcht|!#nN3Go3EP45SD+&#%~D7_{^5ypN7eL&b0JB zmUC4TX(E@PK1QiX@_H3S9yGg>V{a<5v|C3t@4AU#?snqZmJiCmIWrI0HDvVJhs;f; zJYCZ$GjURp(Oz9=o>jS9-6ssW_2;uHbBlgXC3CZdvkfFRTgWqH)g0bu$y{$L{l8Fi zVrJtMebYM3(pWQI{Yc#Fn645?%Gt|{JW53#tDkaL5lE}>Zj%$_>ZV)Md1i;a^li0V zkrps$n%P)*VpRi!?kyS%PmtHF%v%$h`3~l);(g4vAz)jnEyS{}A)KQ9zNWGX+~24n zAXa8(@e@@#iX^jz@0VFNWK#}*>^ieomwyA>c~5KhTUq%0WkF^w^sQ>&7Soz95NcJf z>jqk|T$W;a?!X&fvEABYw@6)-z40S#Mx(G^<92!m+yKK7t=(VR=x%7cWofwlx?gtM zdC91C)^;1mE4-?Paw59B*Gm88h`Y$6G)SrksPegsrOFMnk9^(`|C za#IJ6FG(a{fv4EHei@J>zeoZ9GfDxkGinIUBQvaF97EFVPFrF}$cRtSXs#nM(B1`N*hq;ruRI zb21yDchRoB^_omKz3ViAs|A%3_dR552Y#eOkkb8jw_-W)4@Qju8L+AoKgTP!n}OX2 z31eBV0qzX(sv>&8ur&aK!@t5 zXb|>PEfK+rox-=S>GUkM_htIYTki0Op2C?HyjFW3lvgH&x7>YDuMjWM&)Qq=FzFXs zg?gVfdulxiCmT=hHr64jSniwSmpd^DkK9}Geg3PH%B;}a{HD5o+f6@ZS^?)PRlShc z>mJhDA~pnGY}7FDzdLkM`cf1Ge#@v4;0^cig;KJ4q(-zc;u2Fd3Z&cMP`jXXoe8#q zHybqy#F%M~6E3>BK`EM+?2f5vO-!TbU`NyJSbH~aHh~{i&nrPnvD4&gJngiV>$GK^ z?qr>|fOIYwMnHNFcC-pg(T?OwAhlyGXUCY?K~Hjh3Z$FR=~$-IEbMFzs}NL0~PA!#S`%E+du4Rl*&|CnkbspQ_~qP(Y*21r$Sd*-29Gs#hTOc6`8cnDAh0uqZkIJM#1~bRSyIDJ z8M>fM@Mo1v>|m&ZBD6H2jB92U1isRsm%EY<-t1b2V4=LR7Kmk?`vG=|AP|e>auJAe znfH_DhGA9LVQ6+82C4D>gkcD(a>EdK@0kumsKQx%3Q-L_WgM%9LHdTkQyTOhhB3^J zvvQ2?nToQo1F=BkCsIIWo#+A* z0+9j|jnY^jigmg~d_X^^H=2{vCZ|!nUFq3!9u438ggVULMQyCt!7bFft>rhK=ky)0 zfVS`lXWGIY(xol@QJJ>H?M$sxHz>Zni%K6zZRUeS?#AS316h*8Z`CwbB*&(lWJol0 zK1Ay8*9~WS5M#d#f?eI&d4Pk`5L zfDOnn7;|~?H393e!4Fe+8~oHEE(_Hrxf6+*D9t=km$o`GZFOX~zINBXX3CA`iNOh3 zw_u)NRty0#EDF;Tp^?Q5h{;j7n+>RyV>kL}ocZGBX0_`Mvu^}A_J7=#83n#rsc>gv zq@XnCUK8|2q5f`D-v&O-)E@~%VNl;T^*@b*z}Nq_>ah}}^j2l%$`6QXm0)k{nyGcm zbJjI8)-BIj*DO-&mglTX|LAfWyH;x5vYd4-v+h=l-7+9n%Fjlp7Zflu#Hp1rd0Ecn z^q0K%>o8-KBrxzWvt*cIKFO#dAa=+N5a63_REL0!sz~`EZKzJVl_?HvX;tMTMi3Se z1QL`fBD7)QNR6>%svENr$&~9i&78Dfn_~5TH)4wA`fVp)QqnJ-$YnlZW;BKJx-vHd zyrS}us>6q-zrkVqmQ4WoIoNnc_kLV?2|^E^|)sSrzW{$QR^E45}ML?HB^O zqXA{Qw+7x%tLW|KvNYqCk#Rq;j9YVBL4h=ryEwpAmN-WOF;S#2jx>`k7x*)4xD-g5 zi4>4D6J0H&X}qF0I>I&f)c2MDUko+TNht zO_@44LgDqo|61Fs8x-$R+Bfqt(`G(M3_8OUez`>yh;87ywIdVdz8w%%VaseL$|}~Y zBhhHPp=QvIzBjWOvQ+zEWL9sC_>e8tO(6d04IQscX&(*NbW$|zP~cbUc*~BoN+F%p zH@C~RE^uJ7P?`dnM!7xkMs)}Ea;N5Fx5}Lw?*zQ}g7%+RJ>l-lD!IC${A~-@5D+KJ zI~G9}7gwu0=%9>k;u25x9J_n(!#kB$ANKBAeF}QpWiOy(lnT|m)rz!La+;;Dn{}Y7J1p{)dI9rY zOV%ME0gJ-)^v5U+#N;U4+wx`TqjBa8y1Sm*_qNk}8~7xp!WP{wC_N(z0-vW;c_*8~ zXFFPZW7BR&=-{c=!3YqY1C0}=7Zm>0L$rC*hzWxt?SLItY{o9z>c~mv&=`=e!`}o2V}@{RQ!B>Olt>FReqXkNLtwnKP7yIYL= zpXuov5~PFp)@ea44n9lvxv#qW`ds~#Z(f1_WYjQlNU5+CjTDqFxu1K+$t}E8eH-<( zu9h`QpR!Kdz|R;p3jBG^j&?z5@^{$L2Cg@16xcLt+rSqaH41#0QElLAqeg*mFscoF zt5KuCw;R<4#*ihCT&5vw=0euA@DnFF+X0D4byY`cr3Ir2JWHuckkVVDAdu))f|QL>^#hLs>C!YIxF31cNl>1|OXkf2tAy)(U)i)$+r*H$jBEz7%iSh!k1 z0u)SNJzVMS|1=H9a^V`w zgljApt}zQ29?1hxK->v7j}??oH=D)x@6Z#T^nWjIXtd4z*i_G znvV84g$xdh0YD~QWho^b{)_pZTKxaVx^S_OWiDo@Vque3KvuPC0YhnMz`-abMM>Rt9P`0%Pu#%?{(*uykB0b zivt<4@NlU2lgMJC+|hi}9dcXSAC%xpY_bQwm)`KMJg+lwUSHt4>Ee_pi2}szkoB#k zgNpTCRIKlW6ZaK52r{*RpHXTcbb=Gh?3C05zF4VJj$Yp!<$%Pll=F7r+2z{lj&jqp zytZClUR$q<*Vag2<}SfJ{R=_|&QHVLJnQP`2Fu#k!z~3jrFxx$7nt_j2Dr$zgISK>6nh3+3+xM)} z!^~Id4Rz}v^Rx}bW4YA~MA1NF-G5B<4q7wQ?(7X{*P`75i{B`a&{iuSrMsgbkg!&Q zl(w?CjRFa2B}nOWQ4mNtD?v)cBy+$ZxVR%HG~naR;N?Jk3F^lzh3<`lKzykLDQ#;8 zF9+gFB}nOuQ4okPl^~_)S@PXV>Y0t%J(fwsc=Q|dVHj(__eYX_82AUJ!aLd~w+g?zCf?$ypUha%|{~n?lv^%&KLK!vEPQEMpX&V56`MxQ|gI3l%=XqP>g(_(G%@ z0G!TzGxZ!bCi*u7q?Lr~f+SS5Bc=Kq3k&dW>P>jQ0S{3Y)XCvdwQ?+*aY6F)IekWF5j#{JA^XV66>^m6iL47+E0`HM9cpb zmnpn1QoeIq9cCg)-Xro%6`9!>zp*hWZ~rP_o@Vl&niXWn3^}=sA2^u4!fdSyK46n##U486Sramk$G=IX&(TlWm(QFSJYIZ`4wbCXbFeiUf|Sw3XaUOGY8 zFry^Bhu_kEXem1(?e;bE$AC3e7bS~As;ZT$O0K?X^=ma`P`>6{DdR?&3b#u=isd-6i72(E>hCt0W~ZnqVPf6)MSY${wRF zC49O*z@R(xO*)qBtc?fd7!e-apgd1`nb<&fJ(pPel@=yhygIOVst7^i%$Vr?;ko3};X|+;h(1peYnP1*gT#B>;-9 z)Xt_i_795z#ZTKXtN>z`N_x{!c%8PC84{L_J=YX4D99uqeMGlbq;9Q9-AXo=?`iM1 zoBcz;UnwQAU*ACc9d^VT0uskk&Uaao-gP>~`e}{v-!}@nKwrgrg%#UNW0@~rVM%&@ zCF#}Os$Sxv+_VQ`UFldx*?O<5fihZdj9Ob9&?O z$^OZHil0<^cA$qQ`;Hr!?1TH7)@`SCCp6@}+$YQ&oVTY4QTcrpnx#TZC;ODKkM)Y~ z(@%*fa7cwEQovs1`7)TD{E- zRS{LK-o{7aDeA23$Hv( zAGBj;Zu;zy7}a{K_8gm~Q{{0gv&gJWrpmLl3M(?1lZqsnlZqsvZY3Qh5%XHIS%RKX z)fck_=6Lfq$rAR9DdtHkl6aenB;LMB2khOJSce0zP)g=Wa!Mz`r?Ah9WP?QaZ_&PH z@(PLq<~?Ruk`m`?)f7ggB1w=^kt9eT){d?-TMh?)S1Acn^6Jy`dfVqkvO%KzkEy;G zB$(^Xup~$vHl_$tDv|^#6-k2hRkh(Zv*mCgeM^ugY}rw8jfx|anb%$<8zj2_rs|79 zf{8sOLApR^;#D>yM}Uu-Rv9V>*c}!AOQcpQ{Pmzoc?Z$`7;Du8(mHJW&4SY5rmG2j zg;JFurPyh5uYpeSLQaN2x{=)(c!rtQ0!H1*9*%BW%b^)aYY7%GDoSo&U83DyXWcIc ze%h#E;J1w$eX@q*5R24uV9TiFjq|>{uz%4X%ELZ~c7mVV=f0-KeAA3-0i&wb+q_7v ziK^1Q?Gv7JfqG>t9dP@c+6-g#2BK@W%*>(`2qwGxkm0EL;e##$7soq&s zz}#!os0F+~5%%+h*+nkNLO|F^_{`dnwmLj*l|(CT)lOR_0i@NdVgP|G3G!Q>i6T{% z1R7O1TL#*<%SQhN+WC-v%3X7fKFo{FykuFRha)vFV`G>Y>ZI`rnxM>&%xI({N$69N zB+iIjXfq;-A0pV24b{JE`$_$bi3)SB_0R&Imk9erWg-2PWqF%9{{qw3CXjLcOZ?nk zrJrD}mII@rBfqX=^#fbkmII^iWU=$Q!y@LDhs6%-sWd& zU{sYZc5hahv(1_0sBxZFVOi!XG8IW;i^wOfqa=qAp`&aLeMsAXOh0EfPSfcI^FP+Z zQsDn4!aj#cZSprWmr{{rE~O$#91!`Eja?E4M6e|rhby)H)%q#ND~$upb=JcXz#9`` z9|z2$?@Sz0kt7bOND>D`{$N%naX&N8NNtSxQ%}9Bo7Tf zvij3?2S0lI3he#ZzQ&`sD-hX3MdyTj;ay~X&--Yb{q$3M2JWv^IDMzL))#B-@z$^n zd_%2ayP)*)nN{ENH7f9Or9uzs%e>Manz9g;dSQLkw6>?*0!wE z513#Jh+CP5AL;SM+G=`Z;dmqcYjj?0Mk%&1{YoD}%nth)c5bcF4IFrp4Z_#~3SSy2 zpB!k%S~KGcHUC@Y{1EU5N|i@K3Wr-C$+Yu*gosRZk4SWMp*e~*VPgl!F-V>yJVz}= zG{gsdgE`a!e%2g1uCKUB3pgfMEx>!~a>RRyIX((}fl>pF6Z;AuaoWvMp7W znEo+fP5;SW|8y09gns61@G|mJrlZS0y0Ib3##AxJgbu*VXIC2Jt5x93m6Gpe`wI$N zX=aJsBNaX-QhO=fM?a;9qrm?Z#fB7~7O8U;5}?diyo}NFEvzjdo|UhHp%7IiYsUy7 zHm{r06~;@G_&ufL9nb3jCB&W9t?E z(WoPUjFvoR#VAD=lP}~q)I~oSB-1<1fH7b+^9T$WwoyGI4SRB1;bM*1=Ph=lz}>C8 z6=?`Y3BfMb-3o6v*>RWYUIyG!b%&>+7tWXGeihq%k?pQ=2)%9D*i#=x~0 zQ@eUjn8*1z3HSA{fV_?h5WVrwX&R0sN*qZb_#JE_ycArApg?%i& ztnwYo+n=TtzRS$rc-qm69Mo6Kr3*zm9c(O=Ueg1-Z!Xso(p)ZSyaM?H-a$d;V&?%D zs<2#O_?{Mq@cunM%fX5@Z7<{^NmqsS1ciT#h4g3pWG67ahsw-|iBa;1qBib!5MLE*wkUNk}By^*|Ng2GRf z6cd3zS1OnYp1+l)Gw@eR1u^hlky`@(P2?7X?@_Mwgu?G5dFg}|wyruB$%`fiv@k7_ z7feu?sib%U{EAY+MDSN5w*>t4$SnqcTe;Fi3b#h`(g_Mbi{wQU6z+)R1rroLvu))} zum}7-<-}~@Ganhnz|U7s#DGIJG4OB{1CG_iz(+(e;2t~0PAe4uXUdVpa!~^$kE+Fq z(zkRdMG(j^Rf3d0sbLa9Ai=5xDJ^w=WdS5^dpw{-|o`gIfpGES9^mG=05I#SKtNHq&a z>iafQO(2t5vUDYw$t>K_U?#J0M}e8k!i|BM%EAqSnaRQ}0h1HLEe4Ym@-#n?i7a9( z!AxY~js`Q4g*yt&JQi*Ye5Q+H%xvdT@Aj@VYNa=Q4L|Z_nc_m1R@XLV+h-Z~xZ|BBx&W)LK%p&>j z1Bg4JhcR=Gd=^0C}`40w-H^7jwd6ck1^2m_6YH3bDyuu@40-=%Nh5!+R+ zmntc|&PY*7;X)-Vm6UF)i&7x7OQjUvZ)rXTB(+O93M97}3EV?-x|E}EU~TCfD=0Ay zN>K_-1W^gRysnbM4IL^eF=a}X6qq!k63Cn>2Z91=CUQQg4Fm80_a91F>{0Xa6SadQ7T{pGEBpH;OgvWf+&R^?ZxS{17lTcT8rYhQi=s{^wdp|vVMDQi`JavrLalWWXymC9AoG~LL`n_r3vS>TA#3SF54M+7*E z2aYwE5CjhEf1yiJ;8*XQ>vuEzEi5Cz_zoon~)EybWT)BijHNQP>Woz%#brh8tE31=A3ukRoUVy)=5(FT=XM(0Ho^Ut9j&`uc;hA)(Y5Po z+zf-{NkcF~2)I8bcS?X<#mWHw zMQ>FRd?Fp}A)%Pkfdg|x=+AO*$PZdD#HVrK{Fnu<_lmHQBgP^Tl!NcJ+#Vv3%+MeP zCb;tS{=+V|XWKoeA>i}ux@qN_f&vpqe3)to*s(+mxTPHpP8tHo)urb^4a#F_;bl60 ze!=@m2G2}m{gJQ0hsw9|KrjVoh6-6<<`pp9PLi4>RSgU`m852IRRhDVrCH(0fvO7X z_nIYAb*WZCaidA9F48I}wqmK0*Lalnkz=fs<`M*n7G3iBW=;HrCTvV_R#s~O!QvYG zfZ;AMswi_O*i=DqKbQuTS+}i$UkGB~J{PTtUlU@JUMoX*IgNauxM*&YaxUMWslN6`bzsVy3%;{I^7+Veyr{c zYz9huKb7`AD(&67QQcy}3$LrFxp%3#SJJm=k)=+hw^BnHc%|{<+vZ29#BXXUo%&Hq zr+$>Q_rsqFkfLYK24o0NmdUgeMwSawhK9wOno7jBy(QK#mQWs78ZhJk_#(e z$@_rnct%+=D6=}ngT#`hUG$=_Qypr^xD@h-?*SPx`a&}-r_^RDS9|DrGTgki2 zj%&0Axyr%c?O;5bm~!Adq`4f9j1F;w~zQGc#Ez@{QXse--y)2rKit96<^X z&q>ZGl&1P|C`m?QGtGQrM4rjp>hB7_;bv=NDS~G;!ccf`E4()s-mg{gez}78v+69J zRm>B4U>=i?v9HFC05VDBmTp6DIzw@VC{=6R{b+m?-QGQ+yaz&Yy%n#ml4td>n>74; zPL8E3uZ}G_S^i3k(Ip{E(r05y`mriWpNA#shp;4l29~5xKMsU{ZRv;Ie^0-1M9bOd z$M!5e@x0GYK8FMU%cvIcBSs~^S@ep+DFaOxYqRisCU3z1YF%7^T3+M<72(jGIY?ce ztdM`cLj`92wVxn};a|aY@gaXG4hw{Xg;_S|-*q`Gl)SeIdA}IL`}G*!FUjzJR$$*l zu_}9C{^Sz?MiRtwf64A{zROb5$F3xO;7Za*tt5TOZcJ8*&3BIS_W#Kt!^po+!*ckx z)&uSn?Vy`nbRx2+iiE8Pyq9t5*2Bzsctx5UY4PkQ_jUZ2{@K0!;pxwFbp$&6d5XJ# z)_nm_C$@ycPwe+-AD8K;e6b4rAEl)TUKzJ>->Ll#%Im?yD}gCrru-Hq-<<4QJ22S?d8Jm)YP2T%rEbM!AJny4 zHB+l5`(Qp>Z?tmFWZz^T6duhK6RH|-E@sh9=H>B-(4Rb}07Sq1g|Fl_q%=b&J8e^; znFl8s&VaKzl$ub>>QItA7et1YB(sQ%_LB79j+Uek&L8v!_#^G;-v(}PR13I^QSDxe zLdK(MjyB6(AWT;SeyE?Nm08a}r8A+!b8m)MM-F?9@Lnmr*Socvx1D}YZ;XYMr?{ii zvje@XdJ8t(VOlp!>yDW0gStekLh?#0sevIJGpZV?Q}17)pEYezXskbZ@LMX*^189)*Vst`I%ja zK9FnaE$~7zTg!_xFUPlh|E)^HH*zvtDPEy;`Of;WYPQ0CS?kKz(Mz4oNhq8O^Q)@y z{n7!Fndy;=B+p%Xk!-wDq0F|z7b)0NYI=&U#=ZBNbRuL<4Q%X8rL47HsN29HceQT# z;aQF_a+I%Eev6V{KG}DI94#Qfp_Mx{PHpTv|K!HJ$$pVg9RB100=KH*tj0(wB4-e& zJG5$cs9L$UJfFbbU2hiZCi|dpzC4ALH9j@)0A-m|2qM@buhj>lUrr&(OP*;~PhAHU*9!B5;@v9N&KN!G!Y7jvDNqD z-dqvHT7FF?6p&wzlwV>~;MZg#2jsV8r5pu*N>+7DiC>(Qq7?WwnWzNvTe4D)0zW0I zR8k_$rKpdysPthLs&|xIX_Q+;nO{@Nr7*A_KS;4~kbMC!fvL&rbE?|IX-Y5Gyf{JJ2Wob7Z7+UXE<`L%u; zj5fjGr}MH30-28?JK6;$rlbf0nPrtAC4M?Dg1~xiwv*fxK_8MzW3R*vkC*$fl$rh8~Ab8P{arr_T_sxnBPyaHbI z=!%lpQ?4l}kVzs3B$I+AVDd=BfUMm?3{2oeY^u1;tvz@TW?Ph|y+Ub^Qu{oM6>f*L z3L;Au#)q$AcqM|==p*(pycj`h^bva)UXLJ;(ej8r)U9fXRPl&C6faAVDjuK!^CNQiX)&zo~k4=0ASQ9^eVv}AYFY(}+bwBp? zDmVJIDnI`9r<;8SNS!_D?sGuu>`8Z@1szYi`#k7)(%ol5$CK_p7doDF_t{W$(dR?0 z%4bB~d!G}vWHyiFHDr7zdw zA3N!Fx-OEzD(y{F+MB1ex9r1e88dl$V`<&8yUc`=_N%c<6Zl*7WjRXUr$u6~^eOu5 z=jcbNJN2WaQ$I@D`{55g<%Oc3dlji3GmT1;0`o`6p@ISvN5~}w1tyG;iwg>jZ}sdr zC1yzV>^LPNUOhYRCrW7*`9Y*g(#NSJeUwVl$LMqVnD{sJQ?@W57nwbt%|-*_nVWrI z-Q-S&bbyZ0&&*Cc)?0+upENC9Eeaaqby;My`ANB?}U zF!9;3VON?2i)m1%4+W-ykbmAe!N_c;^v6jip4=w-{UUi*+IIu*LkaKa8oZxwD>d`p zV;MLCBq?hTG-Cz1SF1`hZ-gb4wN$=40dmHzDe#`us=PC`s&4*}7k`sU!l02P<*hl! zCMKfXs`NoEC4D%vclJ09kPR|;z`x4Gz@NYA#CWV8W)8L&0xktU)2J5k`9>}E)5t3d z-&UgOP;HjG=Q^r&Y&w}oUS6puE0%)H_WVv=j7T)uAAqEMwPBI-tVi9pMa*H_+t@u2 zTUoZGA0RhaOO+J3zgk&IDb7rJd77V@8@2}$nTKZ!mE-_~pC`ketLm}T>`cRaT z=^~Z+j+1bZ6|&1s->fAS)Vfn;562lO`9s=S%C zs&1AN$&M|BFqWi`VM$_N9M?_Se7NBIdzzQy^c=mP{|dXWpWN0%$?m|N1xO6rc5{`(8HHiU6AtwcfT4$J2^RgR>f(S zIVpTgd-;KWX3iTZyg@}eVo#!#B-z%nW3l3reWhf&qs!3woA0Hhck>{%q8SYa4ja`3 zE;FjtOHqg^nwDy_+zIVCt-D%3JHjzb&t@gLqZT@1wBh0K2ETcOJKlVK$i7@&Ju}tt z58n3#@5dM3`@}ubcdBpU{-r!hL-D;z&knRa^0Z2xq=Do9=`5`qo$P~RXu|U{(n@Nu z|7*H_#2T3IYRxQZGtnp;K~2+f42?q7WFHLSh!5uYfW(e`NMel-3f|*}tS8uyNkaLa zgz{Yo)m!&+bQ1}Ab_j@{{f&tR7@xxY5>;BtR zCoAbUb$;BYpP6mucAdy&k0H?~b0Xau2 zSvFdeeNc0?s$6ZtPcEbm?r~bTRak98-EocGipf6cC)Bl8zt(`l<#LUXhLe4MXs~-* zRU&_`5=i&5Ed%Ltn!MI!x8xLnE_{;748)M!4?bvyOm=)yq0FQp&QoggLNe0jWv7n} za`QDZgYw6}!bnqort({qygbWyii|Ymp*AMfqn0w7P;IT+QlG-C1{3x$t=+185>p;} zt*W69uj{RzmW9IUa;&9cI@VbJbX6kb1H?EPYam_b#~NMOEzfNNF(mg`EzOY0j&&-O z8S7`O6H{vPLfvLHa=5@)Q$Aih>lkau6KqUmdP*M1SVNtnRa=IcD^=BC!k($M)x4Hx zYpZJLb#=YfL%C2mU5<73pr^N+ z*52)`CNI=&UVFKjjTPldEuAk&UT!fw9C(FN`GC<7&t;~Ot9W$R0~)bHS9yn#qSE~4TtXU#*msagNr!qqZ9=M-Tqrhh`cS2=F&^tu63=qeg&7?P5|Di2tXaxAApmdJeH4Mb$N{~{tBN^`0jwW^-Xm%_I(sNMXG&|ZR*aQ-GSy_OujZ(l1%z@=V zOs;yMMA+q#X<+mznUMH2-Fj>R4=`#Ou-$<;yH! zz0)!4l>8MzAkhn*jun)?Z*GhMv8NKGbfvj527IFCseCmzQBWX>tJajBVEH=+#ND!W zOy>%v4lm6)yfowR(wxIfGY&7!IlR<{W58zNQXt+0-gp@Y@ZL-jL=f!CN9$&&dpt-8iUn%)3E(vnV$GaU?ublS78 z@WU#iUK>+qXUn-3aDh_O<cA|K>vP?C3gl&ydbR<7q0J0iYZ7F@!#-`0M zkuuWO2Y9M{S!MaM6!@>^<|z{eg_u`M8GVciAp&n}hE17%7%e}UCVtWWBqPbGDU$Xz zE@|bj+pdzoZtEj)vJ=Z>XgV&V$&1M!`Rz-(Td6a9!e;mg@Qm70Fp@MpTWhb3 z4S}ETXh`Y%Q4qNGu9f$)LjoV6R9IF<3Q9lN%>;q#m6E?YKfVFRwR6U`3yk}g8P^8> z*r+ieeFo#wzX_5?q?zlqX`Oz}dT0XaNwN-D@3fhCx`W1RZ~Y8+I>G;F+`a|)Fyj^# z;A4#2uK=H6-2Mgl-NqeIfG;=h$p!ce#vNFIZ!_*G1^7PWo?3vn)M4#!JgorlZrniy zct7KwUVxVv_lyF3m~jUe;1?UWxB#DFocw`!#W(6_U*m8;O9}e|b77wXOsM6c1|-t5 z0s{%ONCAnnNC646ESW%}EK)#%EK)#XEK)#1EK)!sEK)!MEK)$?D^fthD^ftBD^kGa z7QiNucyG~IckSA9rX4(gmHcTBAKm}aiNW+4XsqqGS@R{WEJ+GXmcholqX+lw@2OFu zj&hcOFsV21-1KfpuyUPl-c^WBKVgf@VZi9=y7g-;o_(6aeT~C_EW6VhL6Fk7Y%~u8 zu2-rO>^6)nZX$Kpnt~ETCPM+NsT09&>R6yf-RWx!O6ZUv18eF;u$wwIBT;wr+JX`~ z1THtK1w78ECh*NhwSez6stx>KqgueP7}W;;#HbeV zexusJt#w3Y76a!Q)dn75R15e#quM~`yyyZl{bkU70RLSTT&$mi@-@)R2j>)&S&bjj zSSMe9&6~YXpT7B@3dRS4g$VTOX?8e2Nmuh1J|^jHyT-!K?t+l-(S+zS2>D*mRlnY& zpW%H}@;wSgtRD>YNP54L`unuGBqs2WN>yhjN}t=kvOolZcPUi~Qrcz@6Rbe|pH6>k zowk90Flrb`&(-mWQdFPZctJgmNLB!8U9HNLKA{eXAQ0c=?%UB|JgXX0iiR|!hZRZ< zY2^%Q6&P}f8PWpMP3W{$P>P)<-!7+}j^#QXvrezKPRD?FENA=if&zU9OU5#mB$v0h z=~&YJpCz~Gl!W4sKvBk+z#lJ%1j=OF4DoIxR6sB`GebrB_p)FJ45}y-C{qQ&sL{aZ zi??xxyb-7T!?pDIX&ox7fS~JHGGO|`f8VLK>D8-T8WI%xt%jTuSt5c!`rovK@&3!5 zuznjLdav%U^e5?C3VVBGkH9qIL3_#!O4;7WhEAD{q7O*>4W5|jVu{(@tIcCIIrh;{ zxxMm#3h#>)a8Q@$9=~nebPey*zCRdk01`@P1N0TzVr%)6JhJ$8{hK@=c*uMBy!ODD z%b~X6{*gxe$88UIqEX9%Cn*(@XQZI?p?Q_k%$J6!pKH3>z|}@A1EMgfZ)emmbH+DN zzrb`gfyWs&3aqPdX4EI&s-gZVrmF>fl~Kc}U1DJvev85wnPj(rkt&FaMp5)mtr~2c zC@D0rpuF5VXaZlUlpNQu{KQ1sZPQ2W^{RX0jHMNpJQS9_NiD;r8S?jG%3Z{JWfk5x zG>mCbrZ?7o@)HvUMHI_o2xNRz(kndsPYM>T9?D>STrIm^KV{K;n|l3r%aS4B`;-a` zmYfl~{gJRu*Kn|Q^qgn|kc8`D!?QM7!_lgNw8<=bm#WAY%*+wscZ?ba-lJ5t=u!Iq zf+}138pFV|9#>J7Af>U#n;`J|z3PILRz1N4fqOl%E=cLy%1YP3-JVnzq_jv`5v;1? z|AYW1+c-#OVQ9kH#tBw#WhTxs;=T*CGd;vv<{NDMx`g zR0dKa6w(?<9Lm-dVo+AZa8xJ_N;4OfWgiVh7Kw>t~IC0zpNF+oKNJ5r!6i7*t1Cn~B90ih3 zDwvuQxC8ro7OkfNNZpQYP}HFa@TaJU9kqrj>FO7z~jEGSf;q3XF}& z0hwu~90jJG$oWz8nKeT)t~_kw$g^z~)(Qxms{KY73g>1w)?LEw3^;tWjv(-~-Utqd ztq}xn*Bim%y)}X$9Q8&%E??3a#W>Gstn7@-{eG3TIeQN=+ zm2QjHIRt^W7DShw&ssnbWowa!R?SVrrxiX#UrR!CN@s{T3CxgRE0mFdAR;s5x~?Nc zuP}$T1;pN#h1vpwNNiP}GWMIRH_wR2UUmXdg%Af>g(`%IfzlRfptMCAD6Q~;`c@LC zQ#%8N6Eox+j4~b&E6wdSRRw-UE44*xrBvD6? zLE9-w!BUt4`6$QZTgEAHHZ3J7a5j}oXCQGX8&kk@x~I%^ER|B=R9b4FzzCO;6j-@X z+bbkn>Z&C-Piiq>rRuDOB`90!`c<`sB|6lGfuCoGSyr@8&9E$Mw5>G565`Ibu&jKu z)g$>6-vt~aF}8MJC~OUF9fogZ>Eb#&)PKC8(mz_u{~R9TV-+r~UV zgmhp(!Jg7Hgs5b1=@7zwDoL!Jw6juE_>W~U9=6G+tbbIpF?Do`WmK|BbqHY@l_Y3~ z5SplDML|gG^GUqD`lvm1$>qFMuausn<5}%W^E&Kc5hr=4l0}9bo})JB^diGzhHO30 zwOz8bO=W}6di3UO@X0@wUOxwROy&UA`XzD4WGW!(X9Tv^X9Ti-VsyxAQXuQ+$L?L{ z2QBa|gYC(sal=tNq;Rkac?f6hc<}?Z{xi9qinhl~YfsDTRB@tLcr2 zfqfU)x#s~rgDOptRcvPYfLv(6Xs|%U8t+T2@w-~A>1E~2(#r64J#!t)Vhs$o$}@uU zFrU~Ont5|OD!s)WmEP!%%3gNQF6~}8KWBNp&f6}RDV6P2Q*XVi_V&AK?}4k{Fdr7} z+TDk$uB#7Kbypv%>aIRi)m?q4s(bk$H}pvy?z{`N?_;WVKh)2G#_@dxg>(L)suZ%n zpzx%}8aYr<7>?vbl4#P6cTFhoj!^0C_T)PVe&t!uN^2ToEFJ6CLL4AN2jCG$El;T*Pz!Ym=93B&WFg&4}$bs3(_*+ zz7~%)eFcRdN3y@5@KlS(nt_7Ce?~G5lC#2x$yNJ6x#|rIRW+CLp{nY-hN`M&ZV@v1 z-UN85Mej)9ODs_1eFcR_SxmKU?XDpTKQ{u;Q!e~5wG%{^aN=>0_3i}@ z2Og#02L^5@y>YU$e#%2+^8NOlm5n)tw?2sD$rJTUQOxyj`CG%{`FHdbK`K>WmQmtZ zErLK!*!B$^1V>tt1#%*lXV4y=Ujq7j!g6b-zsn7$eml<)iBgyXQK-VQU;5o>DV+(- zn&ifYsbC`@gQ|N({kCy&bS_m>;4EueAjrK^nnLI8r}XC(UvoD)rSY^b0h*Qz(6k~S zpnbCenpR{2G~OjZq9PxlX}JJREAjynEtvpKvzgi%plQi>w{!I#D;KMFn;s%2#^3mt z2SkDamsA-{M*pw=ek@)N>&_rgr(HWD5 zs8kP^XKu2FYg7YYrEG?8RXDKlB}w>JMR$I-`|S56n8F}{E30@h05twc+1=u zKi)o9-Ob9^XhuoGuOBZcon#r>1lHz@2zpbi=FFm(#`z+$lvc+10<6s!5%e}Tb>&;8 zpOULUG9~2diQPAxrJJl_7v0$&8#g`RlWUva6gDi_WDPh=_V@mh-{*?8@L6INKKAs< z%;Q_0`chv|iD6TGljSYSf7Rp7%S&~$b#2V$(|y%dGRq}LSQ zje#m0lNW7xRS#$;)*@0%zhSjO{1KLj6fq*geo9NVI0>7Hld#0;4;rsZ@78+DrWQSs z^_I?1TmB)&1-Q3SM*tTZl|;`g3eoeL$~TLs3WAV{-XryXS1o!M#L4?jU1h~fVKdPa zHWMddGjS5uN90V^f02GlJ^?>$)Dggs8I?rDD+&?unnFZSg=3kBFvD2R>Q5uGsF9BK zI-pQ7f@5A#n zt6r*~@>mvdty2B+=(GRl%`4x$>76f0Z(K=w+p;h2#I%1>J7zZiKlaW%K+meW|L;ta z$s~XvDk4V2Bvh|wIBh|*dY8fsIEKH!Q~D_a$@ zLfq<#VzpwmD(==&^(SssE5CD}dp_qrbMAd-CPJ3}+We9E-tTknIp>~p?tSjFyzjei z;L?Hc%>^*eF{iUT$~eeuY37j!IlH@V@GaDhjs%wirE`c}3*}=el#irP{c?plDFoSU zdW*_BqmlJ?BkN5^)>}VQNA?=aRR{Pkr3U40zyCK~{w7?orsUQQpn1#>qj}5^rg>~8 z-|Fi1PSvE(VNH5BYSP=UNpF}~o~r>!GXmyS7Udn95j?e{{FQ8S(wYrj1ue|7bGr)tuNvnG8sYtjd^CVfzd<(oACX-2@@WKrIs8Nrj-Jtd-2 z#oi^Md^Cmfu@uTjQYarsUtChTvI+DSBkPSu*4vG&Hyv4T{qvfm+bvfe;E+BEA8c*% zy;|Ps=8Slss@hZXS68ohswRCnYtl!vCVen#(np0@{<8)k%?OwuT9kKaMtpM*i>Oqw zcS$H8O`&`&h4PUU%E!?cmsGB7z`e!DdZUr`b|dReN7h^atp@wvTK=*v2mX=Gh%*O# zH}7}O;jBp?&6@PVtVtgfV%gHbr5ORUzeRb6W(4nk?I~-FRIztS zC?8Fsd@P0Xkrc|u(HECgu3SfWi;?w4BkS!()|-y3x4u+!w9InV0j^R?t|NA6M(m^g z_LTh9)$5(ANgvLd^wF$IAIzHU7t6ym0BJ_RJkFxLLoyTg!0i8%EwYD zA4#En91Al-t|Pofsb^um(a3tck@cn{>#dJyj?T1Pb%3u`O0FYzXhu9z`|TSlRlU=*)Nt|4M3U^FzYPJJ2WGBd2>&Rs8q3cNhlvpp?oZb@{ttE$FVRY zj-Z#vfgN9 zz1_%q(~jiiLarFRMXBeVavd>8%l;4a zuiR?_@^LJ=ju-?#%Vx=Bc1zX|qqt zOT0@vd6vjkgtsWOWT#wPJVA4`+H%zao~M*tTkOzzW0CgTQ}S0Azjtb9%m`xn6b(R{ z5ik=Lx z)_!|R{_5iQPVI~tK`hs60Md+rd6`9dhh_wyHs4bsD&_B8+Q~CQt}VR9cxB+XZ;|yo zxXAiVTx5M|zfN;>jpeEX{DM+)ZLvc$Vzc(!Q}S0Azjtb9%m`xndJRCD5inO-ly_)G z@YMo)N<^jny-PcJM#xo!wfa@7GY&~KZTYl|J45${x0drJQ5;`dJN zj2S^J|4{>wW(3S&}vp;Gs({-hAoY;S&xDzsa>W{4&>}M=X_(?kv&wYW;dP z-Q}GBu8HPp+^?+7V}5;Y9`mbg^Vm$j$(->n?c`}Bml@uoOrxE0@$p+t?Edx*?Ooux zHjUP8=3^1R(~n6Ew$^PXro84KxS3*ESFa!2&M0Zd)zU!>%P;B1SG-{!Y9r2m=F3;7 z1Yh>K`)CK2hsgShjI6K4$PzI=0uWgrv!|<_=a_>lfu|VN1+F)0_;Sez{;oz z;G2{RUyahMXniUP0&kuo*w5%wtz#E>g;A@3mj5ZE-D|5Py}?ODe*@C`nKX7m$D5ZE-D|5PzXUuYdWz{`vp1vZU#%;B5T zqj@b&s*cx~uI0ct8Z{1llT!6>h10k?$pQa9$!)A?d^E`cKau1nD;l>ZIp9~5+yxbl zA0|2AkCNPmipGB>IfF;UkjpsEuLiWSzy6h#0XHSN^%adjO>)3jCAoDK zjcrK|cv+HLThVxTl56P{q(%!L`kOc^rs`zM6S&VA3_qg%n-S(3Hxy=<+ivT1j08 z$S9-WOv4nZtInwMk+uu!M6jPaqz76nfDJ#k_wTqO4QtD~iWWNLtC4^Wbt35fU>bf- zr(@W2g1=+jkrnt(~8f#>V0G1z)+1-_qgkE_5(8FxYjZX36(`l0SA?$E!3trIFR z>spBE%yB>5;!a#;y#W$d$t#ekiWHEbiWHETiWHELiWHEDiWHE5iWHD|iWHD=iWHD& ziWHDwiWHDo%4z{5tkRK%hv*6gB)HNMnBts&Emfd6dN2=K>7^?(bsR>gbZ5~F&+ z#~RfJHnNe2zURxC^E2n)a5)R!X39r=w=H_;ZTgoVsrPbkadMJ;FLINLd|3a=fPqw# zoUFHHn*sWLSNjFe!2?G}D`F0~*r+b>1fzPu38T8e^Ns2OUuRSo_+F!Wz%Loq1^&XQ z9`Jx;aJ36O(x@KrnMQSiFE*+N{9~iKz*iX62EJOUz2zR`OD~j;z20b+C0*1tzND`~oJdgRNsKFmWAh9b184r=c2b9an+hYuxb__)Eq; zvI76YxJOlBf;-rHbOk25gRRF@V8R>DFJR(3*m_(ACcxpm0VcA8t!34Ajq<1TZ-^iG zdzO@JVxi&+6{$;~#*KDq-0`!Bbp1tA4g4#m7Rs+u9U{BlOXn`G;Fw|Z4P}>~ap_zR z_tXACRV90`Xr8jfdc-k8n5MOU|!_B z$@?4^nLp6Kh1q;3;Wd#FhI*?Fbp(jb1@eQaQfy&0_cs4f6_EkCk1?}IwicjeU-Kb5 z>*ONYw}>I~%`~`~e7Tp5Z0|ee-$BDqIyh9zmbr^&F~+PWF`dFNHj`<&dFVyv=mBRoREO@3g4f9wmfv`*Y!yAHh>%*#) z3mh|Q0!52V(FE`irH162b)*_9+K(~O4)94%ls4||6CM%DPyUFSO@-^6+|_pO>KLwm z!n_{?;@q z773ZPK!UNyuNOFcb6WGL&TFkY+rSSgb&oK40q$mI+0 zu|`b*mnjw2-(E%Qxk(WC{3(L{jDBB@%LNhe1k<$&h?!utYet`!1c6PX`HN|c^yg-@ z4Scle8Ur?sw$13tNf6jHn!ikh(fQV~1ALTGqrj%oju|~k1$i-z_LwQiPQVB1Ax0xF zrU9Ea(!e@qBfgkMJK4x~7HhT3)#gSI_(`Kif!|UpRt~MVpFq`(0-yWDNJT+f$2`dd zfp33uQ;^m(o??Q)Unvze(z^1gCI~$6X-$o^j#9R8$A=KS+s4}kGH!`HkU_^frp35L z5XhiqPG7%?2E!IPAY-oOXfSG#12W`Vjs}AkIp7T%l3I?&r%p^k=~lD|PAy6!MJl_7 zAyULbRy`nr2w`bgv>y0a($fYWrj-2-AjH*47Wf}Z)xQHs{^wRu9pGt=dC{q8O`4A#VAGXOc1p-y885jqZmtj` zxs3-TK=r$$G>C-A0V&B^js`_3azIM5mZL#YiX4!U)p9Ai5xqu3#%TU$RoK>S^#6aa zynejl+!$_Zk$skt;jmGWZNp-#XBJs{(58+TgdVc)4lS;rsAsy+rlcEPpeSv+(5AW@ zU7)CPy3l4JZghcSHBO_1VM%Uuf?^h7$4?|5TA6@8s$rG;@3%tufz=m|erQ}}Qb4SE zwJ;3|LtYC7QWLct4XQ!pfD}S4M}ra&IUw1uKLn9u02FR5=WDeleU(aQKh%7S)n%yjY(Lork~6vIFFzP} z?4nBt7s^wxY;MaXNIt%9K9sKaCg1b+@Mgfg=>eyDb98@idbu~f+?(uC)BCeBSdKTC zmKTtLhuMZ`W2;Y8mgjyf&;3~Lqw>+rkCpiGgU3bTf!0dkPn41q`-X}JA=_6Ln~ps4 zlE-7`w-z0{sNw(_zF!b%{uFyJMbXtOA^6O)5UAdcoiSV<5rvFnJHK&rhS zj*pDbQ2qO*rB$fCWXMihY=LWG8f<@M-h6oz4K~2FBn>vea`P8RMC#5o7Q2v4nn1}hXAW*WVGf>$0|E-nl;9r5EGu4l6Js{Ei+A3yT&GG*w&@F|R>3x-(i zKV`iLWs)`ogdo|2Ar48eFP(5kZpR^0Lexv+&`fdLE5_~YV(6wE^wHak=t1vO7so?C!pKydq7;&d-lw+6T31Ihq=Lf){DFDEgDm~>=udIhCt!+4J_mbi4r2Ix z8m1ndDHHdwOq^d#hClA}AvY~$lrgTNAxnwx@3XSEWhM0XiSgQfZVoKwCTQ~MS)8TE zGE=8C^zrM3_uVJSd*|~Mmi{nTBu}DV773rKOkk#L&1;=A=g32gmkHe^;Rt6bc1qfT zCL>%BZp{qUJ)oHS+Jh!zYV?3&_oO{&GRQ^`C{_yXL6dD&qX!h5tZ<1CM+n1CtI-LH zRb0)?6x885sNu1iKaYi}R9ls6s{9fx_4DDC&6cT!P6$^sDr0TPVC!1Ioi3LoY=X4 za9^1)~x%fr$0;b@BI0m-hVo8oy$vg_=ocpj8q9+aLBN>jt^!V-(O9F};D zuuMLl5?uql-KusRNW|nlIUs?G&PAngshZXR2~<-Yjv^B6KmyeihhxHvcpkhp<=~ZP z^O)Yjc!B;MlJ}g~wjNXwoZFQTkI=zCHrey#-J1thtz|KQ#3ji3I0ViQHgMPLVsL~0 z4cAEPS_iFd9h5yP3*tWa`CYsJNsq<+jXb%%Fqmy27} zn~Q0JDFYhHBKzOzh8qLgR~J3`FqnKr7%jpnZ~XxOP48z$L0Z4CyPRDh6&(eAJ>OR! zz!OM$8IlLoo9&tP2=Gsp+9%%3==U1gUDxnob?fHz$^h^yO3mQIr|z-_|Lnf5F&{rK zR62WjnQ|t!=*@L*w(qg`-Pt*C8(ka)vT-QhQD=bM+wymn;8{cN-g{nnjf*qhlEU4m zrx__<7)Pt8th~MeWYLxMiXe+9+p@^AH#x@M)~RK+^#uJZ3n{QMdkbGz_Ytj`&)Tz^ z5#TqJ3e#k=wfML`Z+w3GE^D|*mz+y&=@|jOS*aO(c-39jz?wh_$vO#Sy^Lof?uAzV zfsHZ_g6`kFveL4}evVB&>MEth0FFt8yG0gTZ9O3s zF0fI!K~Qt_fK<4`(t2-kyY?j$QL`05ids^qfds!iV8 zPif^gBG@R`N2qr)eqo8=1u25U79uEYA%emZ!T&pD8hD?+t&`>d>R*Wg@V7>-0^X?< z5u}apg$QB*6(NFPx`zmYQB{oK*YXHno+kV|)jNrxu!UF&TZomgU)LBlVkNAP)v2og z91F~9;G|J2ft!^Yk{eqaq}pg-Xrin889+sd6`1ZJR$vBP_=qyIm8DU_dEF*IS!Q>BxHP z%;Y~@7LODjI{0|hUtI1ZQj^}jn)Ke)q<5|+y>ItbpYE@Jj? z#JgP=3jIGV@68$N=z9H|DiUUk9+~YL$6k{h4LX4$_G>^A5NisFnwyJMr0G^ortXWBeLF=$a-%g>m7=$_X+FQTN>AR5$|?g zDD+Ed+(i}nCUx{}`ZrZ1%+-2iwiU_H(@=I_MS5SPihLyq<;y@QUjst<0uZX@{`u5O zjmS38I}us$M`XP#k@en0);kng=TZynqfdxOrr9zwmFtS6UrOUHs>shRi;jLl|E7wB z`Klh7ZAJ1zPa4YZt4QyQRFSU)p?nz#MI+Do`$mf zD$@HRRpcu{C|?Fb`5F+)7l2SH5)t`$NR7xo(mN4Z??+_4E0OiyMAkbLS??3pmsuKT zw~wS>O5-l7$Oo&V57ob^B4HkBLz!JgGW0Z*-B*#`7pWp&2}1cY5X#qpP`&_!^4Tv` zr0gTT6H+m7=$_X+E7 zury{L{Pb?u`$+nwH148`{4;g*)mcTtyxxW~yET%br=je=iuArn75PdK%9nvqz6ONy z1t65qzE7>xi0l}>6Or|PMAo|!S?^6`y+e`pK4JY+md4rbBk7mYxQil-*a6-WRDNUkO6_G7!qwfKa{wg!0)hRHW=9y%SP7!g@a<>s^Vg_a?I5 zp~!llu>NaHu7OqxaLl>BR3@5`>QNZChvC!`{U^)n-~ez}gUU$Gvvn0MJwW>=96Jq=~|RiyW2*H@(M7`+oxk-KmoiS?gb8fUkUq+d$oE~?1Ss-yp8 z6}evDMuYhm8%i5^TPEz!6@IfUS&+Sx|Dbhb3+FG9Eu6oEEu6oEEu6oE^*$Y@JE>!B zN3{a@7^7AKPf#knCMsW)R?&W%&P9_^~O1{fs`< zI(C7TQLBKM317h3HKQL*g21NH{HxFzY0o;gfq!Gv7_e!yZAP~zL15Ep{#EE0eW7*i z0AFv^D6na?V@BVlg5kSM*Gb{hey=IW&MfEYAx0zLcMWXXNCUgZY{aiZr=4tMUxj{) zcKMFE(F6WasgSW=MeCIR6YD@e_XT{OQc;lB8A%X$?i4{^Pq`~yT=_TiZWM?=v7Tr> z*Nlz=@lrlP2*kzMmloa?UNOmCYvUT0gRZtkX`cDf2A*V87l=RdiAf;NM0K=YY<=56 zyp4i0$Ffs$xML1uUtUT9Vmw&rRJ6#Y2m+g~z5C||1Vn=Jf3j;EIkg7Px}QAg92GvqK28`F;Bh@@c2V3VnR{PlgTMGO>!H)GoULx zZTwSlnz*IPY2qbXXcIQo($D1+G;ZOv1%8Th2go;u9DV58*5ddBiRZR1o%`_c6QCS_ zj>^4lYJj8kQ3oHjbk0e0o?IJVeCgctTL(``LqqDsB(=Do%#Le1`cUsT{u0qW-0w?H z)_CJDDaUUu*hVBvQOc$Ac;_-ZuF2wUe&a7Cd0i}YH=D}t%a?vlnR5-Pru+5G=`Y$C z#Vi%(qriJ6RG2A#Hn9pZgPOC6{^7^UeZMU_gI=$H<+KO9R;jQ)_9|LmOoG6lJu_;R z6ASRDXGJRZrS%{^J&Pdle5K_3On$IYzEQM-xI%vqCu*Q|m9nA%c!XZ+L_u2DD=UJ) z&rK1eb?!=Y9{4h)%qNIXC0T>|oFQ5t(AItQuUH3ueu^NiXP#``RM+5J4c-5Cv&b5Ml&KHAF#L=jgSTXav5(sIEb)im?HOPd2Ix zw1R1k8Elk9x5{g0tW-PQDzBZfQtjl|7J2QoOSRLs+PTr@dK>sqE0pCxYG|-^ZVUXm zBnG62vd~jA}U=$5>4*2fk3LT8;*F6!O?ERMm2-idu+*v?vGp6h4r| zM?qRYwK7<4P;*}UlNZ8r3V}RFg|zmv6=ONDxe(;nYrbG&ON0I_wV0l~(Y^JPZDSM191RY?ys3aq7W(IgDvypK#D@7fCpPKi~~0rH38h; zb{Abrg2swTz4#wD3=m5sP_iJfEIg59y3d9dlxnT;Sh@sbX_K zSGUy}oRg0E(w==N{LghwX0prZfMhaM;keCvk|}~flDX^hdQr+NkYq-0X%RZ{7Dzs$ zAgxArfdn!N(y~OiW|?zKe`Csx7pB~JY08afJM27Rt4Y+y?fkkL$VpXxjr3&F#2GYj z%rMTDfnz>#W(=H6-K|TF#2VFX^_z`^P^v?tK~9VHCF){o}9YY z>oP&X%l9{2q$zr}{ykXksh)6<-1=CybbU+y9KUq&+VFSr@F7cw5BY-%xl2_3kk+Qw zr32ITK=NAcvAEe|(?I>z0Nx6+6wA_bpH1;m4!-%I{6zEqJIntXARCkLxPo*1^K21a z13X=+A-O#iZU=F){yQCue0i-mdWmLWFFP}>dMhUbI|HnGm_l0uTC04T3tz7BH>#U? zDk%m2iyoTO+PtyC;NPp^gXON(Q8B6ON=ZfTdX<-?N@S<&f#gTEha@%i*fgh@RAj%X zeI%)+KAYy&y-Dg3S)@97q_R8y6#XU*OSnq`zDcx}!wKWSrtFk8#QSrSYNw}^HIPx%W$pX1NFS&D7uJEU z%eI0;LFty%Z?zw%GqDTgGA~?eK0_A))(H_~iDy|C>I_{HSR#b-{^H9|3jV_3Q&^Zd zxtyR5G{*;rQ+wCZKKjr&di+|cPBnYN;po(!-si&RINxDr?@7^X7j0EBx72p<%X24} zvkmU)`Xkrg{YIPo?sFT7GblHT?)p6=PIcT83VX|LMT^xy>~PAkn>+pP^RQdu=8p6Q zvi*-nXg%}f7!MHyHZSt>`#i)wg{#kfS;F<8oqu1p6vtj6j-$kpo3U~W707Ma9ltM| z$NcWTFH6jSV70y)NE(J@Es#f96;bEv5Z=#?Ijv0t6|*B%+H5WoU#FK_n3fF@R@2m2 z;|+>bUf63$NGj|#c{tbP#hPuwa;N@x7S}U7PItfPO%txeY)d==dNcD&=Ah+FRr2=wF7Ae032$h%HY}Qffm{*`J zqQ~lfP^A7=A;l>akd;(iU8n0Si>gS?lnc4Xd-3yzA z@8aWi;W^0`plpLlMWtF<#Tc$1jr!#3=7S0+bo(F_DNo5kUFStDM}uk; zIUn;{E)U#t0{5B}xLXQ=%XZPsp;UqYyDg06LTTUqkL!KR=eEwg`0$Dhun^Cei;CV7 zIlzj*hCGVXq9Em);pJV4B`-TXds)I1`NbeA^2>pVvnWvIb^wB9KI);O*_cO`7TfgX zCJe`X^gqA0CKjINsWF}hJ>^RW~oU?WCX7GrdMDMr_qVkG?w zF}l7KBM~jc==xHOt}n$%`WIq!eJMsFT8Pp0S&X}TV(N*>9` z&a2_>i|+C{p_&5_oG?Vzpk6&iI}{3MQxOEF+6ivvwG&vMeu9(+K65=YgJU)Nfk&3H z-+fhH@btFg%oCQG*YH%>f~Ue3JQbEOrPXANm6D$Ut}=F+p#6Uqv*WiPi-dJXA* z(b)4VjJ)A#{beeC4T#^btB>fYubL!Nq@yND2=kyOeR0*KudSN&!F-aI6fVZe=6#H8 zVG?XFP6A=Qn^A-JGqT>%$a+ujt+!krVh`@Rz=s;u0iIw~_E^#@3U{v1bcA*b4_>_k zkJn?Qqws9^8eNT3VQYAkuIDJjES3GFT!PSI&g#YNbQg3!NAZeNbon&d0qTiq9!--e zrgLOzF^iK!FiheoZzI-oiy0;5OzbF5i`CG?eJ$956v2py<6SS@6pnDFA{=RWf%U_? z>*ZYV|0l(>kFY+YVqMrgoq;X*FKoeoVaW|yh4Q>}-tRxV+ZNN^Q+nT=Q z;$kGaeA(Fns*L4RR=aSY)os$#tx8{jridbBmPAFgC`}Xh zF9q*uz4-c?tw~=9HR*#?lfDdUlE6`~HR+3>CVdUutSQ~|gZWeSbiTcX)d4Orstw%V zsBDqF0aCgnOVI1p)Av{qy1=WA8Ufy5RCdYX z6@^ft>3!O*6smXX%MrXJA-CeMpFBd<@2_vy)TDozrY7wPyKc&B(g&m_sc{~3*QBrU zf;U%{yt!(!;LTN&1#hmJEO>L(WWk%OvOC8GZ>}tPb7je!D@)#7S@Pz}k~de*=uLP{ z!zU_mJ~DyxfeDP#9IP+Q%xi7kPki**PaSx}hWuy;pPtl# z1RbSwdrB-Id0%E>E5!-!L)x)19w^qzWX-3&sjy$_$*rs1yfn!HHzc{C zipGUW4*2RMH@~8>HOT?rnB;~l8gEW=z`sj!3o07dCOP1ZNp4|9<0DB9_}L`4S4HCs zNe=k+B)4}(hFo2+Dtt%x zK~icg2Wxd!3*{%eP=10_(@gK)$#F7^D3BdN;W3^+*GMm>7L!LPAoB*Bw3#=49`({Z z0y4EEF6UJQ;TtIkK;lx%5t+Z<4UyqACl~yeZyY*LRtXpu$MF4if%k!fCwiQPhDBb^ zcteZr?HAVibi^kfUHukneA-J^xZvmFlA{aa@>lY;amc$HRLY`R_IS!c?{q_HKfj5g zoM5(HKo7IM9@K7){JUU?pp9^uT=Z=6Z< z`j+gTh87hxuWy|rZ%v^2op#QOJZ)9*Th1zfAtoYuzkDjL*^m`q9{Le;6JY6M7ciAGYjCCT+u zg)3qWhy%4({rX{70tm!f?Sk+3F?EFsmv6ry@WfguVw}R6eIdpYof>VMB3Z$tuYlF%hlwMwVPKs>m`OS@I&wjGX?3wSuRYr-!KxPlYXbDr~`1 zVGEuLTkuraf~UgznC_*uaG3s;x2=GWRBE_&cKk+*3WjsA2yhy4P5k!B3W^Ad9v?oTh}}oD1{w;WKBb5Dm5b$|yLFIS-~cbah&ms1 zp@=wpU7=XDDHoyEogAZ0tqA4AzeHpF3Jd=jyU1HrSU#fn1BG`aiioC2j4|xwdf~Hu zx^qZ={FPdMlm2~(JQ}k2QkoN_EoXlk_^yNvW!8um*;}Rkle(J*;14 zbAT05{HBbl-P&&&AF>5}+=sK4^8x*huFi*QRmj&%0*_H@PV20JiU!dZIpAIziZ!m=Y>f!}I&rIF@Um!Kb}k*!V8AgV%cs%WGT+c#`} zyxExsi;cV_yKU2uObduKg}an+T0}G+DgJXby#HX4T1})<1XdG)G_uu6biM zO+lpjwK{hy+n3@L9?0qQw&E#JSno%a_3K$={Q?kKDj6RmOWhK?$fm$WG+)EVN;urw zI(YWr)`6`9tPQMc=?Lo^m79+EFsCDaJn4uJXFB42_2=erDIFktHGBEdH;?)8<1b#+bCCuv*>v*$ z)Y9G+llHdOX)JHhzp_oGAV_~d-C4DzC&*V2x=cY+=F( zTbLTc7N&-)RsYyS+n)Jh~ zNk6)p^nDSnfVXk@Z1}tdCM;eV8KaMY>J`c9e8}zQqL-eMf{jBUs<-xG8A;r4^=VwbqF9D?iIGRVXD@a;#W*& z?xTlJGH)84ve}6^H5&7@d#l3xRF|`NsV03*)TA$on)DS>lfEEo($~WiG$DIZgbF^% zySY9@l1pKIj3VoU6j>jo$oeov*2jtZ9-Ug}G{=-)7|s&oTL;$5)PP&1$L^!Y#O+Y# z7jKdYC1M=jq+$zHthaRlisb0oNsbTEgg`Jhp%@RY3FUnf$~z;J&pGBugWTuA z96=V!2T7=Yitt-XY6HQ`%~%q7=|Ipk;!l={GM_ePFvEPn(Z zf3Q+t2h+f$kSUMot|e(;NXYy{2Yk82kL)|Y?0fOvsbP@!+JM}XkdKl4%mKChFl641AC%vb0{Nj|`KWFnq`%%G2Vx zn;GR}56sLcPK(koaR_Q9iqqoepNY>@5z<{gBUMgO`0$i|p9D#X;dPsT1LYLSfkKLeEu=`;J}DA63n>!$a*E_YAw|NLQzQopDH67vA~`TqisVU}&rkgF z2%j7g^qH}5q##w~A!?vS9z+9>SrY|)J|gRrA>Dl@gz{-HpL`ZX)~6ua@$nZyAN(lj zBQJtJ>`|~^%vq!4&H7e&bDt0s4S3?gN&(OCndR_|fcXhO3qO|j@uTJoe$;%ykD4#| zQS-C#xy=WncN z@C~Z+RgA#&-H|JSOgni5xZS=I>IM4fuO|Hjs-0n4aHg*%X}rL`8=(zcr<8cOsiHw; zHNI4m$x-Vgf=q;(r2fBUk26;Szh~44a8CMC$?cOg_>QT8)=J=CH|1!&K}mTk0OT9Z zYFB8aZ}(gY{DM9|tL13yvnFO(R)_7A;qf5Eef5AC0zOiyS^R`taI`kC;H|R42zw-$p>tFWAtiWby$U@fBYq&3WlXf`U9z>!n439 z>Vojpv@ifS7}b7+&i$kGuP6eJ8`Zu^EAk_0>H`1WsO%-j8`TW!%wX%-ENWgxb_hPO zUSv+|%>4CGg!rC483z#a)Amb3!^>uLvUolINgXX_!(%lt+9h*-bbj$KWHJ}YV&p}N zV||S#bi4i?Y#oyg)yu@o8SivK>f-z}^+>(h3TXt03*q{ue@CPw-ZKjgww{%Z<_fip z*^m=3$_qK+C#)8%i369M06FDJfdA|e9~Sx{R1RK2st|+6tL@X14?xbE;=`*C_dd`M zp?HNj3#mWPuYCNnSJqXKnnu?_Cr4jImq^k2BV{GhKu&Z~&=0^Hl@i&KrN1DwITL!A zn#OLZ5imX!YJ}YgS19va=eCA67G6i^5@V|ku?PI4=fq*jq6a)&sd!mL>rav(@a0nk zXY!7lJ3qsD*? zFZQK1`CMG<0{_&gG2lmx>H@!F)EMwqS|-?ot-Wy5R( z8CVpg^;by{NEV|Yty_~IkRU`sTH9@8Z6I-og0#Mq1c8Ji3ewu!s-z7hR#7ljO+J6cjLe-lnO@2D_YklLEs0c2-0GZ^0WoW2<7)P$H5Fy#DM=XMIWs++U)6# zA6nr74_RPvA_8P&ah$Vs-xnDAQ6@RbHT6mS%(~j>V*bb(0!Sc(`|a#>ll#?07t518 zH~^xn@D4cxBk2-AAPQpu(Lpx*Ce{xd3#gM2=DKW3P!_>x{^QrVu6C}gjf+iPc?fe| zU37iTCW9^G=$a;kX@H|BO1-kZHp@8+j@&yOf3w`1oK)Ln7#>x>*5mL#++J_^eMYY{ za9&^OcO22hQYone;&#lFHxOBGqMTDsTPtm8ZUawKYECEyY`n~p(*_c&S`Mo}PjWzF zSIhapF|u(rdSLhrN%)=K`o~NcuWKC_f9Q2_g#>OsAh@@LAKcBv5o0um6vRcrr2}Tb z54FE5J_7fj>hx5#J~I8)`hZQV)(4VZ!`CdKBfn92Ps_=r0 zyETG0>)&vJPUUm9EnT$xelpvAH=(@mBf4i>V|%wA@Oetb`1+1_=+P6O7mpNfS&ZnR zaTDnQPgbg|o1-sE9f2?2!H(Th$NU#NZ&S6uF^gT`BcC4!7+lRe{^a-V_)G0LPt{9q zfcq&m*gD%sy|up{TV&lvfDbjQ3q0Pa5g@}EY^|H?wJ%fg5$o0iK3}PrFj_YxLEv|d z8UrqR0VUD{zQ(9A;0;Fgfcw6Xj$^1!<)aW*6jn2-~F) zwk?F8vJkd`1T7eCSF~1I2-`s769sAAo&h72}Km7^*#$s8%QXkAgvUb z?2dCjT_>(n`^*@5;pyIL=>hupVEM4}32`5FcHCVpu6J4dE;@3DsQe-Ihqk8cfh6;2 zakIx+Y|5ruNi!s(*K5w+VYwRxGR*K*lG}8A@3c6N0vYm9+<#Tf{l{3OvkQQSXj)%j z>7Cf9(u7|E_!Fg4CbE~v9;FJHb8;d8qFe6qZmiJ$Dph%~oE_?4HD9(=ce;zG{F*#+ zuT^=;g*=Lyt_PBrX%ERo>Ty`HTF730l+~=&h^?QNT$*)v8g(cgVFwYLs z@Ps?EVB#|*A5{*YJ*0SWBFK`2y>)#J92m3o_XUORUur=OOW7Y?Ki7p>pk7fhv`J>T z9Lv2r#=If`gRP^p?Ic-e0V(XPZcy`I*S`lcFOOY%cI(LCDUn9vn<{xgQ?en)@jFdp z__$>~bA!%?BoZwBi_(lec4?TgQ|u$fIbo5UUQF-0O-|Y;;%4=R-ZD3UteW-Q@cjtg zd$s?E=aBA;57d*wsuYWJ}gu~IPJ-yF|=p=fKZB}}~QKiC}r&rPXmm~=M!xu-bQIJ+! zS$T~dc=Ah{f-}3)ExFP)R}L@>T_DDTE8U9L>Ldt!;AzQ~Zbj?O%E~P|VAJRPA|Vf7 zyX14*d_LG*X#?>xp5bYYB|#uwMnPIfn9ptCcw=zwiWVc1dzrxI;PM-B`QSRG!F6nK zZ?M|v02!oA7a$`H!{}I!-e=>T05ZZTNNb~wcLLZPMt*NHAI5lT7~?jK*V-_~f%}@# zF(Cc~@5Zfyj?)sC+mXPgcjMVPA@^>1$-Cv|9buF=)qu^2F1Ltou!t@P5?Of>>*9?x zngdyG1EEC346r$n<=H?Up;5zodFaq$(RigE8xk(OY)1R77U>b-wN8}wH3@UZ;W4lSmhF{@JJ>zS1`13u@QaXa!S85#7k(sp1FHP7xa$bM)n77b8 zHX9qfV^*@kv62mrc^jN$sc$)*0)L#Q1dv$DJ)jn~&e>OXn?TMeK@3c;!#(&m9&$zr zg5YP`oDe}yE-b}C(Er<{X}l=SL*QGDYL942yEbb^fae?4{;|S4j2Z#n%a-2kn)qSr zGTs(8-nmG&zw;tGA}Zxbxt1DLv@O= zwVCxU$9Jr?qlHsNDBSTv;U3|vh7|a|g%qInI$IZdz`s{2%(DDvUGt9JQpfyiAn({N zb+{GeAbninD+zhVPu10P|2 zjRA2v=xSHA#*!cqm!lvpTq(SPfggA+cX@z|P5mk$mV^3^`LQAi0{epH(zCg#}HD z*>5Jz3w1TzRQS9bCXyHVHA&`Z_Kpo9Wu1_h=0AFM_SRynh1_kLf}z%Cxxu`sqMhn$ z1W0BOB`sUjtMElyK?QA=6uE;0yv0PLHO6{RO%AI4@&WJT%P4Sh$`%KI<*_WTVqYk4 zS}1Q-C~pu$=*F>~H9+jmluqREOI-IZ{oZ5#Pc4x{HF9Yz*;NyLyEDK^BK1>zgw;{Gp-s7>4CLq9A z51MLuimZ>pbh17+{YON|?xC|}I<3eMUU_+lHv;1;NmKmwf$#5wp9qYUC0Eu=+(c<% z8O?jGpclIcSy_Q7D*Z$;nqFrs%K5UzUR2TM{2+@D5Lbs`QC=dR!X=-eOc2mCt!sV@+h!vNY9t^1j`tAGq73evjO1lz!07_|y`+LlmgW$1gb-z&bL5K00~VPL&pO3Ur7*1z~sgE3o07KEB2=KRSQ)I zNXX<`2uQqQUs{Ar1cAgW3erl^%Wj|M(Hkdv&6*l7gle2nF_vI-yrRW$vEgRZDla(GFK6un!=5;)w&fxuywPzH|T_?@~EIR6iuL~?hn z=7;O}NXd?j(~k*VBwbz;0HQwTpFomc5d;$3I(-C}G)~DW5s$^w&le2Ki43cl5DEk6 zDxB^Z6pESw`l2o|fdeP)tI>}8a#9Cw)Y1s|>Ey@OD%#1B?EH@-gmlK}e$p1e<-p|J z8k|cO*Z3Izt13vnM)F=GzSm<571d8>#ohURMMk}kOlh`2at9R1 zsuF6pTQLasJ0b{do)EisLZpJjomb5b2@wjfS?q-B=h}TWM-o0@)3u(t#x79=flb$X z1=mJ%*G6;KvZ&2dmso`5`SAMeUl|@dRM{Q^o5S0I&1z{R?czwiN7>R4Qcm>z-vwxLILqFo?vKUTLghA zYir0FDU$Dc^$U?uKF*?RCRceNM%0B? zK``W`3WkxRiZ+ojuY5f0fRBnD@Ug+fOaW?~2FfrLP$fSj*tXMHFhp&mY2|LQ|NnyfPm<$jyo zs;$CfM%-peX&xgrH>v-sjwWkKvx^^0pT~?$HsTlFkH!;bOyKZ1uOLiu7w(PesK)rX)oxrZRv_&`jm zq&KTHrkPPn)8zhw=mpNx`?hmh^4ymOC8^!L_kXEg-I`yVArCr!n&e^+ zP0Oz}Ei8rMfi(C^Ti-|g`X0n6rvr2~3>^6W%GD3OX{3eha9hY&*+XaW&$V;m#s0(f z5UL7KFc4t@@3TGh^+`80#|Po<#hc~6$xO0^-?{Ya3iFxjWk$$@@Rz(#9)uHDJc0M~ z6yAG>-Z`4Su;&X;y=k)qN_c@}98sTc5yh)zngK-d>SK=gCqCr+d5a735IDDPdw#5o zls7Dtw<(l2*Kfo-Yk1hZyOWIGM%erie(xA9(!x2>=6?r>x$sQf+Y1~?VM`>B(}5I7 z5X%$Q@rjb}KdJw*SL^@a7YTgq^EjQuPgli#AEzV0Sf_iK?PD;VtdC9q5fQQv>ztZS zD>8(4`p4-$@cn)8`Cz0h=CbnQCQ1v7ao%esz1S_w`V2%-xCr90h45Ke8)#{2&6dKmjom zbhT~d=jvDGi6HQkI?O0YD;doe{M=}#WVB;O?`K9kK#a?Gg8;D{{ODA)k{{W6misYY z@?+fmz>-`_12Gefju(t(H>623mP8k@8S$NMboWGjk(MkY-X-ORQ6euT17w29JuP4} z30=!P6HEkw%&w66Zbd7VTQ=7=>)^4pOO2+8?}pWmaPL+pUYFJ0E|_?^(cLrgazA## zI*-v^TJRX%!)mR=y0YEQ+MO>c{P#owInB*!t)ExX;G84B$~&*3@qu)11b)q^mA_QD zuWdy-z|)Od3A{k5u-OA|Hn9%y3Zqs6uWj@M-)LeT&XTN-^|1VQ(q}Gw=U;gALMDk| zxWnI(v+ZJAoe2;OkXtHWk(YnI-a0(VU6~Udr_X5QmFTFiT283IjaVDqtqpLfk; zOtx3*lu>mwnPtr`ewOulia`vS5_0tqBt~~-3r+CRxs%lYT31NDkxO%L%We<&MT=nl zNhzBDY->>unDSfrq|_a1a3_rm^$cW}jccVwi~G#y8PWCl^v($esS4^IUZv{_>0h6`}iM58qZ z!jgh&znsAPfG`SNoU#|eUs(lk75hSY(?WTxLV1H2LN|`>tN~(grgS2QU*fuV>GvM< z1}%}d*^F!hF*hiQ_x1uukU4e86*iER2eCZx*h85pk@Yc{PS(e!|A+|Lt9ABFrxlq?Rd!Kp{nQUnP4O2gl5ud7 zULSwKR=$J1@j)`5sQy0ti9=*qqGgR`SST$lyLqpb^ulFXiGe68t;A?Lz*e9Q@?n2@ zFG|~-Ph_zI;%c}%97J==Bi?enJ6u}bv5!(&-O-2Jg$4gM_3qdDSDq0A-=ufIWk(Kt zr&8e_P_Lpjd49}AVe5|iIi{-%yvV54KorJXOtcO=5A~~oA5|&}(z+!H0&y)0`jW}L zYv zh!ccG)&?L!nI=NB3rFsZ6;soh9Vn)f-Mn&gyEes|cDq(4j94C@0SSMcIVHRK7r$W_ zBeEb6X4DDgbrqHkuPAJ@y{5oq$;`z`b=Pm#-NbIcPVUMwpOzg49!O!Hc42iCHITjDqR6bk(%57I5*$6MVdV7H* z_)O!#fjKRRIv~Xn#PTF%x2}&;hd7tHdv}empF#q^n|DwfVjNzfiko*(r+h^d0*v3? zznmiL<1(GBkJogvK92oIPk0}%b8kAW$dHQZ->vmY>hFUuSU*x(p>Y$Xg+)8>wSr#k zE@T}CqNucvqv-**9>wpUqRrVzmNXy%=c}r6%Ol=$|F5dTK1$`>fIi$VoG!Mgx3AK_ z@@N_O8l~d>6IyRfg22C3s(!LW;~^Vjf?{u*eqf+xK8s9Uy5#a6S z$l8q+jRoq6#C@T{dnf8Zg}5tHGyB@bSA3DEVeUW8t?nY#eTe>*_-w3bB!gXV(8pD@ z&m=NaWPZ_%%x|?Zn8GkkWEh1+28i!-S`ugP8ZjX$(hoSri@VX}bvmG1E%xI;Ob^O~ zS1@h}w=wr(JTHlHKgJKLp)cs)1LgEme@$Wi5!l7&rLQSO0)Gx_t>sx$CW{PyF7mQ# zo6nh$Vl*NxKN{NbVy*m~sC~VmA7_7$8UBN{I{Hf@K{?3Srh3qbgOA`c)xN@Num@ae z)F^O`QeiRaRkVIO8MPKJO>*^J)YCgIOtfw_qg~)vjG6%cuHi?wqV|ZxP|L8=E^t_FN4qH6)hZ0IeI6An>Lsg0#Mp1aTPe3eTZ(*ZwCKRUU=^U9#$R zDinVH^p$YERO?!90t1<1vL^*{atwE)wFA{3)c^l>o@CgJu`uaaPMPr{0%W?!$^S{6 z%Z)khPao)_qpY~}V&(0-2+ypmo$G3&mer+PBe`E)bg?Co8UdoK@Hin?G>W3sE4znD zI9Mws#K4=TESu<Iv7ePf0pfO?ciuo`y@~Q1 z<=Wz%HXYi)4=5FW#M$=Bm)aa@1Bq2Fht-cJIUupC<@{h7*_CSaKzYr#zJ9NN^I&n~ zMO-9s^8vxVCEQ5J#1UgOhZMv`LF4uS;beUMT|t5f+<&UmQ`P#&^jGTxHmzD8NO`@W z{`VaHE3be8pJ&u4@KmKjar7!$1Dm3b!c@=IcTrF8psQ=TUTsFZz`ru1;ajz7y)y{{ zZ&pek&`(x0a4wpedHC&;e{J*c@62o)c#D~ppDm`*G@GBE&ekrdkncJ6EB)-OyiuhoSjn+hjL3`w?AI~r&bA`%^CH{z&}U4f z2mFyyqre|46<0;C>yu8`VC(D_%5E`bJ>Tsodg#2(@rW$Uz|BfUd$c}~1cCS768n}N z%Pn+E7P=J{-e?xOz_*(#VQr@Mp(F^rO{scqrqT2-f1x<{uU+!5ZT{V8X4}ATn^{?# zX*A8|YxA$O!0#rrdsa3G%LNwXQR;$t7S8_{sWkbJMG#0f>JuFeRE3JoQ-iiQnh(3A z5HWkbnQa59WI1~Q844#l+6=95qQk(f$90`~(gps`s0rY(PJanH5GTqfx=*T(TTIso zkdB4R&sV85v%henLuAGQ#`P-wPsgM?I}6-}xYLz@$%zg~z-Ar0B^vB2Ea)S^Hz^fX ze*f<(J@%jaSC)C;>;E_k%c}>#Hz*ZfWf(qdQ3dm&KQUq8=}IMGmyPqC4$0y35;NGpwORmyTi`N%A1r`X8W zFrw3J3~TWAB=dF+@N}iZ`;BMHE9$f_N}>$m7#qSGoPDEG^!{6I?_;&U2DqP6h~7ur zX}oLvc)ztJGT+s|@(u`)UNR>x&T5-}%qh9O38WHcJH4h+b*hDYf{;JoLOwyrFH02g za-(`zDBNt(o*>#!vS?2bZ3ZpQ`%wKtLzKoh0;HFO>hdg9^rLfW`P-HFiUJO**VMin%i3T*rk>47+G)eB(;Rca>l^Xe>5k06y+z)iQeNc zimFERuXOafuT$w?Sg+BeRGQv}O&+~&wO(0OBk~~?kzaeir8RVx+y{V3j%Jt1bpBZC z{E5(caRrk)cl!+~iDaYxGYuQLAC`;8P1{>T@?8_|-?>h*5W*j*$o;M0J3yv~6y;T$ z-%b{ENvO#IZLoe6c3VGF{c}~noP#gkJQTh$6lO28JPv$#ChP{*LM@Vo?-E6X;fQm< znMc;o-oEPD{MM$!ht8IiLZG@3Vn&rj#(@n-t}KQ4%4{)snmX`Yi`uA9O`+uJg*}M{ zy*@}{3lD14#&h&<_?49KyJ)o8xrN_V+4SX5fH2tEOOAveTs>)#9K_R zV8~~IVcGo(!9%Qp`NVHoVhBWD-#hp+UIlO&MNg>Ye+*oZSeGKG@=jJp0k8$z2iT-qeH zIk}Lv!o?tp?rG=$v0Twuu4v2$+bfdaHXp9Ms+#o5YSOFoLIu}zk<2MCQZ%(lhyED- zJIHgAhEK?_Ih9>Mdm?tALUE>GXIZ#5K+%KM*u2&;VJ)C}jHO`|h+8VzaH@tVJY0Lq z3<;~olCvrp6cmaN8);ah`LIUwVP$)$lhpdN&HoYLYNaIhKWL$SjqOfGfW)zu^CoN3 z4;`a;Kex5%XRQho=!>jZ7}=RL*7f2Q)}+^0lV05l9VJ7Q%LgFd)dMRi`+?WhLK!`G zWL&=Q?1yl;Zh}tGzfwfNCn*)K9(&oz<8U3oQ}nO=b|~;fqeg(wHfp&d`{NabivcuI zhv8xXe5$D$F&zq?qkoe~_D1~kGLdWtthc0FuT;^(zEDrOtv@r75g@%v_n$tX;y3Bv zd94*&2e%H={IJps13j{JVEvq}18`r|WB1ZyXSarqTzdBK(BdT`#KEtt&;k`&zI8wu z`}1##uGS;6?tVkz&l6?h3i4jvyjOO9&U^LpUfBi0Z`J0V`d20ua9{OMV!I!je$wE) z+w_}nBO3uWRJ97Kyr!1#CtlI*+etg{cBSUa4pJ_ij#%XN`M%-%_QH>nc?J8Rp#H=} z<<(rOcB)=?@y`rHtLe+hk4QCwqFs0_o_X0n# zqJ^1wCyrLKkgZd1Q8n*33#))P8Z`?1C!@yhr4dxADqvqGoLf3I>4mrtoi%=|Z%m!{&`ZgOFlQFYR?25MTZJnEZ)i?y_=GI<=;^4w;noH5<*N#Y(a2Zrc|Vzb zOP9p;-?9_A(5akDfpk5m7W zQ=Rvy<ogOAF(jo>id3g!AV)jVor%ih@;$wac@pQ0iRuTYY?NLG@$NS4F(nV}hd zDV864&QX1-{9v}~5!pakmKAV@n`Q#$BH3tjk!-Z@(3JjzRqRUOtxCxR-01yd=1k+C z7byk_^Y2z$!+cX|1#_c0mQ~Gr_0Tj<{?&Z%0iQ509+FA=WQDIw)L$!n$goL8kp4&aGp$z}Nbj(p zXjinh%|}-o_!gz2AgyFHyKu!QqmVHJF(ca(;JeJV4lwD?Y60E!mTf+e-V!WeQj}fZ zeNoN+o0(q){E<kAiXZJ;f(-UYvg+?Hdnc-EYPT8uyvq)zhd;yQsd`YK-X#XVgA(I%hnq# zT&9&PPtvqA42xFb?mjil#4BVY7s*1Oi)3*|u6y< zW)?cYPh`TrP+6hBV`rf*i^-(v>k`Pcux8_Yl1filuT{XLX!SmN&bqIiyH){{?rc@@ zy2I+|m4{Un?WiiPqU+SZv&_#95I+Z7XYKbObug*QS5ev#=ca(@fYGCv%30 zQJAE;NH$4xkt_~~{LYFZivuF~QjEj*wg2tL6&0T0CTWeSP48d6ZUbyEo>L!kc(t-$VIX^Ao6VU zGK&Ku_)?6+6Se<|+0bB4G7HOrBbl&|!!liEo@J|O57<>I?Ar6EW^486=F}1RvS~ZM zR*zn79lOA5l#(xdTMNE6ai@a+K5-|5-nM1%{0DX1M2`8nDjk$=3$L9EHs9Aj90XlE{G~*#sVAq&-41>*xx6vPGdoj4&w40%BW!HuU@oiz)2c zjWueX$rd)aeu@PSFOD-W#(-OlS_vGnFpdF7jT#3&+o&<%^NbqbqVSDItpt)Sc~nN0 z(h#%fUb7vdPsT^Ij_m_?f+hPIuVWs4g zmy;EZ-zhm;v)z)}ZiQLY%B~C8w2*(=Hy>ELWT9;qS~jpY5DW4ly31m#e;>l*&c;r5#4;RBJ6uih3wt-8Psw4LhZQ!=tYXttE zrko!`p6;Gc_ulfZ?NNtrY8`Q?yk?gL8U-&^E5T{-+m(}hOTc%UGZ*CFc{H7_zc5|x zzf_2$h5M>_uu(;WS72NzY-n$CnnO3Nd@?xz{EAYNfz1^S@?7u=RX3Yg=L~}BHrP5R zdzy;s@0$zR{@IJnmpASf%IP-C*CG4F5_pFGl|>qOno{NU*mtZ~3*WWh((rU=H3XKv z|Bt%sF#rm?X}ikYwi7B&wI`>PqM2UYjx8#CNctik5Y0%(*l1)Ir);sm6J5yu}>Uu5Cp$Z zxmu9M#~zdf!JkvE7Nl|Fen~K*^gkY7ZXS1m?=@-|h>s!S-HO(u_lo-DffC?{_l{H) zq;;$gLj(;zpefj=t=H?P2m-$|O_0{m1I;w>M;d}?n$}AmY=Xc~DHR22{qw#i2s}jN z7X@h%)xu*m1oc%G!8Y(MMsz4Jtw8QUOGi z7ux?ONh0(6oumb=OZvC<*ZV)_DpPVn>tZ(fR3pk<2>0@Q#Rl%HTD?!ur7duOq?j@b z+=7Qj3shO)j?lG_iVIvXKbWXel|WpptJnKhlQ_pzsYx14mYL=aBNI74FroaoY7>57 zBG+qNq(Y96fQ&1YRVNF2WEmOGMb`TlS?^+Gy^oQd*-hJHG~H|$-E3#o5sb96`iN}F z&30BJQMQ+xlt$wqpy_6}=w{d4q%?w&Zbd5<$}o^MH44%q8DiaMLa5UCy4m;&d;EMV zu5bBr4+nI=l}5Iu*N|oI0b&_G~L{`1F|;ojkcGS^IIAuNFEkr zR3FkKulj6`#U$NYul|}6iRdJ7L z90J~JR2w*=$AKZ9;MI1xI0U4wf*6>p5|@D#SP&z#U$ljHeGAOOD_u$UGj;tH3iuW5gsL{=b=+}Dcah5Ix^IF}h!KuM5+N55T_K{JVppaapz4uDv`6*iO zN`FC{KAanv;9xI>pKm8Q8}=|{`eC3%tZN&CkO-ODa+ z9cYb%C+lk%ORHdpweBQcac14oFMjHXU;bvTk4#Wgs_zDO1gJEbc&cThj*ee@R65T@ zE$uV$X|1U5*-7n?7x}zyVqVJe7?4E$k35KETnu7vc!x!{oJ5vPF=ry{r%YtY6H_Cy zJ|B_o6>EHwcdx%-K%N+73b4q*d%xhlXYk%Pc<&v&_wP$u3y13jmK`0Cnvm0?mYy-? zORR08RlRk(w>0?-vK7{NHDZu{`wTRP%Fjj|;VPt&}BkQ9WSs%v8 z`Zz|`2lA(SNWDP!JMv5uaFJ0Xz)W8rvbcE9wz zPqN>6S&#i)%`+wN3wiGuyq^~EeqzA;+Q7fvUSqt2jop>^rLn1(JEX8)7yaT7A%aAEKJ{KGmf6hNJl3li=VE!HL!BqZ3&lp8Ko5AFNC3W9+I$2l!Z{+Q6fY$`&cF zC@igLIzqdZme!~0wzEvNU70vWFCfc)6UepX;%a+}s%V?4At2hN-*uCD?G#>P`q#SX z{=I${E{}XV6KVSj^e4WvNjV3lQPyi*t0$9J>Stkpfyfmq5>f$vopJfa_&?We?=;n` zy!AI{BCEZ~20dxzq_YsXO_@lhrL8-(U70I851iL1E9$JyT+oWVA1{|WUZ@_2D+=x&dSRef#dkcxQ5_3(+lr|^D8@b5s(Z;6)`@d91!)V+g}P;d^f|vZ)hC^xxKJ88 zg-ug^FkCXIw`>YqVAgAvk|7vU+b{&h`Cl@`O7^Wx9Sjj}sPjQ4nuJBZd__DMqEL8_ zGeTh<8-;Ie<+@j)o5;%JXF%fCFE45WV^_FMU}7a*fy5?s#V=wYVnAFCV*bzjy8G)1 zwRnF-*6Xf!(!`c%)Fk0x?8V)=zok%KAiY$tBPTC6U#3PU)W46p7u;JOT0L@^eDC3) zIpvph`3{98n!6rUOB@PyOK8di^nDqi?+x$AbDJ&`pV!ZYtu^)A=PsQy)d$B7m<76R zbgB=Ei<5P?^TT>E)d#mg=Xj$V6mzB74UQGauh6xZV+?0k>V~}2x1MA8fzjf`<3ct) zcg5;_&EI1Elv7~f0Y(i0A8k~2it81HqZ%|V({7~$%%gSNai)3*_#~x@N0VNta2)d& z+MnPRo>IOc6B+V3^_5kg^@V-{h2L9a63Oc~+j9LMFyHOn=4$7fJzn+mSvNMwo zkc;Vs0}MRCMyOmc%0uO%9aTZjkLE+_-$%^PAt0*iYTi#hL|3uIeNQRzhr51-sy{?; z?>tsNMFe=bQblWCsG#};?N21i8TC@Vroxj|+lYP+$d}>5p*ih8RDMy-ubJu#dqBt+ z>Bc=;tEc*9+lBIS$EiNp^W4@8>aDQrbnBj>@A(_$pugS>ynA_`6;nS3#)VG-s6E;J)sWQ<(CP7_#*qE?A3g9L#CW4FIC*R5b&L= z@>0DHZjfUT19Gqya!&h?HEfG&Uh;a8X=Z-PwU17tNJZ_ViL*Yc23DQfp_?Tho@2QY!-E)e&=0>>SZ1sapUu{-9_C2)QK_{IbBa<>bJyK>5cw>%0j6p#Vb4?Qpf z9%Zjd8a&J{oq;^cUdz#7_#)>=U(5O7i<}>QE$4?Xax;(KzKg)6 zvL0X4A?y@Zu-mO@QCp%BNM%JqT4YTGODXGS^GpOw`RZm9O$2)-i}8lXE*UUWC~%Ao z=K|-w3!KkL;JkBz^VtZT|MR&BGWd3YerGChV5YvDT|Y%PpK4#NY6D-VRQ;qYjjK(r zV^E*{sO9>!@%TjcS2T`QQqBc|?{3P`Af9q7hj4yFFO1hlXizai_FMTp(7w{u1|DkE z2=GNpg|L7(BrzZb7{tJopLhq{S+Z-7$#z$70tJ2G4R+6H1UT6c1GBS`q){hJ9ug}- zJOO@DDY3e-qQO2z&)Wt{0y391|8V|bqNHsS3ST!G=_0ei>^YLn^^bu=z^D%35^HFPd z^Kol*^O0+I^Ra7m^U?c}9!s6A3(7pXq$v-xZqnwfr_-!8hw5^n(ySsBp3=mTLQ&ce zSZP02r5Wc#bVe!tCT&tUO8b#3?MM9(9Tv-O_=y)8HO(t+)HThh_qI{j^iplItUgvA zQEWabe8jRgaxTlL@WT zBzor$dXC$iq5$j;=G zoQ(d#zVdvr_X^(o1@AqB_rAe<@8G?Ew{pU$^IG<{{R*jx#^X^w`Hf9^zDMwx86moX zOpb;E?{Q;OFQ68*XViGjpv0$E>M@TFOkzAkH$1e490^c7i?KDjmN)5;E)0{kCQ zi#SaP<1*l57+D|0$oeQo)`u~&K8}&~fqanO0y|Vc<%$dNaYl84#~L*@lcI2W2Tez6 zx6(bzKhSNgxrN_8eOe}xy=L`fJ;z|tDBPfXMkbPJS>%>8*>SM4k+XUSL&{j=YLW)K zIOz;z!K>wbW@{3^nI5~bYhw_29P^ILxDr|KOJuz>k=>mt^FGNg=2^{M8(zVCzu>)R z@ZL9gKL_A_@p}L2HOGgnCcS4h>D{VH?^8{BhicM$!*TWRQw9&y%LymjWrZ&A6r62Ah zR9>u$%6k1Qh+L3~w0%rCPb?>#g=J?`)@%HPu4Mn3j@5dJd`U$@BEVlaE`L<^-*wyF zraF5+<$q)%*#Y?9>m{%+=%)k@XeVI#&$fM8rISde_*Auh-^J2vltpCPaXI;c_e%u4 zPr#UZx=KIiwayH;YiT}D>4kxoUpEcEkP63jh6m`jwLHEu)yF2zz89#-$W$K`XX5EW zRhW;c%|<7vkLjjC)i>1#^GV$j9=8!c<#$|_;0kE614DA^EpnG}s*lC?qnhD{2T+tu z@&JZ%Fb^O|P4a-`d`a(QHS9ofb1QU`@kro&=136%m7`A|L81JRgu*%|gqm6RIA*XR zw@zDNVlA(^0}|vQ1}5O)t}A|7kXqvgbwvzab6>M}+!6=0#}x|w1b!Fx2;pGq+Y-ey!6h>xQ%;jtF}Y+b3ooe9}c2vAEErB znwMvNr|G8pSW`Al^}&obJB0^arJqs^sXp2)KEaAxm?_+u zKm>d8^ac?9vTw+qN~Iez<*<6G;_7~)8%|l3m+G}YM~*=Z$ni@ici zAyW>kmnvrHg*w8WtMXF4=Fr8e?Unk;v*NU`RX#o|er9~?5%Mo|WBr(^Y?D+#y+b#J zV8<;m@3l_x`BUix#gOU@)n|r&dBiPx6drFv1bgAl zJ6GkUdd;C5)Om&+?)=IWqIr{UUKnVJWw=rR_ZbUy+$M>D!Z)2kvc-oQ14tHrJunx^ z9yaqL<*<6GVpx~!mA>cN6~7VSdZofsbt4t6*Cs*WPmIcb`Y~7EEvfIC`YTOW7nsy% z-yqM`w@d2Vrv8nlz74$2s17h$9Q6szEq1W@LQ^zMu&yzx1AMPhSz4c_n@}h#C5o=H zUZX&q?Qd-z@?PGi;y<)*nMmtlvD3-=DFA}nVEj zQIJ;hBTIMgM;kv*Ge1@Vu^iO5D_UopU>iu-<@v6fAkIWVTFH-W-N6q6DmP()O`~IGl=>AxAkhm( z$0}O?VZ#^$;!hN$b(;-i4EU6X$I_J>l#>+=ia7SB^(3p`F(AXOdq+i$|J5xj=W20UIG@OWXs2L;nkVO2M;u(yJi#Q-1d`)9tqqff)OLJo=d?CWT4vvE`a0iJNQOg1$m`Uw8mYeS1DWo0>=fyiKhL>A0 zj03N-VO}~}(MUxzzO$mio5IyfiiR{mF=^4m}s$Ztb=?ebet8|4Qe^B-V(mG)0zXnQx%g@~G%NlboOG5@(EVsy1F z4SwaOn#)8=AN|dg2Jy2U!_IO^qJF|=(kz(Ufo!eG?**{t*z$_9MqpJ2wtNc zNTZzfDNO47c5dcIe+#P+Mk`xf&H$#7Zr#4o;`z-a2xQrnOBy0b>n6+Q8NjqQ1wp?b z!4(!aQFqtIiWVu8Q~(?5M6j1S7HCnod7`3)4k5F|mqv{M_p(yx0v~DA2#`51DFHJ53rDOkPz9Ik=Ro+rjQj>5ZC2yGTU+It`qG7m z_brYu>tX}>LEs1ke5$;qh2cmm{HWy-kIJk)sCC42b3w>kG$Bj|A#e3O9oNPBIj=Pq z?y}It`+-0g9?rQ%yDd5}Dwe_oK18Xof{j$PKAHr9OP0jGQIOUuWkn!ZO6QQgGIOl z#KREbPDSgaCDDRZJrEzGAT8V!L0~h&`LFQhK97}r9y6ccVLp!m@lq~nY$~`s=3TDk za*xNn$FB4u zZr)3ApoRT`P(Vucx3z1?y=&tgYo{D~1M%+u-_3fz>d@lZPUHP6jrUwlpj^Gg?Sq!Y z)5d*&kv=?Ji^l>B;jUq$2GL{3!i)CTKq;ZRX1p9$|p zT@v_#s+#mmc5z;AUJ?zwK|keU8t_`9I>4KZ8vmJk|115Jj=%@nv2Q0TPyc&!;o>s( zP#!MJp&eC0E@_GA-Am#p{p8Zq0<}!H!qup}SI772mvbincajd1^cFOGk&>E!(^8$I zz87|*UZj||pJjT^^d9GW22aXjh@PLTzQNYozF7S*$4?`3L-gQQ7?S=EA(f9?pmPhjvtz=HCJ84E&DkD}0u{NNM_Aq1_o?&x-Yeg^Kke#fsgg20vwfcYydH6+7w2lIt$aU36cq zy3^c+xz?->0pFz50dlAI((r4H;abi_*q0xCMEzEGEqd<}doJc#&&YYfLgc(iF>blZOz5~PuiQL3Yeir9Cx?iQb(_Dvny;;rX`fGI)Ui6HF7c4}=ixeYqof>?P`P2d8 zgG6GJk3_D!P($c`tLjdXfO)rB%_6}83|{n%gcmGC!iy9m@p1L#hvriUh!3UNl&XO4YCHh(Peu-ZHJ5@jB;12jIqgDc6JMC&At#2hkpKsJZ#zeZn zEk>;ZqA;9cbt_s){iPuRTw@Y%p!3XuHO=U0h#Zqy2Uh&A#kGR}AlhBXq^efV}N0T)d+_3#%V8 zhc~t+q$}-P&EYnXhz!V8vmolz`WM>eqxv~texQnrWU3_&B)lVp9s%kKu|h#lFO`K zQ7hN922CF_`)ho+yw^&4{n~o1^t}$(O=MXftfSX6G}48izVWu?1-1hfD(D|l+&*Z) z<_WXB7N$WwWTyvYAr0TaoS;E=QR>IxJqj08<>7#W`LxCR5H77sA6w$d(Smx3U*9TB zRBv`sG&@UKz#4*3z;Ue!F*MkLXD3pZOxsRLJfw z#6q--%#7~LYh66&$b+p2SOb-&N$zDyJZ<2h&|#)w4%T`v6c4C3*4?xP1}pL&MNut_ zZoU>vjHBUt>rBQM){2f;Rx;n*<{en1{faK^g$qY4+XW|%I-s``aa-AmqYl`)6S{q- zNXS0a$G^vJb%QP^%xDq&tR9k{@#tt~PV2(HipCET*JC@eHD#o*q}D_S2X56h6>*>r=r0*kr?qOLPGgD%1qm&Gj2dmd)KxeX=f-O z(ozi9y}|fi@9VbBy;#Jd*6Z0>5C`K>L;NjzzVaUweuFxN zRe1{Yy21SlUZ;|~Mb?29MIvMo(knN}X57SRM!3MrSDLEXu zKD#1IzavyAj0}v=(($&ekBoq-q}CYzL)E#oDBm1$f(u8tCXsbh_=n$`h9tF zd~pT)7uF%WhKU&zg*<`j9&QbR|4CI9bN4>oc#GLv4Q%wgJ-cY_g$h4{G^{zGmJ1>; z$V9R)vlY^`eUr^4ia-(~ES4I{6P8)S-ZaYkudoXb+SKU&1gbJ#e;mq3RMGwF5Hn({fy&f*h zcpdOF^^c3Z?;Tm+_lH`(JM>eoZ~{4M?3YVz{?EHy-vsp*i+4sbc$XR+Y+a@|gkgSZ zu4D&QnC1AS8C4GG@2hNJgQ$D`@P5?ry+VfGxSD&dk@Y$w>oqRY_FZNL{QbJ)KVgBt>LaD#|E1_OLiSxmSpDB)#Bw35S*2sFDk@Xr6 z(?FeM0qX!cA@6Txi(spmQ18d=6N(o{W(kEk(fr$`gi@6ES3t@5AroC`r0* zvhrFZ>vcxfYdl>8wbBCC0otVoc}&Ltc}MEpc|T^KP`v17358i>{_RpiDa!jRpCDi*d`-I}fN3(>& z{I&VF0>~AtedV~lepX(dualbr2laTq&eve`*onpH{>r=8Pkw!l;C+_7?~+S7KlWN9 z>vcxfYn;+}Ru-lCG=vQjxgUb|3-li*Kxgbbw0gsvrjUfJxxsx%B!{F zcavbc=5zK%7w=1R)>5eVSCYMclIary?^EJ^m&D1A)@zNd*BM!_ajgdGDht>ckXz)k z`<>ma^?uAgp?I+_ODN1H^KX|DN>Scl3HAC(oDV*{4}T${vIq5AC3UmjuWr>qy(t9@ z$a4U)U(GH?c}MEpc|T^KP`sd)2Iaf1afZUY+5F3HS$SWYGnAsdzY^;8lepbTsO(q0 zR!OL^UT0)|iQKG#`m_bC1N?zfvR|EDLcJfePbgk|OiiYQ!hF*F+ob~~it_$SsMmiV zBvkgRUaO=|*g`^u^$Gok2I@x^ur`o~0c0OKyM%f_W}i^J_%}-^`%svln19(<&386K z{bM)H+3N$;+~&jNZIC#xAKvSRpGmKzPWGW*Yh=C7$j+ouzF#`Imp!U-AR6;2GGC#&A z=Tfpjda2d9RAGKi=2DViuWLx}$X{q**ck!7*Qk}i_bC;>>_h9TNf7ui(*%1NJ;OS7 zfj1eo3W%9tv|G{odJ+USjplDuWu(uyj&0yy8Z`>UO#HqHtviw+uxYg2Gra4qV+Z(d zqlST)2}V0M(pyz9d^hBL`7R&r&zXYkSnxHviP6Y|AHb%Kw%Pc5vvIN5SX$9eHnOwC z?b>D06JmPhaW~-MMhyc`RVr2vty7Mn?=bMG$3{w?YPkA58b_QE$=I9Ln6mN|aNL~! z$GJzFCtV=6gRX8xYi$w)t}|*3h{f19nTsel+AbMwo6)n(LK}!Bd2kNcG@4h~AFGba zbgcls$Eb1O`;?l~x@=QL<8P82@Y6|dqN4HTBnP}B$!)A?d@soXf0*PpR5X5(({nrjORi)bxOgGE%n zM|bsk10oGUq|rh_8%ttya*)PPCXEEivNtx{d#81=l0 zR#MjiGRi18(=bKq#`CItr0s$_5$vT7X_+U$h99@|?)c&~tj*_Fw9p~@USLC=2zo!5 zy5H6*93F85-)r2F75KqNL=F9|qbl&B#vNUOpJm)J75FUUj;+8~7#0os1t4e?CNfr1IU#GF`q|$) zsRFYWhM3MA_ropj#8uWPAYqlf0*R_f0ST%|0g0(d0ST!{0g0$c0STx`0g0zb0STu_ z0g0wa0STr^0g0uo7C^!(9a(sYu24XND;sGpufRW2PP7C6+o%!X z{#sEY1w7iQ5n#utA>d1m8Uen>s3G86jT!-d(x@Te*Nqwh{-;qxzvkM>W)7fW-r>DU;oAMz)3$D{mf2*IomD|hR>f|K(UgXOv@|~mO z7FnM+udir5+|DaTfKOH`3eq|?2?D#*1Zka-1cASvCP-_cp1(-_04eWi+K=R`I+B}o zB+?Q13#AsymvNWK(aN^DODnh+sr5a>es0@bHbY}mz98nyV=M3njXSOaf5o`tEATzWolt>^ZGY>;3QTbOTTiUOL^teh!Gt&b zo(h=w_P3r=feCQfi-L)4e`|Rb&u>$v=I`NEzv2V*Q_2H;5b#WMmt&ZsVM*r*ZU%Z%y*Uu)C|kb;y$XCM(QteUr~g0GsQ zAs`EfJb(gZ4UsdYUu9=XJ5j~?g{bl~0@dHQ<+R(<1vB;P_Ud)H_6qap-HJc1pYpB$ zyJev}VsQoYMcpFq-+eBPFHaYjA9jP;p2V*_52u!?bERWC9RA4h(OBb6?B2YhhwJb^rCAJDS>+kF>c`r~0KD&ip!R=C z3L}uNGJx&b{CSMZBP8E*_o|Ook&mk@a?%LALn+z&Z$Ee7$YopQH%!(JF2?D5QxD)H z^o+&zfPR8*78^i3b3MG5&rmPBDF8s+kpSF%E)5)#OamXM#%nnmm=`&3@?15w$xLR) zhiB-f6D$%#4E51A)FB`?7s%IBq}al7mbdu|RY(R37o8FLh}x9lqy-`M%ibJFzcuD- zwr=JkYpMtZT2<@LXvT8v?$`s9`)yiiZ7c{klr0 z&<_C_guJBzxQC5r2$*zd-(){orJtL60aLGRJtjvVPQ8GsSN5j2JyrU|)XNX`X5I9+ zy+xj}meLM*QzdcQm)_zYijox*P&NW?K;A4y$24Xqh z3#Nq`IXnQ~WqrHA~vgM93;G~=g0`V}I?3llAHh()nyorLelF@9r&y9|ijEw18zc9MgbP{elc*Wn%X2lTk#95h3 zc=%%*-w+T3{jEz{-~pQuLwRbmL+{*$-&K^oC-Uylxs}g&eEMx}mh)QJXI;vaFcL%h z(-_q< z3T-sV8!J^q`AmxaSCyOqva3i=z`R?9K-+IK(d=3^U4H{d=(-zLKWGleUo)WnNprXjBqF_j_~6aj&z5_N1PGO$&wJD9CDM ztYl%VU||dk*V%~WO@cIXt?s@)xo3lEHz`lQ2>E4*+2LCje^Yi-$@SZqj{f24*eNEqp5{y zP%|?2yW}?t=2j3C5n581BYIRe@|753)#_`l1hQ7hYVd(gg(czxoAM=Mr8jwrntZE% z%KG~@g`Z5+rxb2a)HfCWEKy!}KAaT{hfvA2R^ZjDsPw}Hxu!K}y3y=sx61NftLXJ} z>$S>z{RG`a>g5_7y$-k4d$sT7ZPmX;fC~DDyoVnypv}~;g=r8E$pnzqRWiL}f=08L zg3_5hN4yuv z3;M$Z@f4Cs_ue>jEte<=-{aBQDM}MIJFIX6rrimF@Cwu&Gf!CYf zDUa$iw3O2p7_2njHV=hiiSbT;xWF54zUtkZZ*KDrOnVKycVVUZd}j0W#l3YUZgPevTqs zUG$#Z2bI9?=l$w+y(rN*YdL-o13zO!S^YVMUodJ7u(31SF;Uo=?U;y))Bis1%_{T5 zZf@JH3S{ux)8IF6qVbkesl1KTp>@tgt zaVb}QhRP#UP+iEIeDxV$o~#NaL&Bn_kxgN5)`Hqets;ssLIco>J`8{lnP_XT>1yyh)_ZG z-a0cI(@4H!PCE8f1rlOmQPW70u!STEOOo!es~$naFY?q0>xcW^KZwjN`Y9(!z)u@B z0=!+RMfEh)UkZTyvB_trNmxNoNHm!4VJ?D+g+w|}{x8(dug%^Fu+cAneY+PbJU?`| z=3oz-i`k9H|IS3Rr;rZL^L$;(^9$5%$+NInYUEkiLY{>!X$2&MM7n<4Cz{yP5tHmLFuOrQ|{%;IlM;G@un;-FxOiSR|0QP>HxVx zv|;)yT#j5OXY~hZ8TbL_zRMiqY$*A@s}sv-)@vk#J|OTjxhdJ&JXv|Ik@Y$w>owAs zGabone&&)#N0La-6rI^-_BKt@XS2-0eAzO4Ch(svvlq!X<_1sNUmoON7GKV@>zUMdr(j=v4YY_eD~2tjZ@}b~jI6Bl++#g-;Gf@^h>dsZ3Z8;Jo@lh{Y)8zIZC(4of+VBlv*NB z>x8QpySav*rRtLRyPH|Bkqr7Qz_;9=53*!S_Ksd_WWCPFI)yFzzJ7V!BG>=Ch&Nhi zmVPNVv(4jC!C;d`vW}9@f^iH*yk`D8J^Ci1$-~&)EyZD}b z+1)&OjpW0}6y8VGyD71kW!Gzstk)S?uaUmru-L5eBHm~{Iq8>TGuu3URxN!&KU1Dy zzGlA6E>HNLeA(SRd5z@5#}t01kV)+20LN>Mtk)S?uaUk3c3)++7b)h6eknGy&C@-a zg?sfgmk`EtK_?bc`v6r2l*BV)`GqPSIeIIYJncdEgeknGy z&C?^)(j)aVobUmtk4bm}BKfkr zdGZ>`hmR?|kE(Z5VlVqTuQjq>XJoxb`Zk`H@FL!5J)h{8Vl&%3y-l<5t}IV5H>NxR zKb8s4?4=~@bJ{I=xw{$k8p)K8HvCL(N>*eq=(R@H>x`_|NZycer6`#hJrcFp5y2MFH-7}@@k41PT9G0$ZgtLbRksS!<|JJ zAIAZ`p^qwww(Q-!)B(AatgT+^q+Ckoua|mmE+w_#rIzb=KK@WY<-R)b&y89MyiTe3 zy%AcUOoG5qPZR8A^i=EE1y)9_0%9f@?N+pIO@hFt(foDy%eBLZb!-E#HEI-ynfTQO zTDK%YVAE*+t&io};f2<*1N<|ihJly~MmsjrcdKCdnn8HoJ?#&fg6yl`m*^%&BeNFR zw9z&j?=u_mb@#NBjkZtX?b_wr=Eex{-<67GLhGEkzPyZHPA8^$mY$Afp>iq;Fw=rEAs$YVo5#uEF|!n?xliaeZcT*GqE)iysC znjdXo+o&!Of8+r(AkIW}v|eg`+dzUZ!vYfd*q7F;Q(qt>je;}hp;H=c#|De<@?|I> zjt2{!iWZKFAP_I*D?Vdj%0pWdgQUAjuKVbTXw`_Wo z$UIZ~JwzU9J^J9at)=lbSzDjGZSG;=?Qm>o&~Kj}j3|B70Y@#H^U}fiG|$p)b1!Wj zFfC*asdJLl(q1yVuIcE5z2Ept9PwL|)bOZeEtksu&SiF8Q|>qZlDiMYyt@i2yMI7$ zsiymNA@vq*jAE7w^HKO}yHTs`6E@36fVV0Yru#@%rrUMvk5fnB-8=1g`O~8hcj#x~ zJGk3*)Avne1bCNHWvAGBZ|VsA#V&U2mOAFgIIU;I(JazWS@nT?Dis%GTF*{`z-pSH z&(MLo6=|^moL$G0Rl&m6y1Co>jy*Um9^S}_zo`*id7&yOJXU>$Zu-CaIj6O5Ugp3S z-S|4wGX$hxzg#c%f7hw_r>t8Sc&}0+{CVDP{7pN4Njv_FSsVi1X;c@u!>A!3e)YG` zpX-OydS+zy(N8&W10JkYEPPs*B|+fZj2Z=g-KY`Z>CcLmme4; zE|9cEL0XqtWV%4&7zJsi5O!l;8vkoVh?}erK!O%@wJn6Nu@JU_1T6~EdbkCl4J1BM zkk-4CAdpZ*L0S*9z_fvcA_~%4Z-Hq82}Km7l>(EU%;nQ{eNg0Q_jhN44I!}VDA4aiF%_yP-Bm>^saoe=7sP}0B=+( zWn!gQ^b1wQoRc#YAiCwpj3z3yKU-BEAp4`E!tE1RJy@Ru1xqlW(Jd4VZIT(zta7i8F|PBv{uMr|Ed?vM?Q|*++^6 zf3fU4bh;nA?0k7K@HA;3<=3e<^p?U0vLx1p@B6W^Uab8W*4e2GyMjYO=?s@yg`UD0 zKz?YkvaL%2tB6c)Rt=6Pg!+U^v49BW-N&`>7Tkx!ept3x^-ojP&Edk~)E;%Zjy^aB z1HW!4^n}CF={>#A+jSB>$Mz> zr>I=N_);dWGBoMR7BwN0aAW|U}&!$4vcjE-4RtT*pEz&|o-42X-dFRhQ5UjEz_Y81!_gLmDE)}A(uE|3w*!~rrwxy%n7)Q5;f3V4cA zb6OiVRy3MH$**b$+{1Fv0VXH1ojj4CCgjH+fK4aH%!xlYC&qwSme<__u`KF<$%$1nM`FAb ziE(q{By(aMnAGQwnd$sSr<_kQw&P6EFz^*hh4-#8oNqKZGRwG1?`K<}@>mQvNW(jk zyjgf`pRE1r_p^l#5I7az&(@L`QG3^{i}x>e@nQE~?mRBky&UTHRWx3&4$I{VAd7~q z?;HTDr9y4dN^&fRD2h9|x|qu3L9pwow~9Ol}nTOQihSI&26pS+*i@MMA<>9jrbG^%xjVgAVCe+c8Th@ zk{B`VPon@5#$*PDXv%2?keyU8117?A!Y+{5u#*ac;7942UJJ5>U#Vp2qBIqAy+z`( zR=n&E!?42}&%#+Pe&Q26Zaqk?!PCZ@LHVv}v;cFi>p`20ryhRx;5|NuUw-6-#{1da zhOby?ZkvsbBlV=IWP@!b8*KA7ZqVpcRC1$o02GyqYYVO zMQad-N!h)ywN=*Zux;isj1zermqirF7~~XZ$7MdjBG*ehdF^kVkVQCfz4{SZv3KPL z{Ng~bCTZYdIKczk;a}*Dw`qkCk_IzT9-KX>qCw1SJ2Z%CEjg3hECG0qWAOOI5&=&e zPtpPa!xBLc+PIf`%;cjS*Aiv26c3Yenh1*^kig0TFp#1AUL9f|rn{o&r(F`z{FWbZ z+u&owV2E~DL?{3zS7~AK{pup|9gCIYdp4IHub{cDm*;!hI#D{eK_h1`y-s9i(k^$0 z=T?|{fSR1sx_E9yg9ysETro-j3txP_Z+c09mu2ayNk1NMPlT42dgebZOHURM>3LE` z;~5tGHt=){&iRuSjTD?V!C7d>ds?!|<^5Czy~N~_lCw*))s&w27q0Guk2GJ0fXAAz zYx}AvIje>1zwcEM5?jY2`5jR~=&_lj2;HO_gQMV|C@0^92i~L9oYrLDF6sJ~=^6rl z*L1BNsc7tN+0f&YG%DQs2kjSJ1>>KbOs5VJq46Q6xea`@QgTmuvZ6su3y*4`YJct5 zFJ}{Ax(l}ve5jes27^elmfainBJ*WGyHI`!GK>2~I%X=mP%ybjX3mS$D~%84l{!aQ zJmsj7{U3*ag6xF~y$+TLbC9Rd<>v9>uoe)zV^-;??QvDV+( zIN+W9k*1&P0fnhU_b2F#k!!WU)23a5rS+mD2s~7O%ttP@0M{uMj^0KpT0hac5JBL* zO39U;9h(M@UR*&uSdaaq23iaBojjre_%fwT1H|a_VmzV-T8I2$WJLq;7^O@D#H*7m z@NblgN@@K=5(I8NyJ?-)waS_)h!0K^_leB2)GcMHTal%AS(duMuP7x2cH;z%Une=> zdHRmVT8_pwN`{Np3g_x{t>tKRloUB2B~i=KpdvyTyH+0`wnFLxDTlf@4XPpbru9~> z7_kkc6zbkIsD#*?*70Y>px2@_PE%4;0zcA}qw%XG2c%$XeKe?-Xqpx!Q;X7|VxlN5 z%B2>i(X5^Pwm@Dx?NaTut#(GOCfdL^T3S~Fn;C2uGPs%ylAbV-c14Q}$t?sRL63s8 zh^GhwiCYw;Mc_mb*bGemWuZJUol;;r7MQnLU^+nNk?hZagEmzTOC)DaS|&R`tpf2My&tLqBsU5z+w?dG(}e)!t8*K+G9bN zDB}#Pv{2LlORrFVy%h?}tdc_c^-d_iPh6}yVa|%*K++ZpV$7y5nG-=EnfvV}@jaH; zl|Ygiy`@Do#akdLj)Js^u?PZ5d=#Wb?nMwt?#pFBUMPpCjuw+j)B&6GEC1!HJYl0G zYNgGuQ6MRjJCUc6rjAaXz%iYE*x>x+335K3fg^NFbpZ<;OCK=|9FcpEPN2XMGJ+F0 zACkb$T&}~6e%wYs3~Y>kd13U+3!`6N82$3X=$98pzq~N|<%Q8NFN}V9Vf4%W==1&M zn)m5)Ms>?|m{T+#&(_ZaNdlR|wpHrdyfR?ImCZ$}Q0*>jXIjxC_3WMv^$^r6KMSTvs{zX>* zOfz!pRbG;so`Wv-K=N|!AxTX=E-pVgjqH^6k))RTTs-~B>F=v_ic}|$RCbMy2|Qp+ z)!`NRY03@Ag@^>LRrx!zu_U? z+^b{ED`HZ5n3{9NkLf7)V?xX0%+nvHM&fT(a{n~RGFfC;`uA(fPMf+u8t+o=^pu(f zvU=7v>-#Z(-k|*#)=`Yl`NE-~^w9I2+7Fq+g&2TbFZhyAb!vDetF4m9QAaB2^k&fEHUMu0W)wdz!}CmfFMQKyZbKI9x7 zHI90FiTv)4L2(if>eF&|XdDOEB!lKixF-Fva7{AHIU24>e>_~1GsmOH^*B7_BHXTD zk7GDog!|2}$Fa1C21d?xx%=c29ml3bFdW2=RJ6E+vWwT_$on3)mmURD3OtKNtRJDL zQ1Cu^+jVZzU-ss7rI3Uo?v#;GzKDb(kd%;6e!xP_Ueu)bqb9v4HR*k+ z$(g*7%R2b-Lc5H-8c2vl)%J;kd)p^)@9*@ga4kvWr%H+uAaz;GF$^vY*CdTKcJX&L zkige+h2d_Wm}$7u^Tr8k;e}RaD}gN0GDiqYBL+8?VsPUGRdcl-%!!ekC;sfY(U*VH zPdOI^60=&4Mk6pcP7s*JfNz|j!7)nR-Ot0CEQIF93FhPtdP%&N>=iHKAVRlH6e4kp zMWSB|zLuP?jc+!|J2e805#C}WoYE}U-SNBO;w@Q9zmv`AHQqtlSNmcW%12r#A7`QP zfp7`sgOq#lzdMA=7vg=gLm<-sWk+vqAd`4*>%uMjRkWzv`C)f-kZhnd*e6GETFmY6 zOx~t}gRa;wyH93{C@6N#Q4uYcfTRe9sxw6pOoyn5N@OP;Sz4@#Nj(gULX@Y)5@AMt zc5H($_hf+yz||+~(rY@989Dj~k{0pqihX({Nj4OcBy5i)ZQiFxlB7!^Ny7F>k|-)C zNxBr0By1r`!uCj#s4pZ*SGq4YxOM zD!9FQQ^D=cn+k5{Kauncb(f--vru3ou3JhDZYepqrR3n2l7m}H4sOYQ7qVZEbMQtS zBv>3i7tWlKW=UNrzW_QhAcq6;M6}P`0ea~8WIKe+o?W_!gDn*aS6liN?`hl-75HJs z9a(`nO%K-s$Slif;K(7%U*MQjEMukX2#2UGRvbC{15!jd z2+A;Opv8DZ1H)l~5emr4VoD*Hbs`HS8qp5%ptSFk{WJkgL6KjJ%~B+93l)}8CAR{L zTcjz<`Z_DvxvOYL*n%Bl3wDI{b{?w*@qhGF9^V0SYBX2A&9bDTMY${pU(o5VAGm{} zVD{8UadfkCaLwq#?(B8Sd)|>>p@NWXMs|PI zxkqQyLcL8P2e`mJjA{e-H7dJf;T45_AetVi-NJ=pKSLg++a6=8+rSh`d4Ij{=YL>bb>l^#Gm~F?vL;v9?8)7S*5pBq}U& zQJxkR8_$gH=L2{`TrT!B%;qkZ0mUi6U~oIKex^k>r8c61b$MAgQ>|fLSc2TJE^NWN zum$VF_OkxUv^xgAL#ensrbXo~4CkFa_Q!k0bNfbwb$GRM*zJwLP;Gq`}%#a@7pW1l-0C#8po%t(|A>rxI#_Uq)%#1dc)OD7jG<2 zU%WBB`}KT1cYhb}KFEAJlC(o9Z@@vC9c;zNCAh z!jgWzuGf@DVM*VZiDXOqr}8CTE~hcE+>-uVxp3yEj4a0Dk)JH-0v6+^Hthly^~qeZ z4G&)O)`X4u6ib!0g>_565DUwthcl0%FXU26)hH9S=P4fr;9h7TY#@%JAHnMwrI{U?RtQT*4S8D64 znyeq|r#!q5{E1N=;4h5Q=U~!cyrQsVqUq<_t+Zr5P`{CKgne11<5MTgc(N1GzFDt! z-YZ-E_RP4rkk`|DPf=1ZMU21EJC+sB3UXPsC&pqq&+3@P^F4{N-tDM;XV3_BW_!Yo-Lbh z)q9tXySZC7aqE_QnYASMF4@7aDJ`2V0gzw zVZ=Ko3M1YzQ82t?qF{JOcGtFG`1X?F+e?OTFB!hQWcczMwTaH8?yqxN2$;SJvxmzZ#pMUa6b*3cW6rH^eTk>aOtH@~%)* zYoaR+tx)86Flepy!4Q>ZCtaYZ_SA{coT*P1q%AP#>6V5%DCSA(0{va9NTY ztY|zY$pMc}atkUNCnP!GDM@Z&MdRs74%kg{iz*u9Ne*~+lG~%A@uDOLT$kkbtY}=C z0(`h&pTD0$G7XZ&;k5NXVjSGOS(6E^A<;9~3T4>*xKNw$b_4 z!Z|#U1@;bGh#LoWca(Mtca(Mtca(Mtcg%RCAyp2gNjAAMzw)^ zTkg-lq@wZAB-e6(sG4sgLqK{<){?3NlbqAkLKUva-U5gN^5&dNs$TuDD**&zt#-lp zTcfVL#9SH!QuGqg+iVJxfSM%qM7butg?Fmve>MwQDn759zHB1d^Dl^?MRp5wvmpFL z`?HUch2oBhfeQ-n14MFhn$X^FL{KI>!kI;ekX~ep+rYh*s`E!9IgmvKRY_#TJAeo# zLba$BdKIgM>Bi2saA=T=WW%R7r!?{|*}O0jYRqguid>|psrTD~!VvJoO4a2}1D6T| z!;vJCU9mv~6QTu3ui|9-$%;6%Uq3f~#tT+lW|@A$YD(Qk;q+^X%&5p-tA{-w)X&KJ zFcz%exTz{w7q(zsSeD#|bzuwEh4t1Ss6Y07rd|~4YpviAg1xU$oEQ@!p}d1aVUbf= zp}e_6)!e=MDGzJ`e`(Yx9-M3^ki)qL!?`(qp;0pWWYxLVbhpvHpXqMrin2+KqF-8= zR-)(&rf6mE-MwHRdidbE|W6t8;U!b8{@W7AMT@c9@={ey1?4 z!SBzS-)nNe*W`Y$$^F(P(%fzprp{f(sUvJ*>IhqyI>KVQF>!@Wb|d<=R`-QE=)u-h z|0VrXeN%lfOse^^WkuLPFdn4ANL^rl_)JtL3|(wPH#Z?@Q)P^+9dcO%INnw%)aA;K*g+3wuis5-GO+ zsY=bOrTTl>xk$z2wbnp1Jg7BIb^M`?UV45vxguIvuP3WdRZaR#)uc~UP5M05q#yQ_ zh7H59l)UB0def2hwj=9}M|Kw0{A&*S;!4Ot0UPi{1ETzxX(`W9Y^eAM) zmG&;<7Z&A_GjB?$nR-b$ZZGc7ge`DgG6V`r?+Q;#rt8K%$+`MZ8aXbpr`d$(Q;&>FKHDtqR13QfbwJTa{k|6LW(*$W9 z@}8nAN9mIQhwk?hFL=SDp86M*6(mLdWqkTI;Y zj8-rGD3emlHT9_HnRT_%#rh%V)IjnU+|S=U&bX6bqv)bz;f2RYQ&18uAiZO>&_Sh$ zAdoVNf_{uxK%K-Z*JabXsVkqYxvqAut6lOdk5{g16kRy6lX#`k4ClSF`%moASW4wF z+2>w6uhn7+Ab0RMaF$L-cpqD@H@x1#>kORNS9%Quy4VRwo`AR=65PoQMAn;#taliV z*Xt^5?} z0CBog7v@#2>2`+AHfY$=Q;(B72l@#a3PDI^xVlL)VC*W&7X@6o#O zSJyAt))#*W)(>a;RzE7FDWF-`5*T}c{ejbatg*T<2%<}zY z;q%>u>SgT!&E7(JqKQa(*?sf`@7Z>uH?mUU*@*&fG-~vf3g4e7;65*kR^;Id;KP*S z!ByIiHGZx9OzYB$_WP5H&nx_*QLBJUUTlU=RoK?gIjzh3DjF+Ht_^%alIyQ%ye!E9 zHzc_^6^$#B9Pr8{H@Bj3O_Bq?Ey>NRXuK!M0Y94L1}Yk#N^-!@C%O3*jXRPY@Y_jl zu%hvUBnP}F$t|d8{36K#2VN3GFL7R2(O9IUOeo+(liZ?;#^NLgJS55OQPDUo$pKGD za(h-Zo|5E%&q#86RWwdda=~ zZhl4MACnyL|0TJ>ipIYsIpB|z+=7b6jwA=1cW$HlXzZn=JW~gJc#>OG(O8z`fJY>` zJt`U}COP1#Np8=I#!!+2u1a!yRWx3ZYEZhuGvr2BCSRU4yu6KhCv59)KeY45OU>tOwZ5OYzG)>bR>HV}WJ;LNUc zO0IOwm5rKtIc5TK7!xdXDq62ig21LL`PqLSm9dg5W9AA6MslqJc!~x}9(lNVk_J@~ z{2Qxik?2~K25F9>w1{pkN`pm2tOJ|l%deW{;~OuHZ`{Urr$ulai1Faxcts2Q@(c&C z>B@L^ALlgHjD0!Er0|I3ZX9@Svua%dX%ghc%UX($PTeR)e0Ye*(v2Cp7mnx z$2=h4>e?iS%M6@_2%a{p(0q}XrDUlc;jot21_+8tzTnUrsJr-)q%91Hd|}HGZiStn z$d}@1aIz*}o#?Mt5 zC$d8Xrn8zc&T2SonbW$YucE;zije&k4bDe|oKw-@fr$IzSZG-{WVEK@vtVng;i?dk8SnJzF>H{ZrWlZ*%doP&>}|_`F9ZC zuKhV48IUg^UnXm^zd#nAzV&X>7umwM*APLy97G|_^|rzeDDO3df=}6a+rYn5s{ZOA zjifz$L>E;_Br91&Fd^q7KzbFws5pzUajeH`Ij6NL{}v)bjn&l$KNrcqbcV>ax^PQ= zfK^({Wm;XP7ZmDDbli+#BcBx~2BR22YDRs=x2lI|&rH=WD)_1bt{Kg$;>1u9qmchw8 z&N3Z&$5|G@K3X2frpf`n+^AvTYo`?xtzWLC>av?_x%w{Z=@kkyuP-_tYEHC)uP|y9 zh$lf;yP}1K!i{~@cz{Ln91q#2(3F_!zW zy5z^0`SEn~V>J-V!H=YcPas0r2oFHgw<5Q(`-(x2Hs%Q5b(1`t@xJ0Um7(8e5ft6D-Kk6bfSRA8#M&{ zL!(vzFHow#wY~-3lEi?2)(``~&BTV7%W0Nm3kz{(Y7EF_s*Q3rg*Jf-4~vY+im=qC z9%~srXtTD2M_a~%DhO7XdPBJL4S}1<1`Lj4BQ4AkdGDp-a2Vx60nsWe0T2TsHIwde z*J+{=8LE7~@Uq)kT0!7J7*4z10)r!{qK!AE3Ig}&Fq4lmGHg-KLIFv{fMoc5Ig6r= z8v}BiC-Ai2VKqJuWJb(sT{@?tk*3G^541~~3**cMN?Z0dKq^~KGjE=xL2Zj1kkYQ@ zXrwwH_u-N;`$$cwV^jJ$PhK21SN>^}AuN=)7lz;H4%_^^Zf;{x%XgIAsR2sL4r@Phb#Zut(Zzb9e=2uya}q z?yhK%n$Q_cjtZx5tQ?5Q^OZ-xb)FwUtA*U@nzXPgtd)ADP2I}DB9IXx%JA+q73bF~ zXp?g}r3O;Us2~~#X*A#&JUN8-`GNQ2V|2J&NQO|}x=`M*P~N6c-dwK;{kDgVA2TH% zxkh-rtM_Gi{CcsU+-0aAjlHOTH1>k@Xe>*}Ra#Q}${Kq7vP_K4*QnS-8)9Ly+mo;e zFwp&{Tb?HCBd{}BUpYIIrKDK1dXFB7VVw;-(~8Wf$}X0-{MBD6yiAs?G;UbAB#4PW%f?n+FC1yYrm12gbM``W` z0WI0`(?s4g)1 zm7Un-ezo!I1~az`*z~Ji@T;BswF&Y^kwSw zoYv&Vibm7x{2{UZ)dqQ&@0PZ3_c0cOtu1``N`w23mf&~wGxYU;=qD8S@oI6qx=*Nr z3%E+!N^a4Qz)P}#sERp286g%yAOW5hZH&fT`DkDk8?ydkfR2SF(f4W&@>z|1fN@Or zjlc;N<1~_XvYK7v!zyu^F+@*Q8zKly>eu*CbX7Wu40)0NS4eIznHjz8)v=KZFR~lK zqiY&n7JytNJ6vYcu(y-zjX>sTsIzXxASfmg#4^*raIF&c?5m~pfKBxy1@*(Z`po;c zxur!OVaxp-L(%EBhZtkekX+BcAfD^)vFFMyuEZicB01&;VmU;0R{Nyfj~<5rxgV?W zgFU*O`2d@K%xdS3AC$*^Tq&rhJnr{OS((Sn@^*q%%1Yq1MhyY~+Nc#@QusrohJf?! zKy=043gv6e^`D$l0uM222smui3gB5vg+m7LB}okUnuZwo4JI~3>QnJ&2SQ|neRjAq zAAUZ^#|)lAqsrlFGxfqPmQGOQeW1htXYb7eB(18u|LUHZW`=!H)?ph46%l3F5m^En zNnGfvu4=kxMU4>`5ExdOEx4iJG7cKz_MoC7ZkV_bjS4PN6n#;n#@9qKf@0i=D@F~! zbDw)Y=RSSzt*L4j#rK!~qw9OmbM|xYefBE9Kzc`{>%0$*m1xS+KMBOpNNW`^#ho33 zdvo01AP+QuydlzFR6$@h3-?yKa=2VdRlH3@s+fqFV2qb$0s?L$FgR74TMX%ISu%kX zX-rCeZn%7{*(8<13jAA3^fZuJQQv`KkfzA=H&i6ehiT>m?H!JMv`y*m`pyai`dj#b zw0F(NAocpRPoBi!^Y$Ed>=ON)CqD@#w>B6O!j^J!z#SMrx0~AtikaQq2JJA7QcW>D zQprsbXe7qpfgv}i7i4`t)SOQM^(WGBcLdDBCTHosfPsx*7pe6?0~2h02ZjNq35;NB zRJanr`T>jFfmwQO;SLOZCu$ADauV*qFs5(i`U{WyD#*r8 znhod#PYt2_`at&uB05|y6hlbfx{$nKA$glZ^5*(w=now>ehex;e2w&YSMSU8`1NAf zR(((9sKxaiTIqhfW5N>hLT#xB%Qu1)*w}ojvhC7fn{g)r1_IrErsWyDJ_DQM^_{ah zURsJZtAF%x9ILZob6Vl~bLCw~WW_rwa50|--GT8v-`{6H6@INHVW!LZE@83o3Td#^ zQlxNM4*h^AD&0-+_Tycac6cSBLM8IeZG$~p<;CRyX|U8pE;Zq$=q5Tv5`>EsqL8y( zYAI6e>m_GE6qRy@rk!mSUR%E}#dwj@WlaF$YIux4q&Bxa?k&SQIQms+!|6h%{V{F0 zYr=sd$9<(QCGm1{qKOzgjeB$MiTZSzF1htlyDW5g1!g6d7fe{R@m@$iRbKm+tu=Nt zhbXuh*xcqX3O2WS8+)sl>9WOrnxI_$FqbWmI@m5}b%}R&8AK7$=~oA#jNJG6MxlFF zHKgu~{`!q3U`JhM3kM^Aue0@&`YGR5kl$1KkYg(7xl>uzoEKX}t`Y+;Gtva`)k+Fi zX?hi-tuI72Tl8}EUDRK0>L-CHjF;9Jz1JAKz+^EyRn0B7v53N;zHQ#GJ|FMfz#H_Q ze`I8ojJExB%8hnPMmuKoQx;PP_yr?P1Ak|v4)ApIb{dF(!P`#7=q`;xP6~l3fX?6m zrb+=!SpcL;P8@+)4t`9TAIWHThWX2^_x>nG|HMkQN}GcE$Q}g9Jd#U4z-G^Pt$NHO zkp-qYWm77*c%QBjc^llP>d(*e^#|y(+~}WJm*oNc0hZC+=>6T=xyAdr_c}~SmVJ5n z709wLOE!>YU$}t1vZZla1Y4u(to!31NxG!qbT8-YQ-%TWwIFsFnB z#2hyO82^LKv1uUFufE#PAkD?;U#du&c+*Tg=8<$fkZDw3#b>}o5)Cyw1q7E|h2(7t$(!q!p+9uk_%W#X@HNuoUA-^UKgoO+<)SNLv7`%Wu+%c7a9MtV4~U}DHE(afc`pE!$TznQ_GpzC zFI>I$*5%oqTq?U{MmNzZk|1295QUuOQp=FyC_!=tL{TYcXgWfBDLl9sF8(tGMH@~RGR?soNAI&fsp4>54yu4G$oJ;5n)fT*tKdxHYkFNG z(7-J6a-5EN4rPVpbLF)+zeF7?KUFKuZFBT`?OgqAZu2%irt%N8%;MMQ)jvrOAI{A zdcwaqnaKUHHfZ{h+93I{GWX-2cSi1qO^c=uvpC3H-K^#({SmY3YM?KXtHv%B%)H z(MU_X3ZIimz_W}r4t$l7mIALV)&2@U28t zi}Hty;ql9dd^V9FrR?CKGsc;u2@38Z;a5`cf`hi)x+jCT7xK>+n}vcIW@)bl7c#B^4kmBpfR=m3j;nyYR)Hs+7W3uG6A!8$(lv+ z2s*G4?6cInKm!wOeH(!RB?^pSY9u!Lxdw~eM!4|(h1&>(aGBOH+-D585g0S8OGcC{Hg@-&Xgtz#RzE9!ixqWsw&rJ)TrE{%+zTjZ?S$z21$St$r z-Kur-KETEN9dvWSPpJMr`+4>l^@kNlc2~HH(!w4rms*MxF3SV;KopfOlzUC_212*~ z8YomE-`qAhj^$pwaKYbOms8eUD!al$KhY_Ypf#FCqzgIArIsS~DwUR;0Z~-S8JbqA z7+=`LnEjzFEI?d^HMczOEyL*c1(q%qJ32UDRyI^3p`6n@p>eq)rk@K!x@ae)p83xbgfarF?oOPtniGQ zj$das#({5AlKi^F+KR!~65sz-h%dv1iX-bxQ5W6sNu+Nn{15$z3SN?Nkps>!`JwyXqfBM-i4^5S^?x4VAIJwf0eMw$Tbt)wucdljP<>!Xgs)%9F`7xhTT)q&BK zX0!`@ospIUKi=@8TQT~&F?NBA*W%i8;A@T41%B2@%Yk1tQWto@I@1L_#YkP?w2_tr zR~e}be6Ep}1J5&37kG)0mIL2tq%JU}e|dDU@n17S&gFa_NW$ZB2_teYj6kv*85w=W za@_`!;K<16m6pghkQ_%wMqf#cK++r;8Ifu65lE&ZBcs2zT(yBDJ2EmlCouxacVuKl z%EdI0>_^5%>Lbhw&?yz5V{?hD%T+-jDGwGp*6iQ30(5|+JTfwRqvf>&B;}Ej5q&PE zfuuY#G9v532qfi^kr7!JMj$DVjEvS;0XjfZ9vK;tb z2Gm2es|Z~#;M0Cbn|G7I<1CET*IhHh=r0o^@RS*hjGmPki45-wHx6>w{&$AdYfLD4 zr^-t{yS%7d3~5I)XRZscKz=`PK(BslXW0#l|1|hObf$uzmFxW zD=g93iB!&#T|1yAEO&(!i2WbSpRfQ~`{XN5WhI=}g{`r`d8g5ZpzP3)VprLBuQ^y( zJJ;3c5NNQjF1k2&m8t+)whI@U@&G4Ll%%pMkPdbnl42lxjhJF|&@zQ`p$kFDCsOP+ zg#<)VTupsia$Rk7al|Ocgute*{FfbbU0rl>;4di#qN|YNTeRUh(vus5KmM@X802_@ z5|?xv%?W>EXN z!cnwbYrX551(kP4$!}D<>?V{vNU2=0@Ct?Gl@+dQkf?@tH(WDP@-C3Q!amoHf23Kb zhyGBh_I78NQ!l-BYxVE0S-`Jm5ju@AbE|KVnOl9M%-rg0izQl7vIoTYz3`6Jt2A5j@H6*0VLegVO_`V(w!zRCyS>S6H-5GOV#S;UIRHz=irU{DVP6%pEA+} z@Uu2!!{t9lPuUPXC`^-FeHZmehZVhRx_)9tyTD(Y(eMrajJ7hPUEo1VlCMl!Q!&7~ zXeK(E{^$O+Oa8UZzirKI8+fppJ%4V+plLQg0{E(igbHaHAPVNRX045c$-x7VY)Rwa z=eZ+@`6(qpk}n5kK;jBd>764FB{3#Zh2u925X`{1=9(v6;1i6r9N05b4~P?zrYo-* zX}Q0hRbWpO90wv;n1%nYHKqiGWdW9f0Xm0T2qxXxf%Lt|JE9~^jy!=ROqK~CDJbXe zZ!~8wNx1`#{Y?x@mJQ&JMj8k1WTY&_H>%Eg#xf4vLP?`?SJ3~xMaAyY&(e)P6@P?j z?g5`{qzT|DN(xI%{>$+PI9;Q1s}f}go3ftwa$Gn4YhnR@TS?I#qdhK(Dx@^PbCpzf zEVs}tS?E?+_#bnn3*5pig!P5dzUE37c!HAZwV6TFzx=1yoUfj_!qWgpqNbk*}5iz@K>MtPLWbXwe6n1os39YcB!vMhyjO#(>Nf)@%NXvnz z8>tP%iSlB7fC?OJg5y903p=_WYmG^IVMhnczySR!h+xv4?FH^d-Vr5XvZDi%FbM`o z3d*^AJ~?|n%kcP!!cjhD<@bNv>$asP)&t)90yIqk|5Hix!|ybY%|C5n1!di(#@x@u zLNn0?9%H0&AlBsxHsBA;kySv+zXI4)-!7aZAz*|FV}3n)v`9eQda=nsQLQ& zv9+3(wd~ebg(DqxBvSDO!;y;RU#jIu{k@Jb9;t(@i;MF!<2BLSu8H1sP4t#)Vn4$( z(!YAhT;GetVq#_rX_)&gh2Bh^18-&a**e44>TzHrfw#2^oq1cUkgeN%IaqV_9?<~>JJjD5$(>sd=)Wv1O>zlcxps>88c#X{knfVoH z-n~HU=N&=InY3qlS(c)V`2H86W{| zDa?}cvB2I>F}$TfF^=9sptowEztvAUG$w@)u|v(}-Z4l8)7+32B2u)&?PYTW71Hw_0to zJixMfTm2NjSQSeG_8B`*5&&V7SYD8MJy+GqRb|Vrml~62*u!hPh#sh3Kxd3gvG0P!pwPnC?}oP|(ELc#9O;bF zTsLJ9@QUA;<*bKJWD9d~gYK%*$wdeFLETi(MJ7E5fyD`mve&4jO&|!UmWwIgePl0F z+3-7L>m+}SU!`=J4?s?gq2C1c5E-X5CDz4dcmYw4!ZO`{AKDS6949^W)_&YVpn z&<@jWJL$IbT4RSSK5xG4#Ds~B$0^eSWt!fU?Kw|SmXq{TQU^R)N#*O^2tM8fyTGPk z;Xk$0|6O%{Po!QA3ARp6l-z3f0vRk${hl*vAFsctTnO@>^mmrG@h zppVbUH+RH8xrqX-lq5Ggwm07tz@rOKOT>T;F)!7E=T*8z#DK3;(iXC6%Kmf7LNELZ zB_J>+OJF!YXN-bvP*A_SgEyw_L7MknE&t=B>1l}sJX=Y9z4Een_aBb2=!@aIaBZy6k^80;`7@(FQmwM+*`&NX4nlY-o_PVU&!+_4Uh zCEZK=>E@_f9&QJ|OCy{suj=nzF~Ur|6vrr8U^1Ga#hQYnEjufK#~En?*fG-NCe6(2 z^;6!t23~8V>T71 zLJ)`*E;N2a4SzQ|V0l$=YsQkL@RaCFYbIHr zMq;OK9(mnizIo+gUZNd0OY`zW>R)oI^C-2P#PT`#;Y@5ChY>5Z>rI;FuUNJ_7L zTCG03zl5~R3#N{?4d%#Z^Y5da) zcb*qrmjnOZ6ux5%Bl!T){Y6vM1|l5}1=|&)T}@XTxL8S%kx??*?q`%Jdbm`ssl`_&fR{ldXrS;J5?_yrP{)Y$@!2k@SYm_ z!z~3_y!&P>nU?dlC*NSy$!7lZRrnc%l@)h?%qz)!k z`6}w=WEH*Ii4|7S6P4v;OF?!j{J4xITSeQ-Ldb;^|C}W&8Z6Hp3gt|*FbzB>WA@F-oMB=VCTY%+P12kt%L6RyYzk$0fCXQQd3c7(KgZ_XIXX$9 zoMjf40WZjyeI9TN+l4&jELk3MmMjmjyvw}I@&F6I6!Y+Em49s(8k9Ghg=N6EX3Rbh zxP|RP9&(l}4>?Pg2UxytUS@fK1z(DJ_@v5zHVX~P&1PX5cuU6Y^KhNcoI7n&^?(cJ zN9K43h|xl&%3(9`(+eV_Ts#53W&5TqqbKcXjKEJTDH>t)xt)v=_H3&eN14hwwwcF}~~HNAcv1EL9@bSp*#B+@{lsiheZkMIGTq2>3l^3d9) z(ApN--WFOLc%+i#Fzk61gR8V`TeLRbxpwS`g%yORKlzITxj&tfKOOT2d$LahHZ9~| zGn8AHDp{B+SeU}X(=6hRcdTX5G@D-l|7bQ9{baDF(JhBMz-1b1=?i(kl4)9wrqx!7>`*h8T7}fn zwqmZz*N;^!70hhw3`LxCR5bhI;j4gzHt&RP zACw6>#`F0X5+;wdMRel{Pu4XqEk1U1$dQz+V!%~;z$nW6;gfK&60!^m*T3nC$U-5c zLB6L(@gbxwL^A6ZBFu}ojBdzkcGp@4tGA9sIp_nPXQU30T-2gI3pH_2gw0;`F*djP zC}-8DMEb52>4tR--fCe@1K)0>4zLmFhINHVH)J=X{K6Zfxt}Kkk>ajgV)D)k$rlBw zIhvnv=?FZgwD8T+>stF(`|9{bdZH0{y@}GLC!dZjTAM~!Y#P}#Le-br!M$#Pu%@v= zo(SAD0^#M24f43*rV$7nGAs}q8{fY!Y=VTVW*!2^6b%_j%w>~-z~l}aYR@2WU}6#w znEMSYz8F4HWAejd;fXb$C+I#+;TLoQzt0lSnB_^56veClK1rC!b7UeuM~L_o;g!!2 z{CR@R#GnMpg%E$39!_-c>yxdcbWiaNJATeCi=3{TUf7@IMY`!N{aLQoO&{sc^7p#w zvpNaqwl?0du4436rOLC98+`ix$)L!^8_KSdDuk4BmX{*#i%O1|20)vj=!Lr5=(Y-`Id~O?IUjKzW71BGm_SqUt|IKjk_Y zuxq5Hz*ChJ-WIq>s>^t(v1YfGF@TCtT`=OIx?ln>+?hI4tt6JLm(I>uveG?7vvh)$ zF4M9^HyvXvZ6H!o-CMKj-lo-^XO{~%ZdhM2+Rly%yTBdQU8#*Q7OE>$EV4L8-q)&Y zjhYK@q2@yMHQ!phXLtRSTQR`BjnoD%Qc{@F&y$)neu%MVw_q@UicoVf;-ThXb|G>W z?R`B)tsH08IzXhP6#4t|o~ba0zpg5oOa+$zP?jH=^>H9lVtvDU@AdDs5%x@taMk*X z(Zf?C02_@Uj6}x%Q%Ww>LL&(E+Po2jT4)5JdN*%YN4}LD0)E#>%YffgQoPQvuwwiJ zW6dTK22c?i0gQNP1TbkRl;Uo+GHOfBI1ni*#dTRJJX4{^I-0^&Hr>a89H|sld!~YF z4xTwMmP*RL_+~wY;6OZ53so0hU$c4JM9M<_lvDxtGEy7(2qSd{DGIeg(*Y_LY6E_v zscL&wN9v};l0CG2OvaKm%Ne@&yCCfaft*iB8E(l|UC&f_tE*53*>B?R${9OjY&D7_ zyr=65(3p}!^@%ul+eq1{pR%$7HyNo1e1nm)71S#V#X-~GsaPlu`17VJyQz1bZu&=K z=>d_FwBDJe^&gZUrb1e8RhA!{>MrnZB^6bBrh@8!)6=)c#M!XkDt5AJkNZMwHR>q5 zg*pnAI$o;vj2hmaS66s_b??$1-q*s-W&$h^R+dmV@Lon+Qh^Di@I5AnDEr}NZ7C2b zNp8O9^Gt>3A9mJ;KP1gGAM3suOV(I-<|Vo+mFRPtEJ?6XvD7G$PzxmzYN13z^$xC6 z{a5Q}p@(3(QdvSVz;80zk_wE=g<^b2**|R7mI0BHV&r=t&s2E3eUSWYc5KY_alM9AxF_V@tDh|Jz7-!gI9`=L zRzKx=WPSVu%9G4&_Hqqo@vxCVoV{<-jseo^wL|wIL+@7(^h!Bor{%Rqs@EB*UL&$x z_LZi;D|;Qm=c}#=l4NT5+Woaw=Gj^8pqy*9TM4{SN&D0{eB`3n@w>})uf_HIo%~5HVraB^}KfIUN!VVdZmIJ)>mF@qNO(ErEjTr zekt4Mt1c{(WNP?oU!jG&GOIR}H(9kOfLAMN@0Q%uJNv#VHn#u>p;1wvy~ZY=LZ2Oq z=e0xks-X|kD+QFDq}Lj$UT36wjmUo0(%bh|0P9IzRpiLi@TI;%EqzKq(__j|zF@v& zr!PLSjT+pgQh3zZpT-kY1^VEXrPMqNO&}?^bb(+?Mk! zKA5^l$dRYvE4Qs$+EG7K<)G|gzRad_^d25H%Hvbf*fc0uD&e(5_o|@}(ks=tVO{05 zMyl5tsa_+pRAMv7JTvATa^z|F%6(tw#t*YO2j!RdXwK1lc+@D5Peo&s&xdbQD&e(5 z_o|@}(ks=FU8C0;sa|KKdX32bh4p!M4)24hi-a6`8oqKzs-Z&ZDOu&9 zJlA}gP34FHj~eCisc3BS`RG?Uubm2c)zAm&m1@Xp>9t0x*BPl^BeHL_Y_0MvKA5^l z$dRYvD|e|{x;(2Kl-HRr*_|)%OQQxvfJco@J{65kgMy_JUORNJ8u}o;QVlsm@meF* z>x@*d5!p{!wuU=GL5@5PU%B_IrR%fGLAlXD34D?W0TK^Zw)HpwL|x+ zp%2m<=aJVMsa|KKdX2~)Wluc~H|LO>edQjimJZa?lc2_vNYT3-6mV zK)RI}8U^*qYi#lv>{n~Aotk>p&VftTTAjD34D?W0TKEzsh;-RLHA_K1i=rL-veb zYovOek?J)fJKt_BOmZWDyCaQPu(&lc>={1V$>#9&WkH?H>I>zm_o%*HIL4z!5q-iM zn|#*#)z@pMj$SqNL3*WTvZi{ik?M6us@I6@_bpq)t*OY7r{ODisam=`s~nWqm@l)L znOp?MqegjrDjJ)7KKfP8Yo|h9HS|Gxr5duPdaaS_bw;Y!h%66l47;WxN1leS+()!7 zH)NHAa+6hV3iu->?bW(i9?Bd3HpfPNo@SGNE|AB&&Tj2~R%`d{R_lSv@)rG+e94(y z;SnRD`Yaxs%gTz0>`(NG8__6KFWbvyWi}$q8xtZcR4;o*E-U@uxmpj3loRw*o_+*Q zD=EI9%jmg@5x8#Ws6+1C0(aRnlA8 za3GEc!EVK9t%cDA;=AlxfUSkmgxvO#UrE&g(L{xe2uP%XL{m#MARgfZHbcu_7j9*t zwM(J3Ewpniv^MaiN~&KMX7Jkw$Kd2W%6ky*ljpR6O$+(! z!nuX1l7*>)g()n2&?4S=-C71sv-#`7+dRM$o%=IE_ddf)JOL!|((A%r6R!n(b`Og&|<9|LeluWc>!PuOhoL{6KBZ%z)yF6W7(R3qzoqXlbklHQXz!=TdJ-o(#gw zdz8EMx^S*(Ihy|2Dv`Y|ykBO26;iL#sV8?k^7UgCOU3VXZ4|Bt?-Qfz>1W|K$uXI^ zm6*HL%&pALt;F2crh8>>Ze_He{(E!vOTncAhnIbEg!QOqW%{GIDy$gfhATB!*Uz#jC5x13{&^LtpI??A1V()|;_IcYv z!X()Z`=x7j9pdmx*uJdCj;nTwON;yv*GR?Shlw~^F?iap#yO{Aa7iMruUI!(#l%cn zw<7W}q=xaKq%A~5zJ_ikV*I5YC$9v))FNCrQZd-lf>}3OF?d!Y&Z!u@IT76z(t3G6 znfm-u2E|qmSsOf9=om4c+vs{w%-NO*`1p z^0-gIJl%AJUJ8iUff${nRJr&7q>mz_SGP#7hdj!zi*$j98fhAMgpzo%nDNoZ+U;im z6=C{<5f3lXfLEBREc7Soro@t6IvvkgvbRGn(^Hza+Nn*Z!YE-vy@$Q4F|C9PDm)3xv zHPTYx=as~Z#f-mbtUX^{44@)Z7mRqQE|`D|FBboYT1hNfFWsK8WTpF%=I%>Yx=hO_ zbkoho(gq?W)y-clezaEiIjOpG&6CkZsk*=ysk>4!VJuWvs91b(s;*G?*HNS9LM_x> zsJtS;dvtnwa^Gc^^I_XwqHj-WuE{%)kw>LOO?cn#f(=NYqy^PRD?zVBOV$7 zOd1NMSglshH*4cSq@)!2t6iR{&|}}#>fV{E4CE}WsM<3XRCCnJ5v^46hIQ4$^l+LZ z=twP8U3h&;|5|0Aub=X&8t{ci>Hx1WQuZpER}^Z4rdO(1s15jIrYc)f-=>=q%cOVv z+KeUJQ+!R2_Bra8U6I%F9cB5ceoCJJkt$spEmT2vg?Rf3RVGysDta2V5$cb0`Dx5Z zp?deewR@z@H}|p#VEMhWhOC3GZsS)O8sGL};^ zmh6z}N`Tnr zG!Uucitd>T-+uoNEd!}3w8956mT8|3RwP!L#-kX1H_+HLNDDWayb9=pkZWn9@`b~OuQO7;#zQnq zCs?*RKyFZs%9juNKcB(6aNdvMtBDs!X4QmplKD5AYSNJRPc^-Es^&8feNYmmLbB)c zS|io#j8w02LbLQ#%T@=-eV0-BU0nZHtfu#4_-f+CR8~zW&o%$DgBx@*dalK~ga?4f+$ektGMe0w@CSjk!x^Ui);j4)kmzWom zKzb@IzR8P-yf2Mv62kpcO|PA*`OHHf9t0x*BPl^s>x5zLiw5bH=F$?%gg;!O|PA* z`OHK2xi3^x_M2X-)J>>fXQXgKa{~E*$u**2VO`4PRkA8&}WLFIGdDG@rBk6yBFcg=y0Jr@~%4 zmGsqs?(5+lm#WDg(`$`XuQO7;#tSt|FS2a4fjqA#d(7eHtM_C0YU0J3teQ|>V*X{f zQhZVx)ubWspK5yTRLy4|y3f6LTq-2{Mz1wez0OGW8n4tWz1_0a2J%Fr>@nASG4DuS zIPb^s)x?X}XVrxA4)bp|`_eb7)csRUubry-%tIfPM5&PMOTE@e^*STfYy7yz`(?{k z2lxvm$-Z>BYI;A0uO?o6Qcb2ECzO9S|JuOMD@nfKLEeLrr}~=T=eVT)KF8VbeQDwQ zr~Y0$b@sJ_?yKb|kuUo?VJA9wu+1~0|>Juo(na|k~+284-P`l_; zJ(3XF754UtJ~tj+6snKoeYvbGyvSapIz*#Tz3ff7?1uEz)H9N-P`&JTa@nhLuB%kX zd-YSEW&yt6)c<^>m#M!+w|+Yb0&m-_;GMd4hkaZ1pWI>x?x-YrIOZ(yPKkCp_`!*G z8hDpPn*#5aXyf2L676L0!bCd(e2~(_J>Z-7iyjB}B1-@BsMz}9W+v#V?2hX#G_$goel>fY_XAH`wR58v3 z8hFK`=#D&~0erfW=Cm#vtr%>hoh5ubDtt&H9jNeN{S*tQfuEe%#uZ+iNUu>yf(j3c zP@^YUT01}@o6|aXUd14(%8reDq}ZZv&r_JB+P*+YebO8l-(RJUH$~&X$0}(X`NgO! z=AP3!C43S0{uSh>B#CDz{CpyPQ{fL2i98-*NgF4Rf0anUGnC}4Lq2cj=8M!hs_=%S zuwN2roYB@{75D~AOotp{P@ZN1VtY>OoQoLU({0U)jo z>+5~%*mW8QpK{Cuyg@hZCO=oWV@%fZJ3lCbgB~{0+Igo6mM%;3)a;wVR zrk`RNc)OCq&N4q*yHmIByEv*8LEsW4ML|YiNQ}Tm`$yS1t@GDZ3|1;}ShL-d*=~hd z)XGs8uxX*29fszCwM!P-X5lmotPRA%oYtEB0690%!N4O;)e0bSgmiSwX4e=yKx`Ji z2g+NV(>iZ%g-r~E)4bk(bIk-rs+k#(LRl_>*hSuVSZ@66k)rYTej{d_M9NoDj8qJ; zB-6H^ksUNI^IA*RwifrTEo@uVY)ckat5t53<#il*rIO^^dDd49cGD&;y!mjZ@}Hel z0Y71+Wk6z@FP^LqzW|c?j6PfpyHM~NQ`iQ+KuLAxUZMlsmM6i1*EaclI`eY(e7U!l zMSxJWjB~m{djXBJAU>o zTgX?iZ7I9kEXNyE$Gc5e5BPZ{mDgi0Sg#gdwBOS3PG>bF7TM35C)vxcx9Fzd7)uYh z?L%TK%Lz>jyqnVM_XikkaX?}O@2Iqzk-@@8Bu4Nfl~yw{c`sjEr8RRM7~0ibfV08T~}5!U%la!J8knRXZrwY9?>3c=Ky=#p* zM}Nrf9C){8q~>G5WQbN6(ML5a13F1m0%@a~j{%J&e83%TT8WDcm|oE|BO0z&$$*v< zl|cHf=3_v|2_KMltN9qv3Bu>wAaAt)Et2HufnB5pa$o<7(R%;KTxCkiw*|4uml{^) zLU>8hcWj_NTD$iZy5I%c_Z8D-f%eQtMGJIUpl#H(kB$qpets}fr7MBBR(G%WttR3e zQ>7*{V6qIFH$*0UJ}@EqxM~wVFyZSL7p0IRBp`8xw(4X_k5nSVxk&Z?MXGl(QoWCn zI@nFyV>I1t7u{@U-4Tqmv;K(GlAGe7lDuKi?C9=KTmX??|BeZU{90haWG~eP!SQlWjRO zyhb;_&ki!$z~9+kR?cr3P#_T{s+|vvw(GMw29Hg|(Tc$nl_iR1r;FERvm9W-S3|OCq&-X;(te=*k{BpymHMUxA_&}hz^TP(2REx41 zn6}WrWH@bsLTbzm1QL`)ppY^XA#dc!HjpVYkaYP(m`J}FLi7tX9fvN=60#);k!`d+ z&DY%KlhWMg)6y{K6Vnv)scDG$@%Zel0tEn?7deqo?op~K*>wA{i(N-eJo1$alB9+X0@Bw>V-HMrZ{RcaU7*+5S6i>hFTwXT}2pkMsd5x@Km?Ff~ir&Qlox+pQp zluRYwWRyXb zD#e_MR6k`Rl{zssBGuO;QhhnRPh+j0?Qr;5D?4I=?)`%9J%jFjgYLb9?)^Jdd*PMV zE^Qz^A*V$xJ!8t3SldL)lTg`X;xp|pBp@@Pq1$`h*woMI2U4Mr{ZmXNPLs&`xi>#y zYNGGUnn;1EW=-@RSrdJ6YoaeJJ3|^!`UUvF?M0j>g>f11IgC`F#7Om7j8vb-NcDM) zRG-L>UU7J)e#+N?0-tT99`HOPO$|~MuFa$AY!xeArnp46z0g$mfUFP2Z=mr^azEL- z^cz*lN+Q=@J)J^;o83<9KK8ed~FjWWTd}pZ#4; zXG-7~{N6L@ep*2H69c;M4g6zaxyNgacd)U$^1d`S^>c?3*6X5QJZhp(R890w)I^`6 zn&^G1iQXHI;vXQ%!5xwlyVGYUQhj>vul_z)uU5Rou3B_}ml>%IyuwJ?BIOl@r4>yZ zRjjnMzDc)z!c^OpiC60>ob)u1Ystmc_G(q}K2tRgM7zjcwLb41A^P6gQy1Nb>u2G- zWB-gLdur34_zqKY4oV}h_v;qDZF#5NzRXGi%WcXMN&)_{(ejJ&zt?RG^t7|^(9MW$ zN-QhAgWpsZPC5&T`&P!1X?dUS&}K7NHlH}JQC8HU&Roz6zmJzo9iLH;!xe?mO`{Ay zujDNQdBvuYbLMOsfkt;M&}~aMjX)yq`SShoA|+c`C%&LnN?V|Oh5Cui*{xa#5*JE? zP}sC-1PYf7>MiTS7ARlWDkVcuC~f$vdNl-zv(jXUmFy&)l}Q~GGTcz-lT0>Ai~Ncb z`5;6g;W^F-3G3Jp(qP>y&`oA#CIHFXsQhpr7`sAS2PRh{3M4l{6u-!UumN#3u=ziq zhkL)C(2Ms+X1(ruCrxh2MolCg#9rK;`&$a-1=4f%J96@T;g=dZq5pl(J>z!r(CQ(J z<-Ldf=3Frne&}G3K;e*nXPsMgOALj+B{k&$y(|HGY3Sa)hpFvH>F1WMRhve~H;r5* zD-$$sz%0;hlbc2$adEO1J6rN4+n_yDRW`&RF;|*m(8xmEw!T(LG(XQ(LLyB~mHmRn zgq)tB8S%pHTT(>KvJV3is?5$TY+>`^1~wn2(AMJwv4|)Tzk=wXaAwmt1Vt+9Iffq? zElxZxWFt8ntIyE-to3*YJD1r#=W*aGl~lCmnF^}^PUXo&xuTw{-&A;mYWskGj>*e#;n1A%CzZZc zOTq4YejA7{vL9OGy0C>9 zJmo@puHw#xgl}Gz=jwNG10O+*$-!EvIpZH`+P13cQrGk3s0ZS&bmP{oiMr;oY*2rz zTj%T6O(Rg|=%iT?ww%2-lnoM5)dj7$_*CNAcT+)ucPJ?qlo9@v3yL~?lY#>AMGBgI zhcIIBlnd&)iUqw>z23Yk&(&`d!`IRrQ&7hHC_NT*Wh^Mf{jD&aO{20+>J8~I-Lz#( zHk~aezOoG(p4TU&ag(na=3y;-Z9Obl9BHitA|;i`-he<1o^mBTSFsXHG{()V@?8C< z7JSE9=6ah(C*rCJjeGqIzziC&EmKTYOG20H-8uFiZ9Xz1LFA%-#BU{g(Rh zdDzwR)HzQ*1sYFYa{M?{%bl$0OBy9;t(!w^ucq&bN!sx3g{wM%r1wMQX|U zcGh!|x8!`gZ{Ojm99iT|HsPBrSm>JPG*+ZCqPIjNkj{#XjHsG0mP*#m=9w^->ebCA znlSb&7V(BhED4w?6llJs0?m6DXucwW=A8>PUyVTXf2;u|-`3fhbwLYERpjjYjk@{m z`YCr$fIn7J{iG{{yNs^`e9A+kKQ$kNmnY(A#o#qcl#?pp)_TEPghwj|oyXaW zY4Wr%13IP_W^j(Z($xmO)JQ$xO-c&s1b;KJ0cpU%2B!VQJ0P7H*nF|;v{3E$>xo!k z1b@>`&U(OaH`u`JETm}k$p*XI)CL}+yAQFtreeT8MLhQHRCva~cauHgO1slOwgCcl zOHa#EK8lmJKq0H93<9}LB2dV-iTIQ!5%Ny*OoSAYnk3>gWFmuRUE@JlpQ*+spR4Ah zvp!qRZ9ZQOl|ExlF`u)Bn9o{M%;&8k<}=q6^SNt?`RqMe7m`ou=e*WfnnQL+zM$v##k6X*Fl$&~y*I;3%w>?2pQkNSKKi)D9SYiWp@ z;hh$BO(yE+TGTc95*@Ov9w?6}HlGw;vj6@1p0Xyv9@tF2fyw_gC2J1;wE8NKVIu~v2v2HpDw-FpVz`v%>62i^O(Sli^O)-F9D zJ<)hP$``+}DX(`ApP3P|CkHY)8VbC}jZOWWejpY4k$;Mb#Ay;)Ut91KrY8Entcetu zYSu*Gku}j5wa|{-Z!VS7> zGnP!tvvp$9LbBrkQnPvoL&;dASKvO*VlwUOGucN;1D3L~#5tP^OuR>ex|#K&|S7c&~C z|Cj3(Wg!8%$XvV|d!wrQx_Q|FqFv;!TJMWhSX4fzi^|vav%vD@j3s+imh;4N!C6=~ zH)VYb?X4$7$LnX|$_OlnDoZE?_%TMyAC>(}-PSeL+4oaEDPzeFz#p!;Uy^D7RsN^jJc{)Bl z3ytdxJLtBvczk8k2%9+jUZ5w*DoW|9fKf6Ds>z{gm@P;J+wI z4!*8j&){o`5BQrVpRW@7kq?J8;D1xW((&bg>9$`ad%%Xh>(+aFo~dyDdW!DTSP%;b zHm7DR*?}_`$hqcSo_?OI-@%9H3%gEv5I$dDJY!+GShv|jF3#n$0zEi2m+TIDJzaQ7rG96C=XqScCJ?k zKJWTb>fbTuX9tL?`m|8AQv zQ#NfHf%3toPlo-Q%Wm%d9Hr@bNC2iOpZ@Ng&1F_*0!qYlIrJ4g&=;}lD)Kka!x7)=~mqok{xRYq5G>e}3IWkC70MnnMGT&DT0K{>LZ8tHH?PWb^_xRqQ*F2F zC(nvA{;ksEv*OF*Q;!fwbrD(+o+XuSQVK|0=%$eDxCP2~CKR7P6(LB3R9C1zGxW4ya^WU$r=tszf>rD{1h>G$`$fl#R_p_ee)VtYg|bqj=vSs_5{R>-t&QV8FAq~APcUN{%W1ml zt;W&=zQ;(Dzz--X_5!1C8e z^#O@FGBTR740M5HATkb4LYpMK+rpgyl8KA`B#y$jIp5Edx^qb;?D? z5k~qMY|mA^9ouT#j8d+$Tj6=Grc1e+F63(3=PI~2UFF%DCR^kpGBP3^azP46E+QkN zu4QH#SSd+H7f%$6IlmMWyy4l4!Ss1zU3ncd^r7T^)S$==Z>jz z-el_>)qADsKez(_R%vowhywv*adHIiOU8=t0ti18}H|wkKDi!C%z&2`skr`Ow-H5Vy!lE0gj7W3g zJ@(CY&qu6#rh#8jQavXa;F1(}m@d0Rb^OZIcYxb!obrRnz7<+uQ}#wrd!C;;&GN%q z=g1FhdF}E;S{KT%XXd|Nb(hMgfZE+JEtE8}C^(samO^m?L8EF#uFEeBr{mdA9z{N+z1ju1B@Xw6Y19prw z2|Ue6J>bblnl7K!Jj`w>%x)pfE@2Wy$Vazg^aG2j3naGsxRL?UMQKI{TNGU&G0TAv zkeH(^Bf^&BN+21CjDwTVE+wIDNx0f#YXfgG(gcu91fy-s%k9S41`@O!&jAm#_}f5Y zC>-_WRqT`k=vV+RHYYm3n~k&_*bJa!0VF@NqrTjasgfU4<_9e#M`XaJ(J3=Zw+JI} z;o-54;n)jI`a;U5Dn_)62m|SrAPoMF<$MY_PiIcl$7sG%gD`lf8Jvt#uCk-PJXh1D zTum2pHC@WpbRk#MecE?;G=Rory6;at(Q6da#(#@9XY$%6%z-G>Sg@yeOeVBNfuDUO@)%V~E{5_?~TKElJ z?^uLOXT*L7BBwa)J`m&Fm9>x^h_L##obIN7<%|^gcI~El@^G4*&N6(b&4V_OLdbi= z>kG;5uqW6`BdyPI^IbvbL=B(iKZFmyRJYYOd=yuyPw&^yc&^C^_vBpjzm)+`BN~)@l~g)O{kU#xw3$!e zx1DA^gY??_o}_+POsy$`YF$& z0AJiV9_>|(-kKPJKQz(`;E!g=GWyKXF)4*h9^)#2aXIq>Vma)uy5`5b&1e_+J|j&7 zu^452Nphp@lF_yqeS=wO1F;mok^tPant%1j1uoP%t+jK_!pn7IO=N)HvDU`Z$K(gr zeH>n+?2Z2jar{9_MyQ)KPjYMbmNgY45+$c;z*LLC*e`QeCUv*2sTiR{b_u|SI$`Xm zj^+^WHmt1}p+m|8Y^W2)e(IQLA=tGQBXr1=1vbFc2oua zoY9W1!2fQvM_1rGjdn~0o}TY(QS+Hn>5D5D);fgfwM6SCTn&z;PV zgDWsOlr{vCLYZ?wGALX?5-40i@+VCLBz?jKBzwXIBzeLGBzM9EBz3|CBy++AByqw8 zByYk6ByGY4JXSL(F#<{IR;`UUti59H@r%!qpG@?fzqMv{-$r)9hbv8V0{_BDUEm5M zb%5s@sSCW!NIl@2jnoDHqmg>Rn~l^3e%nYr;LnWI1sS7;NNIf9=lk5UX zrDVb|Ss1_Nn+>c$ zFf4(=JLoNK8SXNOsl`hkl3ClSwPdEbz~n6&6D9+bxB534*Gu$sUTZ2mp2-mJ#{#+V zR?jUe_H+G|<_8}1=x9sM+kpowDX!6s+KJJ7hx)IZ`Y!NMCba^H!g#2}D5=j*08oFs zsc!=hH1(4}6bAL}icwNO+0Xl5oB9s$;ii59h{B-0Q&2zQ)z4QQi;^yM9F<7GCmSi7 z8Ym)AIR^)#aOjxPMNMSK3vBA@76Q!5gD%?V-gFYW4zM(w3dFB@vi^R2eZjF)2WNf~ zC4Z*+$K)${&X#qYaY}!7?{A0-!Gg9~Xg^II`a27}2fW@$6Tlml6zbip7=14>0`H!| z=o5s|OHHr~yunDzftU$KyJi#%g=Y>?f0^lO1K(|=E)Yerhx@q-x9mEAO~FpKa^eKs za>Ne&Cv_s6QGp3k-T}CNJp(KRM*nw)DnV)3fz&_JvT;-%$DflwdHOaLz~a5Kjxwyv z5kC;8Lq7AgJB7bLpKPzXluEW3vBAT7ai^`jLDa1uaWtV8C>5YOr)>@ z0`}*J4btz6tl__>^+Oe7@@=Yor{D{PACqnhC)tcKxkYHbumJ*^5%7xNp&otAJUhQW z_{rrX-OIm67g-MHv+3K1>RhsqH|j3&#f4sM9L3DU(M&0ovupO zn}6%-namg^Qg0yo#^h(nOFvt5hN^!H1(4}6vmm%D5=lR4p9G1Q{MqDF!d8a z6bAM3W!B2DUt$DeF*5p;{Zx(ZkOaLhCX82Dc;p6cgrnT z1X{SXhb}tn-ZUAyer{#z0r5aj9Qx&&5=wT6>24#7de#+Vge7BN9@AtBGJs0ej1iM2Bwo8Kf?V8+TrzDsk%jbeq zCBdo8)t9^E$hpgD^;bu1e5#zgJS{(WxylJht%kFd3l14uE;wX1bHO3&mkSQrG1qBw zzNDYhYQP^DX%cv+l4AEUT5xQfHnIo-cQMiwa91UjI~PAbZAQDm9~x;n5X(VVw_>!F z8C?#$ZTB?v?1?L0u1v=tZ&< zW1I|RNB`eYcWyVg#(_y{+Pj74H156WVRR*PSr%X+o>|0)C@m$&CctEH$nHWH(?kwp zfJxVoGiDzx)r56h5BMx2tpxt%$*}@*sp^h1d@)e<9+T?=zo?`*;~D)vF&aEBYL$m( zMigQ*$__T#E*WiC7`@AkwhgMKa9ONmHJ{(RsKg`8-5&5bBTWNORMMQ*dFv_$-NXl+ zN_=Z82G2@-z%vuynu@`O#0Pvq;=8zF@S4O2d~M=;Ud7-&i4XYqiSMF{!9OQH;O7$G zg%yKuB|hMPB)$tO2ER&t!2eEs=T{8oA8(xk+*(PJ=kqEC3lkr3pTu`=#o*w?2RtnC zt*#h6Ht_*ZPJHK73?>sFaB1Q@yJB#9;sZV>@tsvMSfBV#)LF>Pklmx77AOB%q&!hS zBIlz6cH25WJzk;7^ht8_up6p+mY6*ia{o-VZwZ9opI9F&prW zMmwYef7@t>R^VS6?XU{G%L&oaXzTC_e2~$OsKBQf?Z^uJbfXKT#BXxm2BaH)}YosplWkwnYzTHS&;EhHa2Y%B?UEnW`G!EQO+gGXq zJith8V51s&>a)vV?CRK+g*1aVnDXpi(&@VC75d5JJ1zJ3GRG&ycO0!ZD$5Ng#Vz5< z$O|e)|D{w}7lB(nw#mroVM>)c0uMA&5BP-{vW)((F^&Ui-Q- z2DrYZ{I=~rvSYtuuH2ek(X5eot zO%e^9dty`|Z3BFWk(L4H>diq@}>^t+ZX>BaO5a z_yi+$fzLG3Qs4$7b%8H3QXBX(B}sZWJok#ZXKlAQJW4V+y-Tw6jH?o@q@}=D z8mSBXdm}9ce#J;#;4h7|6u6`HQx|x!k(L6VWTY*xH| z#!F?5J7i%6<&`tHELm7Vd9!Yjz-~Q*!PSZVR#`FSt4^T2FR@>GChq=1iKQbuLV3yK zV8w{?l z7~N*|9tSo(TRn`;o3jSPCaUDwm^;oWy_vGBn%^h60Yp}U+Hj`tVB{K1d7HmYl@eIk zLwZF&RGv***>K}8+>if=jwv&AUt#E8?@v|lwtpVC$N>oul@i#$p5gV?M6c{Ostif_ zeRD5Y_{sAzHT`mH+NHp2lq3fr|9ZyQA&WQ4v(;zKUr5lOP7=V|%My^2XG`&HFP#{2 zY7C@W&a4l2cXfPm${i37B~7=U!2lm6$3Xn4`S1wy!sksMtEQf8CbJ{dhw7#UmYi{- z|7jutabSTw*5w~=WvyUvv9%kamq5KpK@FQ zL~223peyE{Cy$eP14wtwNqIv;o;zDyK|$I4){^jGg=~R>`fbCOc)oI8Yf=7PO;1$a zr+~4o9xZVKh;#WfloKmI1on*$5$(rJF#SQ<_2ZC-4x#kt6i?NoiA9~ zYBa8^>r-u&?EznIq~*X@D=A!R?Ny8p{L9z_a_`GG1M1H+U0vX1Mw$SkFkZ=Jl+;fQ zR^LYb<)(fTh{B-0UGP3z8aCto8dE=k+N%-?_|Cr|Gb?~UGg2G4mF7>*`_HWyJj9YZ z4kQutAhz%vbXT&rC`%iv zd}+?tY@|euqwGdYRJQ-TNVD@g%hEWI=nJd->y`VxNy@u(liIu^*#_=mQmeew50rZ+ zW63;5`ozQnB$$H7Cn@*YNy^*4UN_MtveW=+h`M1Ike{*G9a(RDSoM9)Om={X&TXx^ z{+bc*$XA>r^1TpqtAZ;RR^AmFHGh-*r@7AgC7o;f>YSeic(Bfx37a`R;LD7(9QaBl z#ifkV`x2v11L}`7mM-uWMp^+xVO+`>CH2{%7U~~oy4t|Yj5G;EVO+`>CH2`-hWb-Y zR|oh~BTWEN7}R$vMoE3P5TC9(E=jtOx-yY~?=e!gYeW%&%2Omj6b>CTx~PeqMFX3< zx`hC<@}P_Mk$oMosjFAemF*PK^%6_NsX%mvhgI)hS5S7UZ%dS=%uk}^O{$;w&={xm zXNLnrR0tNd%|iPP>d<#A@E-6EC57VlDn@rDM&M(f5HlcWIl!KhqO4C)ZnRr6+BKv9 zV@A6`EJ!yFR^KkEZ=3oN3!n{LV5BY(i?P`U8|{>gcFZVh<;otgX(4}rEzjCi$--2@ z!j!jAdT;JuG|4EHXHQRv&z_zr&z_znKWw55)IDQh1J<^a1O6EreA4D=mJyI0Dsc0; z$OVTi6M{7A&^^1Qy1I)sMt0osAl)=;-3acj3OImce1zF-1F-=sW71Z?^A-*peaf-5 zhw7AHgq~Y5K)duL5QY6-bwl4F=HnC)(NeRWrtB{?u^#X(Mp_QMMoDowVYKZNV{nC& z9Mqp>y1Kx(8fgU(g+YC{VwBWpyB^e^Zo1mQw;O3C5QTB!V3gFa^ff{~8If`T-(>11 zfhY{>I|c8v^)XjJh5FZ-`U%v2L`k93!nwyD!;5M z=d{+$tr(z0Is}Nmn$IiSUZ>3AHg$TybBwecc%G8tv}1HrV(h1W7h~xH&o$BtAPVEO zW0cfq(+=FNNVVWbHl3gfh6ba7$?VlguMls!oW zo|^={uF&1*$jtK!3s;rVMh@iY2#D69(}6C^CCBQ(rmk+qh(HUs#?VD)$&w93I{4Kq z=*o`O(bcw$^?-Pw$LjrZO$jACSl^(^!g3J4Mu_oC%}BNgAxc?hQ6@yOG7GDhsYBPA zW7#6KQ8!(spS)1UIEl7_Bo0=eIaI!0Rfadp{m1 zq}ULH1-m7|{N%vO#x4zA?rIAKOSs? zUEncBS`Nf=@S|HXT9z1r7?-esO+WI9fFIkKt~T&kBdq{pIrz~wKUO40Ajaj^6tL+> zKCkeDV#wkFJl@o=1UCKXm>*9|jKHSRd`{*@r!ab?>6*aU$%zCcHu1J!fl=59DCu5_ z?x(13-jZUR3}naE2dK#5=GHhcNoDtX@tnq$y&f=04cWu!N`hGyU?HAa#D^#?B?l6~ zWN^stLKo9SjyQlx*N`*jEY0@|E$=(F^#}S zDk*AYbZlbuoruFfFo(Oq!_1EfATGwGhS6&iBM=uOBcn%{9}_@ajEszu!`a1Z9KPNh zZUaARq;cS=x+cGtFa_SXk<50*=usBLIFR^)Z|#cFL}COIUu3kAMLRxMA@K!wz=tO` zAmInL!3FD3FhU7#bP9oWN`Z9>fptoObt(ee#ysu-_f%3e&4@S(DWIE3&{n zXWPca&JrY-rGYF#@}L=zbTn6^rn|+}2zOaSL?4hPyln zgc*eSFo5}z`-K1a&3<>o&@@8Im!3# zK*uLSht3!Ww-7y&j71d$JWk(H2ZcPLs$c6f$Lh=S4CpZFXS_evv&qChIjq_e!Dn(@XOISd*Ffn8AN?lbHa1riGH-*a;N_swOiWNNd!5 zzKS)`m#`-K`dzK&2G$R4+t>`g=PNHp@2g_r8*A%*yUgM(#h6uPa4k(vDXTBSC$tD( z)lcysNL}O&i>WcWWVvLaKhR8DyxK22wPW%aY?$+2_qL`xKDp{G`0d^fY2V6PV;o4g zBqyQlsm$^~>(rPW^Gal1?|fxJ8U8aymG=)iCgdOs6Xi#9=jc(*j5+1KIYpT`-VFCU z!BkmV*IaOwGdkJ71Dh7}i$(}??keZIz@}jStzBJB z&QtVLI@m|EQa7P+Xx#`tR~4{xW&Ag0vkfF#SQ%s6cIgm^F749$x&L&vR_|pQU|05j zKw5)4>Wr}@Z`52$wIfvEp$s3^2W_Av`ZG+1q+0sZhHUj*S1;GqE9EO+wtY1v(G(V)Ve03d zlneHm4ZhmBV4sOPSg_9;GFY(B^5z9MXhXiz8nXxdw36bw*iY~UCfEfw1@l$xaus-u z36291lrsQt4yUd&sU8ri@N0=ZxrEeD^i?XlM2-;S+g~T|HRtM`;#-T0!s|wRH8xz8 zHY}^TbY9ynDsr1?>jD2$NwIqvJx7n2Y z7l=PnKp@UUb&Sq8**1_gM@B|3N{m2~FYyD3G|CPRwo?kWW5MFP9Df3FJXq*djA$5P z1mbvPWQ3!_2*hW(VtKmvc!5TZ^}deu6Qoz9^uVO6PuqB13)P87SP8~~=-gH&@L`L8 zebDOGuMgVe*R2XVw%C5IZii0(wh!413Z&(uhJH>QXHHB3F*Z*wOUh!*5F7Jk@e?vb zOop|)TUfh^b}u#5Z|%OY|Jse{BUDAdwYyf~*KUM}?(s8ITUy3N&y4gN&y1Y{3dzr3 z)afb_8?se$mIqyn4{8E-{o@)}%qVRRMXeKY?{Zie>!i;1W zEufi>)$P*PZ%dljR_daxl! z3cw?bGzomPk$S*OjWh{-xsj%TuTWAn%IFGX>;b=Jq)FgyMw$ZNK0_U&JB@J?_!}ej zfZOVGC(#u~4^pbc58QVKqn|5zhP$N}NRV0)J)dIzUn*T)>A|mO4PPBwW*{s6~oCX~`G? zQi;LUmSyb+#@GVB$y^-)k}BB^0m)OPj?v9g7D%c@7D%2dSw^HvwiCd+OkE2|s)P%; zr)8}Lj9E)|rDEOi-$YA)%-i zhWL*b#>AsqdPVUEt(;yLFzb3>z#vMc^}2v5Rrm`Sgh<`BHz|?v#u~)?Bg#BUCIT>@_gWz3!bv(ANb{hr*x4AOS#}LUGITk8@`pEsQq$_wbBld zH7r+>UW$DUjcG?NY4j$hd=h1~Z9*#2Jj`N7l#&n~LIled3eT^pb zo0ilLkdBfmUV1!jQi>NCrFz`3iW6EPba9_hq5(!-J#HEH(!~i!iWkWJTA6=fRNv#) ztylk1P15!H$^0{p3Y8%$_-mzHuAfr8K%_Ip%k1Fsy>T90nA-v85oM0=&2*!SIled3 zy+pg}^X<;wD6pxdaKmo2VDz!b2)xcnW5BN(X%x6nZ-mMO02djl3*28xl~G1dFvd~f z1|y9DHyWu6+%!cUqjQWg`JPCck4`Qh9m~fii?jpW*VK;#2~SoNkk~?`or2NnQ5Hxn zA`2vtN|sR!E4fpjhSegh7h3=$K;}BAZ&{>kjj;vX$GjZ@5}w>r1QJ`Nj?wxk3nV;| z1rl2&%P2;wN@ABtBW)9DJ*>8cMSV(-2JWOEIg7G?tf{7rQ z@c+^fy15X#76PkIZY2W=MLHr7uft7lFm4MQFnX4L7j=V{z?oJGO(2HvHy7+mk z?vGV_f39iS7SjVHJ^pq!bSBB)N`_8|$!%T^5W6wx<#E9QraZls%n>TU`v9F~Y9_eV z%mEgyG2jCSsQ%Okj!=E30vmpjyh(F>rRBN7P^!RkmdQnFBd#z%|GzV z1yAWB50-MlU%K7{zczd;{k`_f-R%;$17r!vyKwcLcwwqPOX%G!`JznLC2a&TKf3F3jzK z^N7yIbfb$ozBkkTGM$Il*xYu2bd*f-(&K59QoO(@)#I*?uLyK;15Ba;MqNGbh4s?K z2}g<-$ZbQpAqkA?d)z_v>W|hWo#2X>aa5QT@1ZL4EX#8fh;*iSnH@a7H_oFAa}+P* z?KqDpb9`^68(qxty_xQu|vq%q)=j5G?o(@0~$ zUl?f=c*ST{F)~fStBo`Y{Hl^F*BE^xG6KIpg^^L8{+SUm;_EZbM<@N1FFpt+jdqE20H;pzv>sSzOaSKSw8^mSuw0dZL>6nK^S)dJ$O zZ~ zN}~;=j4DP(6jkm9bsB6p7i`ypWgW@QVIVxjZo!E7v%k)AtIj{c+y`;(kRK#L%Gb*YDOPEP{!J;r(@)tp0SU<8riRW` z`CG)$nKik=D?e0Hf9;whOrqBSolsH@KC+afSD*{5F?WEN(fhJqcaPh(WbS90xqEQC z7Ev<3L$_-YB8pkNU5i5Ic@}TiqLLDs#XEXj-nPz~z=cZU)*)l2PI?cJbpLJdSxjHj%*|u+l4kiXY){T9 zua@@Y^6!@7Ei);B0GZSWMOj@CYPC%lPFT_O)5FBrc&dnRI=|yit3Ro5nwbwmMTj2`lR47mV)ZNK*ncT3R=Lo zDJi@XkXlGCS+Q`>{EAXDNoHzmCdouiE0ZB(@7x@E|8bOLHwak@HDm9);B{PH+8x!fsz-WFfRixg>gpz_2Ed^CaX8dUj zrUkrCNue907LtyPg?r{#l$uF0Q(H4hrkA3IeQ?lY_A2D%%bNAV*+M>6Q~5c|{<`v) zY#Ed4uvG}ZaO;(ZKgS;E-Xrufi`+J4%v8z(24ZT4vx)^gE!P(Vbjeu^$UK&OK8)<* z#Md2Lgh&zAJ&zDg1U35+o~^U~2K|)%Igp@pXEAuTMloPbY?-eKcrK1&!1pPs=UMDR zWk7&)3u6>aa~Au)wx2t#i6&>UA3GK-`DnhZ9!;q$&$OT341KcQ6deT~s-$p7E&ct- zrXF1#1%bzJSMaxbbdH`K`Ws6M@UBK%T7dU5+Q9{QZ=)TOEbJHRu|Mdil#CDh9Q_kY zPd}zQl404Rhb}OdQQ*Z!Y69P*q$*X6E{}}B_Zvx1zR|TS_0ZLk1^96#>M`o&M)S6d3x`^Qy1=I@N#YzNk0dQ~ar-B#^f*)A z1`;oSxB;WLD=T!YoM6thffp$W)=PAJsiR-7;~N~^&oAB{r7p~oFH|qxyI>eIG={S; zQjM~4#m|_HWRLb-)jMX{SOG+OsBvbw5n#N+46FcR09GHx$*P?MOKssU>p@6B3{~ZJ zrkZ0t%y?;}MmreAf@~sCPY6Pq-uuy1(P*kD=~SslGp*}Xxi;zOt5@FNc#8F(b2R^f z2L6)K4lBS9wW)1l@d70UQSe2P4M?5KWfqW^9p**e1*Ip^JQOs@BvHr4tr>Pp=H&NL8FZ=kT!OHACh=W z0_u{}c_1AC!?mX1HD)eweJCtzw6T1kdzH|qb8xV+?D>l=IcTk@US~y}PGi^;;pN|3 zfiEe*|Ex4AEg)vY`A(;rIuF{wOn|WMh!a#_Pz8<_mt*lFwH(-dAqE-?=z8VrYb*z{ zL;@Q)I-C5>OFy?kjEu4T0}ULVZR706YtDjMyR!a)pR2K5tN6<`8kj&t6o~)95ufe1 z>J(mTQ@H|287|Mc!Y3K>s%}+peypEz`U3t$N#QiwDHu)a-;NhX;KEgv){2qQ(MlCY z;JH&6y(>R@;JI?Ux-`%@?bw0=nGj;v((*V>OUcRL@sZdq80@1-C^;FBv{EMn(p7T$ zZ2gzI_Iv%5z5>Kje`9?Ee5g%Z3yAu_=Kmqa6wm%_LdE^k)t(@rc|X6?>+8Djw_<@PVSa85|Q{32()SBF=YT^%?ltsM{(Qa$L8v%91KGas3)9R`0~GniB5t3nblSIvvg@1K8n<`z0y=UJ&ODZtU^ zHik~Ev4PiGMYe&b)!5EZys<_DZ;qlsGLYME(P=(l`RxGX@>!qkx89}fSDRQHh?pd4 zJNqrdA(YJKXRAhXl||G6{#Z%Xp2pYUwtFi0ITiehS!@EoYoreFHX}8G_|@MyHR)7$ zDtlvfyA6(ge$ zL`EQ~s2CYN!xGa1l8TCv(R(a0Eg-3=7#YRHB)1IGZ@Wuawy#Mx|oTa7WzlC71>P5OoO zZ6(E-(5tPe?siqQ@YsqbPYi&MRMJ3W?WO|lC#%YRSR2W^rC7+)=F6$@i5zrX{(X%sRVH(-U?EVDiHb zkK)0Jz{@23QcVRK7_-R7$*2&vFIC&aQuc?2yD|lotMv#4!&Rd0}%{S}awSQfsQW5sPy`T6DSl!aYDLqV7aFMplN z!6W;1&ysE}#?R1}Ko^n4P1j&QA*Amq7Y#&6UN>fUQP#Tq(6Arc7G3|83_moSk{gK9 zZx217N(O#?IU@-TN2f}9pBt~JD2M5%Yz%?VG|~w0SxO2gqtSxVhaw~JBU2a|%{jg@ zmbvsx{pjEaW`fa<8Eu=<4)Cotqn(1$dmC?+?q&jb zi5X}CalPbYuuA#lUULIXG$kX0i=#py!IXRqz7hF=L{{?k3T)Wq+Q1`?GzKKRkfgTF z&sC8Th&vS{BRt7GEJ}-_dzh-N8||7=vLy`%NUnm>t`)^Q%)2)5QX_SNxLCab&|39rrl}1L-QL6MS;Fc+jjINH1z$NX4v+|C;Q)zH?#Kck z`zkUz20UI#1C6sb77Xf1NncAMDNnMAYyp|$;BU*4vOY2bNkd;_6>!+XZvpW`xO&YD zmSvR#>rS-IiMN;&Z6KDVs)0{1C)&X1L~@cR6V!xk>Vb79y5_{E%!w`#%YBWNKrD+o zV01$7bH#k%1mQ~y0M?yYZceN=Czb=F`s4~|FXcw3oUc*)GE+1HyhKUz(&sY5`9qDy z%o11Ni=kBLAZKw%UfiFMC!pRebZj4|a&txJP!NECQ^VoFd5xh(whD30M7)1dWI8Xm zpBL(0wz_=xJdA~XH zuXV7eb(t~XK1v#DY>?gfqJr_h#@eJuz1mvMa$qd7q%oR-LN?{H0?0`!m;saFfp7{WH=Lvb zBlvlm!jh2|ex(uvqyI<6Tx*#)vvF8?^$LZsdH)zc@d-L^9b&UvUi%k}@n%>aHB=U$ zywyoCX2zpLFYobjzUbzH#`{@6hObztADfAdgY}{*XMnA#MIgZ#_oBpfKFYq9EHg{hHW{bMurLBitZb8k zgz|veV!uRJ;TvoP6-eNOFf#9vYh0yq(R{+4MZf5EEt?*$!`H)YsoovqF@k{(;_*Kv;AK35bAJMfs~lsFX!xj ziL*%JU%0yuZkw-7;F$S(N?$SVtY+^2{$FKbV#{3kRA-nTpEwHB*HmM06#Nj&%xd5s zcF{f2w?n!HOji>)WV%inEf_pG^7(jh>nAl=!T6VX2Lg?UndTPo`R0L?2m^AOdD#V3 zN2pxCTup!x7urVf3(aJ52OpMbEqONVS?0=lHv6{B+chCnbf#cZmc*QADLajygDbVI z(>!I@$oX$OyGAb+-W>B{fi|5rX~Un{?YSjaB40n&OV8Krq|^m|OG#nZw?Q^*xb-HR zx-RgGN`jT~4=d;IYHn{lu9l?kzzjj8;4s6hy_}&j36iHW z8)-H0qedD9zVFott_ChSxgv%A;%LF>wMrE~fM*(MHE`~0P&W#^(MYR-|EL%Hl?6r@ zY(RE3@Sl}bF*5pnWCXtYG-OACZ#2?s;FpXv3Vh`02(AYH{EUiJ8D;carAqw3r=MA8 ztZ@8)S`VFEJ#-4{;Y&989Uw&@rF`us29LGr?*J)-l8?cEM?N5hQ1UVO2djz>kWwi5 z7*GizFP%(XbbMWu(rIOMd}Wk;x!ULyzBWqMTy=B`UmYbY^|7*3%Q~K zQWhm21FE9R1tSWhWMxpVlJrqSS|zPqm9(r%HdvLkfWNZ|90%6ts+F0maX(ifkXDgS z);NK{f&Xlz5#aSk zY6F=|@easDhS_QtjQVW3w1KP;kp-R`xquXbZ~>`+$^s*bKp27bdT7s}9=bG6GA&yn z;ITFjO(1P-u0-11BwIKphWv{-(AELGMzE9F7#XB-nO@(s?11l3c7U{#GVugR!!h18 zfyri%BzD~)zBV zs!Rd`sSp?AS++t?)3r~BklVFDiX@a@*P6u{7#XoHgb`R@BHd((q*)jt`i<5cM}RjeDLkO0JD#b1FVN_PFVZd-Xmq|$)M!3a zfsds8rS_meqkFM#0*#!UrE?x=bQkgzXg(!@Hhm4OCi*ih`qjW%^e1GZKOqzS37P0m z$V7iaCi)XH(Vvis{)9~QCuE}6fMO2PQ{JkRH2JZFSA9cCH|yuV@`WgUt2z7bEqu2s z|9<7e$n8nM*{_XqQC(uKy$3zz-CKC6jKz7Gwek{g?G?`$dxic^b8?kU_DUeh39li( zNJAOdPq{MzJYGrMx#vv!`^f5zK3x~gr)L9IUW>Q}5xKc@!w2=IqW?le!j_PDl zN>c-F(L)1`O`8e~K25FcD=&ZR>(Qs`%k5Lr4d4G!ewozxdURF-&KIhLOlp)^n}4Ai z-eXioCN(FscIpe&A6D*|scxF76UmL*laV-4jI=0Ww;nKAPl-tA4jdWv~PPIB*9b78ko8}gnkXnD(d>O0nO9H^Xo z$3>R#!b9udv(7uEmV7qmt9B&iJ}Z#!S=OwVquu^ea}~a+$B6Qb2#MZJzZTLp%0)ZKELF^IQQdX-p%Q?Oa zjbHE2NJ7KW1?9AnMxSzaiR_OhWx)C_vD_+i3L!hT8>MB}_ll(id&TNgdo>z>F=qJMHqYx4)rpHyXxlLyJt9z8qYitKS0(`xa!fR{c&$&-TP~zDuVdtsv8#no2o|7z; zRo-hkud^op)^78x0#Yhsa?55vr6TJ66%yVuS(%MP>em14vj_3A(~yVfK-$A#W8$1W z3r5sg*bR*CF9-arRqRY0vs6(TFa4wmIYWwxk(}Gut+2+}^f1Xa^Ty&WGWml-Z$81R5!) z9R!-M^4taOm(|u!^fTlB4SI-Kk&QhNud0nbezDYr5lG2D;4U)=E=F?R^_!Cn4>khT z=eVN!ELT)t?HN0FX6* z)($a`Jc+3tg7idvI8&KnL_eQD%aX#~a&fhw!mDKnnaR#d{K_)Li}Ks(^)hR_LeY7s`4!zlol zqi6LKL-Z_Ay=RqP?^Z>fflpf|EbBF&gqrb5sA;y&6KcjMp=Nv%YA>G-wficgz%eCN zJ)aS^G(TKL^fo8mreX>iUH&qgS zs!O8JbV>ByymY%=5qGM>@%~g)?@&ec9#vHDQbqMX{XpmB*LD-C4g7B-wSfKB=8_KT z6@?Cprn^;Bu7f^IRX*HQw}50M+d(~3ri0$34+B}gnGX6N2}{yJ*QOn`c7K7&s=AOI z$aGMl<~yhyV4kUX(*q%Pq;OlqPqnf8*|r0I!U+e*tFpjrmTE=Cx#+{a?SC2TTVwyL zSr=-?x=?)v#8aXA@GEMs*iW}k*8%>mb-H!Zn;6wP-8p-@PPbU^@eRuJwfzc4&oFAx z!-(mZUS)dH2@VpyTxPn;#^PtIB>HrfM4zpa=#y0vz2U#wZnLV*RL`@gw_VkgH(pV_ z^=E4uyLKbB13bn^ZQv`6lziL7D++B1O~!zAW2824v5}Gr@QOmB(DY~(%T>Tr^w=^}oy@`E2}^SJI7Ux> zuZ$;C;2I@mPj8+nb9!s*vk;z%3^&FmhesxdCx`c1Q|0a}`WcTv;gQLlATzley@N?r zGLK1kFr;LLY|W#37B`PnRKS{+meW{wI^DaZT6m) zoH$u~qLtnx%(%EM=c3&IOkLczIpgBC%^4TB7{7RJl~8$B_!c3;8ZF5_zf{}B253kGxb9SAvX z0UxR)(Kk>q*eCJ<4~%?+1%oF?KHw3NZ%)DB*^v+U(#SVdFnC$y10EOo<`xWI75RW` zBHwVq;IzmGJUjBuD;T^f@&Vr#`Q{f4-VynLmqfmu3I>-(KH!HU--3d{Mrky$fX^yy%}e#+_u?h*Os6b$x_e87Vu z-%!Ee;K&DjM&z4YFnC_%1GXaHaKYf1$On97>lK+6f}w05mwR>m?q(3e%Mn9$?XS{?UcM+mzo<-&5ft#>eZ)#iZ2@KRko;#4%w)r|$)26-b5 z$gWZ%zi=~yPguz|*;cS^7-+~(&@h;5=a^Ia3kGZmgma)^z}a7jg9QUl^+KFeFnC$i zIaDxsRV22; zcio(V0p~N}94Z)a@)F|Qf&phLAr2P|I4ucrUcrEKkPzn=3^?Hkai@X-XBQzZC>VS? z`nhw#fb)iM%0@>yZi$@nBobTEDq0cehcd+s;>jYpFLacyq!HV_uL8bINdt||rxy%Z zd6{=WQFXMbY66j#g%MS~D)RaDjVfFbYd{<*z3L^0T`5N()=C$=+@NV0;e$TI33fCUqBu?Iw| zwC3e#UUV$^J>JukY{FFQaJ}?ImLx}LdB0tBG=WzrsVqzexRlx6VI{J3yhE^HLbL#p z$}R&A#$8WRfhkK8>&x|SG~21_&OCRSXL1*xWk!FiG_cNi@wlS;*w0q$SDaEE>1!Or zMwtDDki0iSVpoNFA$i>wsqW)U_XxV*sW-vpOa176gWj$ZQmT8z>;9nXzFR+~h|ztL z>29HDiCq!2fcRZK2kfQ(nWk$T^)W5uxXvWWZWW8D58onj|30cNf6~&jGS$5@)x9#+ zo$Q~{eIKRebL#rPTGrb@LMYoggE$p!;1$~0OFkcDN#xv3IhRBqdr5SOZDf{`yp!pT z$U+b*Er!KgQ8Po|LOgceUp7qSY&hM$T8u7&r=j3T~wQi z_-O2a&yXGPezN3i4M!fud%cp6+Qa->mV}zg(u^PGzcG`g^;5D$y9fb{^vY5tLY8PJ zCPFkc6+4j2Qfb6ziRM+ylDve4URJzhNvN4x5^5$(Le12YP`&kcWwNwxN|spRA+lDl zEL9@JOzSfd2sEqtqAR==5!88)ex)8^SKcN+Lz^SnmmW-hwpcT70htOB#G%8iEw2XR zisLY zq4)SOJ=1jK8D~0{eCS?LHp*A4m2=Hnatib;d2604YmMTgQC6xbw%R(C`Sn5nU(MmK z^fR+thvjx<39li5?=o8YhVfiI&CS2E@%zp)_lQ=74V=jI|~U8ouBLd{qgYR0-yee9fv zzo0po*EnWUn(OHUlYNlBt%r864#|#T=-|cY{buvfqWy)7qjxIToRX`*mz}RFoBY5w z5DgCtn~|ux`ja2M^!#z=ifm!MT!wzCN}`{slISO@B>H(Oi9YO~YT7V-|CVURx@>ZM z+~U(c+$Pl)h-mP!Q(3>z9B7HIm*RsoU)v8Qx^*Jd(S;LLiE?yLWnelOLB1PgZW-|6($%Zejr+FePU|>71d8U)v!==Ft0Hh8tpo%Vn~m*DlQL+AEAoR(JJOB$s4W|bSC=< z;V< z7V31?@3-0>(qHA^>-w4BXk4ay{Bv!uKMMRCBaHwHB~@QcW%R+w=tD>SiN?|ao@b;n zAPR%}PQfUuAM2$amu0U7L^|kdnXb#tXbXt?ig9`i?VN>n#zH%1A^izeQiNLhK{;SO zhHl0}H)kRJ1fCy&p(ch`u57Xc6m(2t4(Mk&D(qE-QaIOh$=jPtVS2yauo?gP=2^H`RUFjYI@9E7zMuDNF%`aE2+u@qcipTG1GTb^&QkB zm7hWcUuaIWfR`9)42UN|SF2!zh0G6Wp&n->Tfldk`c*&_2K8-I|DTZ&_!}ivjEvs1 zu_{n`APU@FQ&urDI>ZvW3P=nUqc8L{jBXmn>RcGzOc<+kVRQ?^SZ`sh2I6?|u3Ip| zce%#~tcQ_y?lg?$X&5VWVJy#tu`(CNatnhgkoSIoI3B`So(W@RE{x^L2Iyrv`&6aW z9k5bq?45T7|=ax{qR}V>D$0p8EFNOHC<}+y`OBel*3k+ljE zhy-J@86}|5?~!0k0}WpomfulbR6ro~s0s?nKow(RjS|xbyrHqV7BD3|SN=wpoTnER z5bBv|l}DfudH9RVK^0@-HdPRa4*|wRX95ri4*|x6YXZ|nD+|+mEhO)kkZ|dtR-NOI zQ23-sIz-_Sk#wZOU9AwueH5bJ2atKSz*nZ6Z&KG+;yMdfxPV)A?G8ynZBD-0|Bimj#SM^(?r%&C zlJT9bH8z1yQ&M2_m3WP6MU~urK-0mdsntsgN9N0yhMsc3(!J$J^b2TcTp(7@@8c-S z@vSZi4S%OfdLRBd$xGXtmM_(d>FV!pV|du2Y6G!5EKyXH!i1fYOia%TZIokYc;_8W!gt3Ec#WaB|rv8ThIz}yqU3Ib2kHTY2 zdJ+vu`pG(F<2)-%^mj_aaOU#;bL!q#^)uXA9h@9w_+2H>4df-otnwlIG1`d#gIgOG`GQL&UC+;%- zH>Imv4B423MHH4^NC{K)Or1RyVc1l%WeUQG9v~x(mrvEhkX_6!QzD#9`U{casyu84 zGTpKPx^kjcNF_|=^4qXpa-8Sp>wY1eX{O`Nob7M}Uo#hAevQi4c;Ekd`I;~j!C3~n z_X7E|tW+l?rwa?lf2nj?6+m_YVVf0LIXRwZ<#S3&XUv&0v#_vEVamM{v_+{lX}DPp z2p8c|2tx9a;U>u=1BgzS8RToe-7`l%uPOXjOJxg4qma79EtZO$3V|=uMUZ(_f_6s- z5f+wIOwDp2A!lnoRkb2jHM8pnRPC!1Hdh+UE#O|SYn1XB@pIymTjEugZRM7||OBG`!6#t(ls!gKmS!rjo(k3fJ7%a34M#pWbtVsI> zo~@*?NdpsNCCrE@1E*mv!-_M8P@y!CZZ42+CXj9}kZvZBt`DTVLz0W6Tlh#y)?6s* zed#om<+)InTPT-VSuO_>(qLn){Mt4n<|d?Yc_zT+X@K&5u3Ui2Ex>JBo8_0R7}Sd{ zy<`0jJ!uiuEH|YAO~=mq8TR|&M;L8s0e-a64lWF9vE_cJdW-UtRe8YGAhYBu(0mC5 zn*WC`=1or0K-5=Lhyi9u7=hRJIOBLj^y1`7s+A~mrww|+Mk+}A-LIrR|1kN~(!8EF(q*QoD=yS0)8pJY4#4zOOi%>EA|8}LU)8U@mqW>WLvkA#Mw zd-aU0 zwWxqV-wn?SJF-}FF;}Ym+BQ{bGAaR*!5YUW##DhR@@w4=kO(FmO^lgZ6PPYNvh4lT z3dsi|Bp-v2aQ|GZ>($&5d@MK;Q~_Lv5-X+$Sh}eJ4`0)vA^+@w2660>TdF984H#3gW{@hT4Fypj_7~I=>g8=ow4+K$ zXnMODOx~PF5&qVnIYPsT{7u~2(@N|SLR}IXzD$+GEdrc+)Q9@zB)Ki`*!QNGI z?P)Q$fOr+3WkSKvx+3po{h0UhA~i-ta(f(jt@$y|$%CFGSC*_Q7E6&j zJvqpj-PKO^d{PcVB&aC|X@~!z8mAnDNa(vZ*2&+m=IN@a=DJ({+kJp3sho+?I zbvZ$Z=jp-3EC=4D4B?@!c*Deg&B*05Ylb%{*|G_N?coB;T6m%j_4Zy~aNrkauJnQ- z)2~mQ{nV2>3P%!h%GMI?L^Dy)zGIHGVDp}) z(d*5JanF2yx^|-%=%;MJfV4na96%!5O@2PF{Id1oi{;AgRJryc#yeC`%0X#9LNlLo z_y|%CCV@sH-U4s22^|O0S5iHG=J5KzEek6VX8FinkLjgaNZo1Pw19N0OyOYuPBkA& zJoF33(XA#1qP8XwPi5&`IWZLEr+0WY`Vh|a6sCpp%`{8Tl533cyk1ii-GstPMpiNq zn~~YagjzT=0qmxaVs5a<(4)YcjWh!Mk&?pAuhD|h-@LIhoOxK0s_&rw7E?b4L}5_h zG4;PP#ttxAOm6L_7F$@Xt8bb5-mjG#CQU=`=0fgfLhj~5?q)*n`jE?Oec~s|5OcR+M4Tln1H!CW8NF14E?F7W zqfVbk@1^BOsB(WFSg+Ur@lpSeFYk1sAC!&hACh{#uRD5baaK=J>AcLU`bo{#-{@z! zm`G)$cAxFOuAQ<^WArC)UfzFuh^U#zrxi`}gxTHdWF{z3IT2GWA0*pzI2~3wAxewu^ z6cPdAxI9Yrt_le+Pqc-!`c#EDF4y5eToged-e#`W{VdAG0)9rPx;RDg3Kx)~y{DHw z^q1(5Cse3V7i0%A8ljD>*q; z=NPlyMiUrQlq~c=)m}}{4%cgcp|o?YrH_A7;oU}B`9!tDJcw6*Zue5|#rmmFS{WX! zWP3IU;RrpzlUBxT_wcU`H5#Ou@S%H;k-uK`@EDTu_pC`Z zfvnT`ER*fPEm}G#Va(njv#hXB^~p&GZBeR2BKEQA=@1}wA%fHbF$>9ui|_X|UC7tm zoi*R~(d-?hpK|F5q?t%r0?$^i^eG^1;JrOLaQPvc&Qu54Wm$?ZKoXt`k^l2;md1fh zNh;)L=v4KZ_#|X3qd&7ay;NA7UM#m@kvl`J#s_jxq908Z(t@O%fw&gsr;lqj`Vh`^ z0OpYHn|WLd%SSZ?*>nKu-I;}e)caJ5XS0l>XcLGJy<I)8YGf4nhc{I@ z6Di=`N(vVVqXnY{=fF4$e6x{O13zq}QQ+7A62aAedhqKv=2r)Jv-!0Oh$q3XPR6ee zevO$EtANKTsgh-c*Rn+d;mEjJ@Ljti|5~613DF zupWQQ;wM631QK2_+A0_wYGJGd;&{c#2;W5`upUPGvztugA1sVE5Z|SH0PA72Eez&P zdLEE)L>5Ra!Qr+=#@q=bkhv3ez(-pA<3J*=)G;DxQ3tHYpFW74qqRze(xQNOSQ(G( zp(Xf8tIj46ujPdRyv1VX9f7M8bCz4oRRLj@4XYkTClkiZyAxs1f23gnaXidMClkgDTM=R4 zvz!medraqQ{D{?=^mMN%Tn~Cpf#y{On(vS3Vl~}Q3yU0t$A_uwhh$toBys)v{~3!k z4dX$%NYgMLl#7%wDA0SnNKte@Ev(esO4e08H?8F49{Y{8?>TnvUJ2Y_q&9FD+kcG% z_cu}-NW+vf8ZguJKNtmpvFRmW?e_szKiascAiqplcqml;`lgQ`3GzWQq68Gug9MF~ z(=&9&obTq!_X=g#w5WhU^w9(q@_{PGWFSiTj-4iAq7i;n)F)!Dypt;Dt3}pdi97sO zR}f%K)FuESrlArb4V)KI#F*Gj5d?xmfHA?Ez;vXWsQ`Zri#!HkJRSb=LSq%j~1cA&9g zpkVNKk&n>-DU!aZ@T-yZuL@7Kr8<@-DA~fxgF~MnvO!6OFBcL!dCh&{5V<(p52TGL z7Z=Zp*Bd|_?r&@wWZu_DHek%hra|)Y1!W7OK97&lW2l#lFSMhoc0GnBA@x#(zx9Vy z&@i$<9Erc1iakQ8OG3k!sggdxsYktsPtZ{DM(+Oracq9+SoO6l+&aaQ+ydfNeE0!=aR0c9(j}9u4-~PYXa6GVdiCqO=4umHe-%l%dY&!#CXhCm zyWLo5uymDhr9o$*t%Uawlx@JYciA#LHIjfwDoGx%UpVn<9pGT*mU~CFuhvg}0>kh% zO138{5GM3M`6R15Imq~SM`wJO(x?8UstDq6M}0*h3Sby`xQLHLWm+bF-qX||U-RvrIWnd> z{E_9dg`$>4)FPs5BFRq{%k>kcXeEWjJWC^*UTQp*O7`#A`jFa^yY2FK?>5VHLy2BA z(HCc<3B<7|KfQme(1&nl3-?|1E6z?AXl*MrB5y5~OCHX_@;x;Xo=Jqj8DmH48HmZ= zQKA&iOh^<31)G`6lqH!s&mzA-9=tCvEqpRO?~I&Ep8ejc6?@{FEB*4I0C>8Rs#|f4 z&Wen{pH5-)ex?>i(e-*29BQnWXWaJ{%;q~) z@tlVIsQTm}!@pLtybq>7{2z`&W&U@Oii|%qFgXbQQz|XTH`R~dsPx;X8fA#zQ;mAh zSy4ogdFToYYsIe}{FDU%#AR7Kn+gW3A>jkEoJzhKEGWFhPA`+6)0ALm7ECf*d&*A) z$#38vzW9vrv1m=>so}3tEL*&&fCK0Dp2`^}*LdL}H=b7}+IX2~CbYK)esXtvOGR0t zpYpCd@NgrI0H3L((0xY>M#pWfbYz~Rq5dG#)d3=1b#g|pjEuncU!i^_aIKOmSw_*1 zWXpmd#3t90Kr9DcEsNK5?kDU6J6kBmUFU!|w=H~v4JfOc*I+BN|n zvk7Pef21V&CE26EOhd5RE*L%5QrHGE4HYA!_eVw`(@-%oBEfQ503^E=BcndcOdCjc zD@I0diHyL1t>wC1Fe1TX8hGavMn=p)=F%lie>a!@ZYKR*(ob?j7~M>&ySY?%i!{~U zT&lZ8n(A&Y)!l+rlQ#J-0g#+kJ~F!1Qr!iTvx<>X%wRW`ZiSB0;-f3cSrB-pm1@#= zkcuqHX7_59{wLFtyrSz_!aJ!?KVZS#cOTikKrnFZX3se58zJbpUZY zqzLsaT44hcMYVpt&CF`UH#azk)x+z`NlM|;D$Mkk=g`)_n{t>)fZ zaePoY_mEBH;foXF9G+_mBb#*7+Wx{glye{4taMIgCP6ov+FsGaK{;6TmD)A+?T{~# zgS2x$Q$OX`r>{+aeHt#B(45BFb1t7(d`4ZvRVDYbfwXLJykut2bCh~?vYlI+z^G^( zMPE@@Wd8}oD|x`y$Jv)Vu)6ejI z8#Mhsh63E%d`f>*@wYA(>8NNTl+2|DQaDn-GK7Y6Q@>n;a(t^xLc`ywl71E%8a(#G z-p}Qw{%slz|Hksu0%A8_Q$oR$x+3q#xoVWtWVq`Nwtm(aKU(dsHTz?4RroO@C8hRs zJ%qGW?H*GhEZMCR^TDm9WyC1-T< zOp-I}7#8*|k`o|O`J5og_FPI2SQktafFN9%Xiw3_I-q~GLaqgYLq-|_&R0_KbF^S| zd1Um{nX2!g9_gT~W4aDCT^-;v%xL&4CyZVi8G##=G|4z`r)y(gNJ4Cc=+#fU%I<|LifV-lB~Okq!-H zHmW~g!iWhGMj+F%19OAf7sZqV(I$@w~fh~3V%4H zllj9T&(?reSSFg@uhn|!Pa_MEWMwudc*ErGz+@xJEJ=d~GE36?fcTJ~uq)N+pPTce zz_TZ-NW%kjaG|uSYcjg7#tFV%Y2pd+ziMpYeb29qR042a`ZCuM}|vc7=0F6#-nhx#F0K-{j>F~Z@@a{YpO`G56uPGe+pcyc#}-%|2$ zAg`1o!%%*#N9O5~@yTIG`@OBoN$n7%3pE47jZ&QmLAgke%&m@04*Oh05mNFU6{*>Q z^gTT^U%fkTx5;5>|E|Y&lE0%=dJzsnB)1F7gXc-W;V9HcXvU*2Q1;Rz=DyE;6!8V~ zJI&?BZ&kRwxgnjMz4TMe0i(H9skv3DxmBsTQ=2HAY)nfHxZL-~voq?oEpCSw2&# znN)cyH8O)trBXKIoNHvpfk)XgZvjtHQu$&RgSASmo*gbM7{{}~s9!ayBNjB)N;gfDmG;q@rD*_E}mhzYXWh7u(3Y#*VbO30$ZY> z*G-({cL(ylO|`zyw2uP27gTm6-@pw@s&*!fJ`fp!-^ku5?Pqd|`Y`&HOT&1D4a7#>-UsfHJ*2|f&byi0Jp7CL zN%2p!4p1*_UVCQUp`2-y`&+Z~?~ROs4{`k(*hLQ>mFGFO`}}R#GXe0!|0wH1HvMW{}elaCapQ z$;*o2ug5Zuy*oKGr#3CV+cfD?R=_8l&DFrC8L17#T>kFCR#jV@2j9I?mZUsWmZZpj zsD>KmVR9ShUL+1t@*vl5K=L4WaDdxSNmD(1wRt)Ue1nn3fp1h&eoJto9(|n&j`{S$ zvc*`E+`)3Pv2?tat$OI=#xe?=b0KGe5#X*$s+?x@or{dIm->&G`VMf&)QVnIY7g#RWl7ss}8h;wb zgR&Y4gL;!QEU;dB52!|f&-6&S1_RO~<;B#!((*}OYNGpK1D5?BG>7VGJVf&yqa1SMwd{CkyZd{iKRB*@W?i+I{ef__#@klNmuw?i|QBgp^!i% z7?aH?0fh}05{%iw%ndezaR`JSRY4&csA5d4QDXXlH3exC z{2uk^3iGRpt6z#FjQxeJm?n^QmHW|(*Xm-WAB7vQ^duUR^pkHW8|PVBqT^}^!*a*n zCwYIBdXj$XZ$2>mKT5VAaDea=*2^B}IeuDwqujHejyH3* z!wr1Rd_lplQTZ0m-9M3U?JyCXWuSX6kiSV+VAwl_*DM*|9#z|{9Di=*SpC8W zW6qSBg@ttrQ@SYHqEwqSJVT2?xCoCz5R#7!H%T5DKy^HTPzjf0>0nkUhadW-O)ir5lbqjW;u|Mvo)WpT9K-n+4TdeHfzM?kHLgB03o{D zOl>s1yCyz;^y8=_)Xb`)ebNQx9N-nh@`*TQK*~YBEHltc?1L&6ej6l&%@|o(2|%3a z86&R1mJ8as*te-CBGEe#A&4zgKv~w#tfd zd2@gEu8OkQq?$k!=5n;JvcoHyfin?{JZX;3Qq{FMywl6JRg|uN%G=7oZAKaazFSFQ zS2(K5EBWyM7wgeu^)r*BEA`OH?-qXfC+2|blvL?v^pVI2+~bl;Ht$4gp_8-FDX{QK zv(N#4UrFT`92nHic9Pgrv#p%jmYF@*%(j4kr=$>At6+3>WCZ@u{F85~FsS?2O8ooy zwkl~jC8ru7iKxsn!YN?{;?e!%-~BReq~#P#@(2kapP3EVOO*R%`WY@fz%Mu2(gOTv zMmx9wqdxbd%_ik1@%PglV3uVdk6nN)f*n{9Xq#~mS83D|EfBMEqX#oy7UyCe2v}d;GIKZ6kz#JmVEXkG#$Slox!uHo0{-(`z z6ZkD9$!}=+sc3vekA22?mK5OYjJC7@f5T`87vQKrJzL+Z{D;1$GA-vu;5mN_X%+Cz zN}Ag^@3M)Z!xt71_WwKM1s&^PHW`lZuC4|e81F>?mI($IXcgq{ zv0)PjYdPy$&Q@ab(UzT+C?}%K6&JcGB5B<~6wZD<9jM8BoF%aZ#8pYowj}w;#YBt& zUtx(D!S98Zh!GMo9!bDsjWqglh4_%!b>P8I)${(w(gOT5qb(`G%unD0$2wVJC5_3z z=a$|*+Rs_kP2iBlv-&MsQ&(GuS?zUyNfrFq6t#gUl9{-20#z58s^q@U zpQ(FiM>oIx-pbyO^fPlYGLi&0>V3RTrGMR9%jK%&8ne9`?x&f@N$_xgElRqzo(eOeFQWGqd%2YZKlzH--WdBwvznMcG)1yX}DsUMrz zTdka@pCxgNM$U|9wh+6k-u?Bnyoa($?zcdAoEh!{kM_LYX_Ug4=~b{0jJO7zKCWK& zo@9<~e(cbCd0-3y)iS?cmY^a_7g%%TN=xvvnBerD_jC2&$@(daWQ{^5CbMpsyXTlx z;^pZY8U1Fi+#K3`_)vOBe3_P1Y=vzgm0dOg++kUlM4ua!jj7V%#^%p7hSo`9p`bgn z)kgS5DjaTgua&}L+*Z0gBm>gN|>eSh%-Jj zLh`ouQ^J$=QxpRaRZ=+7rgwDM_oGn+fyZxGkXwQJ16mE)n*;Uju|}|HuPHY^Yp%6` zKkKh(6{8-vkJLgN3;Mf64cX6}4%T1V3A9tdTUD9RP5|roV*}r@VEsB!pmo6)MA;^o z$b|1G@Q0O8`UQ}JH7{GhN9= zrk5I$rE$8fD=!Lt2dsWOS_S@ANg+evclA|7d7BvcUL_4QPVX-mJW;Dg_zqV1nn+r! z@O1qY3#WkJ71`dS@VZF)jzY34`hAd0@Ru#3Hju~$8fVNY7(`Xey{a3O@Is5a^-_gV zDtSo)sYjVq^EC=DiKNRE?qtC=_ffcCB$2#7iKG#Q$3_yldt)S#yT6Yl;AfPyv((P{ zgKHW`hu~Y|K8tE9|T}B!M{@O^#00%AVvHcV>HS$Ihkg1WYP2jOn5XfwaAhYz( z#&rykgbXy!?8{{57@z!^X2+*uCpp0m@3(l#LE%LLAKR3JgoOW74iX9TH|3!B^&y(z z17o6rxEF3poh2viG_^_OtTSd*PQ5-3Aw5%5jQ2wFK4;F^Z&eS!ZJv$+V@ZrNS2x=< zjst(7r0|Z>A$xC-W(|c*Wo}K}sX`AMs4^!feBk~{3a6FPg3*e|2t3|MW572WX%zSm zgIE{?e#S_n!1vBEEo7I4UFnnP9I}8r@m|I z+orw?Brqvk;Ks-Wj82SC@5J()6U)sB%0%9<0Ae{z<#MZ-v(1m?K#F9baq2|2VwM+F z%qOj4mIIH}x|jXJM8ROK5-Zz`2)JaOE@b(o+=gGefks$A*FXXd_kOAF1sbkwP*(!Y z|ND2jbZkoBS&z#&nAc@pzX5+lQiaE&ajRKFb7}rk} z3|f&7IA)~wOp0xVvhz!~l4{_a^-#aO+5;x~gJctP^cjtZK6a?_u;KllynAEG!ymfr zMT_@-?7YU(#Sh=V@8M7W&$}OX!NQ{-limY3O4Xt&Q&A^5r{*-4u4ybTykm14OE=1v zdU4^sdwgT*Ox@$HlUOycP5et9P@dsf)W4{sLY_@_-hIhxr}=U64;=wKPrF67pqk#? zK1z6$e$HteGuc1c&v22F=Ld3gvTx16WFItch&)V>t#1q+ymky#lH|{pZY&u5Oo=l!+sT>j z6qrS=+{6RcEu=5!q=B_^7FuTEPRl?Ghy{6)K9N*zYM_mQPn)WhK;j7LXq(L+8e#qw<@cjG3g8nE@%3r*A;)BCnp=M{4}kzO3=q{(WYgM9P_| zuV8?s%%uoM-mPY(vuu*iXHF>aY|;sZZRx^dps(WCUq7XD0A-Dp|Kx8HO%x3H@%hXR z?>|)RjlQS~*f!D%ATbS#ClhjQoBE7CTnvp4yucK;fV-Q!lDWNffZK9s6?jCQPhGZ; zbh=0VbT5!EtsQc}X4#%dvr7_()t=JfESxHXQiEfd%DgxU9C4L#8HXW z;!%JHCo6043XCh6)9MvYa}dMI)zJaqbxJCCRScNtj8~}olzFwGAB43iQ6U^v?KAt2?p;| zVsIUNd891^e;^8fY=XhxN8*(e46cmC%O)6H9f{i}7<@buFPvboap#yJ@PyK2T7h4y zv4Ou4*?^5*qqYjg|FbHdU`5jbQuAT*It3&C@T2UEfYeUK$mljJkq+<(jkRKA#4LzL z;B`|N8SOQ%@>>{z7tgOql}1LtRjM!o_ui?_$mm3+3M25E1$9P7r&(4zK<1?~&FKA+ z5%^&xRg8?j9T|b2*`;op(Tz$KAAwA0rLhu<|4*m1#gx+Aq|i?RGqFKgD>Kb4rkTbO zguzU9CCrHVmeUrHju3>w^n*&65oHlL!PJE)1hP&88<=$}v=hKAQ=uITW|a!93uciD ztqEq03hgNH$F($twhT;x3hfl|E38;sK*}?)fvHc43`lVXHt>6_P+LGsGq8cFB+&+> zi~<{&`VqERHWi|HI5o;03ShlZ+EysL>S;t6fs{d**LK0^mDUg1K+Yx=Bcp$fj6mw3 zVq`=qh-o0puwrENdRuF4AnmeZWJJ6?Sc>MO{~nKED;!C%B^e3|Ca2;)>cU$8>Gk69dU{*gB^fZjoWqDUv>yLDFY3 z(0m>PjV#i?0}ad-gqQ$Q_kj&e;R_p(ISFjcNGxr;N+9=v&1XK)eBJ}iXI-rNlnZJ4 zeA^8N@*O6sleAvB{!KG(w-(sZRz-(BhREU(7@a{tKCB*%Yn_OInaEP1I?#6(0qae z%`b;Q+kyPjhl1EMbaVMUyM7!6PAW-mqW2XHE;hdAy`P%r+qI)uV8DtKv;D<7+7 z!{QB)H5}A|sj@QSSu8U#>#IyizQ}|$TQwH`o=^D73JaM+lW#g5ReqNR0xNtt9FV5G zDCx?m@{1k1U-4$bFZ*bM!l6du5lV%sh(~AyR6{%hh3;(*Kw!qB2o&bsL?BQIQG{xs zqnii>I#3jWLQf)NjA=}!%6F%z%2!W3;)^IA@wH@TeA9{|)3vT)DJuDj*&z#)!kVXr zC5L=z&8M{FnD10kp)ap^#1~l2lrOQmDPLsuW4_Gl$9$pHkNHxox#EkhF6PUvCguyS zF6K+FCgzK-F6PTFX3`g4O`$Knxc@Nm*39`5tc&>~yhE?r$Mu9aM}G56 z{`Ow)z8aM53qi@$z}0$sQSyuR)IxVqGV?^@mF!chWS>nX`$Q_)=TON$ ze@gbL+f8%C-eOKe{%YyuFvFO-l1%DO(DS)+Gh{pPc=<`^HI3mT_pV+xTfFpu;UiCN zEHuIaXO-^~EGmL2@_w`G+#z9RhrNP2#E_H34(ViSe{EA+#_l%?YRWIq9Wv$bKB&&L zE=mv5x+uwMU6kasE=qD*7bV%(g})Id&)>UAZ!U?nW%lzWkwT#&N+L7ONxdX8gG^OP zWC~bDC6PtMNwFmQRFy=ZsFLW@R1$rXN}^9uN%RT&JMB3i)=$~(0@+f{m3Oe$G!`vg z=w(0TWW$g7g>NT$1v{hQ{JwD5tHA~A$!k%brK|TTUt!PFnn3OZnU@fl$YHTA5AQPm zAM3Q@zgO-2_Xc(`1Nr!fYq}ecX|c((sI=IGm=>E5DK^f*g_wpd z#58OnP8YWPMC-PP%8hp)IdmU9bRR!-p91JKDY5|FCjz?91$3Vd=sqLReNv$Nyzo_H zw&3y|Ew-#b)$42_0lA2lCng*9F;==TMtsboz`H3V?_o`6FDL(()hu%p!yY~|d8#xZ z`>M=R6{0WIlGE2|N%Rv|5`Bf1L|>jI(br~4^fO)({hXIXKihm?^Z{LxTw=tzk*4FD zN!jvzCn<^P+=zHOH$qJ3Mu_R$2r-=-A*OR9#B^?in9hw5{oH(7uV;R2Z__33)WULu zvV=#H;9HEA{^k?Mer~Fd0mhSVvY|OtpB=Gt&g^;EF2t#l(_%VRLQJPhi0M=bk*T8l zm3su-jHf~2#4%0)C1*N+!kNyW5Yzb+Vmg09Oy^ICe*V&2$Rww^5Mr7OA*Q(yVwwvf zrnwMenhPPOxe#KS3n8Yt5Mr7OA*Q(y;&i!?Q{?-4Yza!b&j|cJDbRghp!?K7_t}B& z69nDo2)a)bbe}2cK3ULvP|RIVy>Mf=Z&F9d6t_aApUOV&9cT_`kH*{HN}v2kq%o8#rX7CU93HCBF&d6@{}a znii;7?iUrexO<wQt-JDybUjr8&RBrw&Z^L8E%->9J!b`yi3~;YKMXM`$r~b#CJkD693zJds)7nCufF zNZgPwQ1t>>obwzD#7f5fE@d=JHKE>pcq{uHW zkqrVAlJ`zX-nfva*S!MWJDjgBQ!s_D%M;WI0th{H(!r z26jbo|AG7sVQ(q(YT5I3sZ)<-WE3O|wd2!${#rk4haj<@OQY+h1&SCn z(p(*zD36hYeY6kOR6?Rf)>V?C0evsJCI*SC@4A@JAn71hDI8+NSs~#gjtB|+7|B#z zs(TFGBulmuKyoAYG9i#5BJ6D@cAek5E&ucC6QPO zZ07yTKXAS;;&`tX{|EF_QVjfEalFy7vE`W@DYtOYMong=-eSU#{tV*JPxuzthw*o|GdMiMP>8$`E zrdRz!^fPugF4`4i%Uj4~5UC68f-!z;flYLPBpB|YPj;!Ap4nZQuavNVB@xI0*-dnu-j;fa?1jMJ|#@Zg+C4})& zHu*q&l9R+n?@gff@&mak1;5?-!igv;8((iSyGbR$C@BjfmCURXFUiKr3q$wOk|o{% z8fu(b)(GQat=gSR`uJqunNlMV_tt~MjgiTIIX2k`W(DHYaO2D#HG(ic$}c1V@k#1ollLajdijCe%xeTEqU20#WP!#UC1+YA zUXp&{BZcmxMgB?Z0X4$-1xl}KM7HIjMj$rz;5_+rvti3FWdfr zBO;{)f-g&?ydU9%h(~-FxQmg@FJ!@jIq5+_EK50T_BI2pmo?;OULrUVC1+Y9ypkCu zXIdg&l6K)Eh3=yzr(`R<-6v-nhO2b?<_EG&X!vJEf6_^Key|7 zGai9Lhc0h`$px7Rd1vto_0mPqi?ooYml{I~+fpt-6iI1JOs8q)rQtmwZ~5|CW1#Uy9GlKS<4T6@WuUcnSIO@n1RDEBp6dr1Z!qHI|Hs{V2S!;n zeS0a=kwf2_U@%=}jV_Ac90Gh9(hEst~H7M^Qlqi6DZ4 z7!(yj;oI!FCW(`rXLkcW@B4oFgX=zXW_~kg&e?l6$!z49681e<)KOFs0RVADO9^~>kaU2k%eOKFMw*VnoyE45U#mMd_1#BHZmK18JViPxsF4r<>h8IXx_xe=?w{LeYoPqq6W8xG zl|eVyW%Arz%OCGb*~vC3 zJK08eGGR;Gq__hbtL~<;NS!D3sQi5`SFBvh{lUb_D5>>r zbnA_yZoaYV_8V);4QQh`z&P6WAlY#a((QKbaN?iXlarfZ(+y9srE+wGq}kG~1V3q) zbRof}+mK+>O-Qim79`ko0}^by0|_?U`sHDr;1kB5)tHg^ca7T0 zqHj2f6Z%n*q$T?#Taq-R_a$jYHb?F|iD|?j?M@02i)=t*h&;yRK9o2i2Dud_X^BN1 zmWd(yv6&R2H!*2O?`6`A-q6G}`XQSX;^~2#?1OrGcawimPw#KCFulXcrt}^sSJb;Ag>uPVav5 zFung(xDzjf^4UX%a!Rqm*ME#gaZoIU(7{ zD5bgb%p`j1TT-v}WLE;&2K4k(5@y*IIo+63?uvWuDR;$f*1O_$ttt1-t(S7&+-ALR z@?EeTV)suT^qtFX)b}j6Q8rRuGPsSh8}el2HcCt7M7xdBIe89n8+G5@M%^{HQTNPk z)ctT9bvN8b-HU6yDeDe&y~vVF`NlT!^MX3|wu+Tc1!@8HsX#3wB}hKk%M|T$kk21} zlJu!Yxs-3A6RQ-1yp*Ee@V+Uo!R_k_w0 zC9Tl)ixz!6-9Js&XVY!e2h(lT+u%0pBk4Bk^XN9}!{|2ZeRLc3PP&bH56k=?KZEjG zJZXpif4fV@@tV_*UudQB?qqTK1J{K2Psz1|u+)C#A#+6&r8~=|JZ+@%f@3${@+6Xq z%fIm)E{_{|RqcLaPx-iUy#z@4xN#X%K5ksbl#d&iG3DdNWz>(Glq*crxs%)UKc1bF zR_M}9xk8sQotpAFI((-8?~={D-!aU1p4xQ%*W z+(x}AZlm52w^470{6OLV#CAxr_xP-ov>mR;XUgqx8B=bD%b0RIT*j2!;WDP&4wo_I zcDRgsJL>Q%w0q*K(G;H{6PGr)Bs?q0b(ch2%3s9Ge?NJT&6C&5?ptfBtJiBP#E+RO zK-Vv!kHnR~|l4mQ~*+4Q7DVv!%MC8_CcOp4KMOp4Ksl#a>LT_16iHOn$wCS9G&q?5Z$ zp0&G(wsg<+_B*+h-niZ@<&bOXQ^I@eQr6H#wUe+ttzh8E**prj(m6R5@sP=RNhr9tC!`tzQ>hIS%vFy znPNpQ+k}(}JEcsR;`6+cP%6L2Q;|~6^*Gg)dH?8>-+SGs>fYqs?0~7fA4*&{;F9o) zMXt9d+ETt&lb=({1JK<^xsU&&e7Qr*GnLy@-;dqK!Mry7$E#ocTV%KAJ-+(&>V@uh zs`&)EehDP6uP5=*9+ycjXC#{1$z^z=sij=XTUcFz`XMFhjwilA4orMKFNz79BiAC8 z+RMo=(TGbdy^BuQgyLEz7THQJ<-ND5JX|j$xSpRk{bMsKBIP#fd^*x~J9NK9l%14d zvFrG&WX?$DOpY~4Ga`RT^E?w?}J!(y-6mMcnm}l}o8Nqw5cb z7F#AGsWYSN7v#lKe77cYQd&Akce#`*UH8(ly1oRPtul%8e($A-q)69IQm0Jydw~aNeYQPcxt0ZFg<|DA#!tV=X4#pVM~D zwVIz!ObwaJk?=Wl*OWPIVr|N77G~$C6HA-qlIBxWVv^>#50(2s*b~#UHfymHJ5OvFukOvT*pvMVNB&n_zu|> zWsad(UuHIE!t5~DccNxm%<`Tjcglq6`RT+Y&2yhP_l#~r(u{6{ELX~RogcZBvR!9U zE~S{%&!T9_iqbi{$)%L&I(OrBaS1kCSrX^{S(GA@BE3C}x}DzV)RA(3(<>FbUaiE> z;|OGPy?ra;fFwMficQ}Al$ANxkrR_FHDQ-sQLY)CIBCX~S~OhyuY+AC&r#h{Ms@QP z<$m_jaj8Bf$tQ`b@n4C%ehDU*{S!@H}8ZBpJLr2Up`T*rCdInXlf;wy%J5WL}oPBK26sMSg?yCs_1 ziR{EAlU&MbvN(o2UR^XMQ;`%eawJ88#NZBc**DSDMJ^>(!U{U1vVs&Zl|^2IImzbihTt4$pPZ`>7Opc_6u-J>*gzNcZxXk~1Pr`uF?gTgIf*(pq-r>BQ4A)OA`O zT_$5WBCaLKWzuOeuKKi;M@sxt`6){YV#>qnuqJE_n?t!<4`w_HPK7h!D{wJf3N`LE zzt)qw+hHYSP*blx4CB0DU`4xj7 zd5Wlf6^OUx>tfNjtxxhu9l~Yk#PP0s`JaCF@-BAq6MiQC@JDzZ{s#YsqBjD)32>T4 z-=^cBKB*&-*fYcIFfX*_ok{+AaIvKxo9;vCC)UA**o!j%2rLb4d6$xZC0uW*$EGWP z-Avab_NU2j%d^#z&q-V}7!2FMP&gWngLC0RxEL;lZ^QTD2T<4VDZkFAN^aN!%yKC@Lu!jc-<#&>xi?|f#aFh(|A74s{x)JYekHgT>Us;|S3&tV!oL~p zV2Q6nToY)E>rVXhuovtHheBOnD|EZS5ZD*mbjG1K8P2ij+jIw@uj>)}1m^X+Yt>-zfeW59lJG@Jxy!q=dVmpJ)Bt}SjM@o&L* zEqbCeoOxT^YUZQhHjAF<$hvHC`z`V1=vyFs8a@jfLR){mjSogg_qRIx_$0p`b;5SA z4;%}_q3pZXTgiMR+zR)>gRm(52!O>c{jurEZ@YDVvFm&%$gBOOuktQ|=?@x@r}>G6 z_TNe$yWkfBB~JT)kN;(O6<&kdzae>>!1k~!>B-=WqQ-TllT zfyXWWHb40>k=7TxuKNu7x^AiSmL4~-{vPj<<JT@O}&1t41a8GsP(4dKNHS@i{TRZ9@O#C*rUA6mj%Bi%HK(yT3`Hg z;a?EWfODX?dI9(shh<;|sPju*gQ;&A9BrxhQTkjGmV*P}AUFb!h7+OAuk|Y9SKZ>j zk98b_r{P(65z6_{`Kq*Wx=pp=boe@41h+yRzYqK8ZH-Q~c7~Ci47GnL{62&`;9j_h z_!wx5uYiAb_#WH{)2i4eIO(d&3cy`p;p%1^H9A*BmK$> zbHZX!=ex+dF2f(;Pw)o(6FNGZ{$zq$uLkxpa2k9KZiKN=$M?oQ2o8awr~Sn)c|9Pomv ze;m|$;@5!ru2AB%e?$+X7xKK}P?*Kb_(=TM!yRxJ+y`UeQ5Xw9hlP5Y{88{X=uls; zP}gD@2d_Zo?ikR^=njHI;3zl_&h2gdbo?CVzlT@gKo9<(;ur6w|Eu^dgv;Ox7zqQ( z8{{GX{nFDqQb!c^NFBC*=ymv0x2!|=dkgbB;ZYb1KZU2DqmS9QM`1}=8dis~@Fctp z9eqvwNT};+i(hy6Jlqbed5Evn&*)c$^JiB7fw zMlT<1FvyG>!(bQzzZ_!bwO$e8OTvn;V#R1`(YmhXTVo2{@KYh6gu&fyt+Q|`wM>u{m%gd=$rJ> z7C)N!$?yie3Dc?{PW%G61V%tz&x6!$J73|fCl`Im3k$(R@F>*!n~C2A55g??XNNX@ zZ}kqLtNlM_eK+Bop-#7pKmOIoFZ-_J*ZJr#c{&r<1NMPhzY}%!fc;=7du@vv2=0S^BaA(Il%Yd?Y5fD}=TQ2khW+RGb;L>hF~*nC$zq9@ zI9Z>>#StG%yy&R%9rqe9>$**S!6VH&!=TRh9De=bi*P6$0ms5H_;2gKMjlB)KO3x~icxC!cdej)xZm=V9musLk&V|-?dznp))ud)tZ-}^jX z>QLV^P{&7M*YVNB?}tZWEY$u->Gvn_OLzvx!3*#b{1a-u{G*+2lRqp6E5jCW6^xx= z{J(`-PyA|)H+~(Uj%yHR=5_qqF~&Y*tl<=x4V|Ur6+Iol1OI4v9DV}7fM3IN@H==J zmK$gCZ-tUy=L?~(L$XeI5`GQ+sM{ZgL+RrWjITpmf4t3ij=Z|wSFkUGd!em<9sl}0 z#@`|Se)Z!w`gIHb1@FLgoU@FU^X4f|ufGR6gJ2x|*`ogo_TOcGqSGyz9%h5{CmDMr z{Clz)yPcE$6N!HtFa6izzZq_U7hx264#LCmD|i;h!(^qnFGc5}m!0l$Ay^E$jmiEg zk9{f>UE6xaFDL8G0}H}v7!Q-Ba$idP9P%xI8S&2w)2bhV-g3AWZh#K;RhWFTe^U30 z^mhz=+0yU4*o(rFus7@n?^Ul9I;wm}FX|r#$H7T(8kF;_t`I;AaojyXoJ1MCi;gD=1-a5}sRMb}oZxB7k1OSV+^ zWijj}U@2G@mWNNmR?xN{U-f;}3nPD8`_+nl>JDS!Im`Y%FxTu~ZWszj!EtaBoCa4x z-H&^X{~X=3uyDAk?_pRH2EwP{vryO54*M1OI~+dG=*GdfUo&G{d=KJYfP>*na4i)5 zz*mi4%zVRu1>WfoRQeKs8691p)FW|HPZ&ChahCk|8vikQj=@+c{SGGYL`#2(hpnopW_b`cN~5S1BvVKFU8yP%6c9oudXB1Qjf2Cozb!Bm8FkQ!p5)} zOtuy7OX+jVH%!0%ml_5^?Z1Kio8fMFukmNm(fMMqABCU6(-!|JtYap86Rv@q;TEX# z?ZFY@_6aTKD-3S!oBbyJO=f8Z1EE;{@-A~ z058Go@OO9%{tX?hN9ULOLFAie{iX05Y4I9Q4M=d_$&;94)t{i`*HX!yaunso6w=Yvh#R&5C*^sunMdWRk-5<_9Z+H z-iDFz1WY_iD}n z9@?*Vx~DqBv(V;WkNFTU{U5;ZL6{pxz^%}hFRk&mxO^(FZN1_bfd6zTb?bUXCl0^w zV9qs8w`*>g5BkHxuoQe6)`9h41K0=#!#;2T90Z5L;V=wNgEQbf_&Qt!BjFCX8}5Y% z;UTE|am&Z`>v&)Lq2t?9m$&`V@xJ!!e#hTy|L-+^-dd-7Nvq*H_~|-hk6UjT^`4>I zIoUsZ@qY;ag0Lto0Uw9KupR6OyTYEZHw=eMVFX+O*TVOp)cZYk$HQB&6zeDtpM;fR zO;`tpz=3cC91WAT!hI?Ih@mf^!>^&I`u?})lXV?oJ-+Ub?w9CDpM2dPttWlHNPT{s z3vbV-*3&w^o?opekAo+ut30d@<=ppU{309$<($9I_yZUX<(&V`c~8eV%>m_H*JIoO zwuEx7$1$D=XG6UnxlepS{;yyR`Q^S*d4p4jy2m&0Z;gK^*d6wOFTjy-415_*gEQeA z7y*~VHE;vm2tR;Q?+mcj(8u^TML=aS#3YlfGo3uI%vtdOq8_ z@~}Q%_e&ot(J6vnF!c6(X+6<-3>{z3ht?Avsms^%tn~(Qj>f}@@J+ZC?tnU8{N7=H zdz91d()E2q?LPqfAUFn2f>U9xjj6|9$No2bV3YC70t>(jQ0LE%Jr8^ehHf=FFTr_m z#35sU4Q_zSPklU9oHK_UH=$M zyy)t>>)>A(Hh@iGE2!(yd6(h$4qOl4hx_5QT~4>u9XQOxzHPDWpRf6|Z8Ldu!~C!u z41~XLH-7pYMN-H8s>h#v0V;nm<6iJm8tbv?H$peYgT8x{lKmrbO^DkGcfmbyKTH~*?LH})Gk4;YSy3uDZ9HJs1=H^TCKFnq7^ zp9B}cNcaKN`3BzSd~cD*Z=chV@a1hjw{x<8L?>de({0U&UM83o=77mU-Iu@ab-H;I z-i9_Ew{x<88hz|^kD9|^_#Es9cR|@Fc|K1j<-Xj)IoJt5foCn}!{&F@;vaFu)cY=6 z1FxPic1NsXHrO%FjQ{-Bu*G@9W6+_#dZA;}Q|^vo_>Y0h;M;IJjE0}X(@^WZjC}^2 z4Hv+La4FRBI_@ZbI`274yy)t6tiXR2TnpcaTcEB-_wfn#r4rQsPV7;iI^9!S;LD$z z@pah!oEi6k?JfK47D@I`ee|2c!Egi&gG*rqEK41&$fxs<#qKXLJoEVcWxA$pQu z^3B3N555K$LtRfS_U~cQ`&3^HdWT`*Pt1A(pk7Z#?0zsm^tbp&o-}%UU<~~EGh^@j zh2dQICR_%W!}p-B=PUFa=w*aiU_MwFO1@DRJ#TfMLQmIY%UjgqFFL`@w}su|3$PCy z3`am&hxAX^lU7~XA5VJyRb68lkB1ZCG^q7t9TK-3zqRmvxWy7L_MOb{h9AR2@C4NP zWu4D+pXmh$tNTx3+j+q+1I!L{L7i{^DW}_X1bzYIq4sa_rSbpzYeTv3%;Nr2=De|2 zgAVo8hWL(f2#kUs!Q*MvUqk#l7!QAgsvO5D>=)sW@F$DE;~S@2;sKZ$`ol-yFerVI z=LemCJ^Qi|?uW6K{j&L;v-m%9!PNU0d;$)Kx}KF6jsG6#|GgR4fZAXDj^lq4o`c@{ zpSJinzGU*ZhQr~JAB_F-kA|^V472=X*dIF7m)4t&-dwl^?tuqjTJx2-Z0acm%fd>q zD(nOutV7nL%6If(-NWH17zVZf&1)uqfu9Zgz;^LwZ1ew{I0ydT#!tk5G7N`5!R+L< z<@Xe?>;3q;S;sLL4=ev->@{Er^fvx3I@*6dIy+!AJP1$2^Y9vshdF;Wb%nrq*x@&0 zkNwLq;%~#P(EpAZKMA|QsgAmigs(5*Tj|X>8WwuMjGu;X=Vbrr`oH$F{ucjB>n})M zMc^awDOedk4gKhQZK(U3tX1wy*{^jsobIvgR}lN8{Y6*ySN3x!{s-0mdfG49@0slP z0uTGI>)FJ*KZp99+5ETSzY89M+W%GhcmZC8VZS@wu5xbu@0xKbcoE9EeK4arw_0Cx zW}^2hTme@@|38er=mx4jwMu97C6Dx7=M$Yr(JKvW!e?PPtU(^xC&{ykeDXLfk}37| ziOxFoHozV50K5dR!Av*JdUHZKm%7i=2dPu)lKY0#^%eD{&ur?<1$BSCt@j)1)baDN zds?q8zB}vf1;@bUQ0IFO`);@op0fC#!~Xc6roV0Ji?{vBL)=5qABvu??*;US!ZC0% zoC@c_aJUF+y|n7c`m~hr4ghLC63EvI{GIa~ubz=QB8 z^i_WXc^1Oe@Hg~s!8HY^Xdi|n9V%jI-$<@ z1AZ}C|LM2b&y2UiJ-N+T$6vR^-^}&Tcn9%LXw%D+#l+R2o*<~})B17LDf+?Wndl*3 z0rCaH%CHu!2lvAmcoJ^+pf7sW@fW?ujCDVQvA2hvU{BZ&PJlZ89o4U`>_ZIH{vn(T z?Pu$+^m8HW)_PIsjV9k^c&~ajbC`3X^`xE|mU=#-p0vhG+}WJ}JkPP5w{Y^>`q7cP z(;6>v(oe}N=Uvt9>!h@mWPck z{`vEmeJKRXLMK%6j+d~nhf#S={5Gilzr*hm`~hBt@$fFx@mbOJgLz?5SR7V^I{ps! zC-Rv(op37D{$j6P!1%X-a|@dBYWN=1@oUlj>mlQR7v@B#19bYE`CvF2z3EW$+4T34 zXDjgsq4qzV#(G58Q#~#7n{{-AJzzihBAgFH)%rc@A7@?X;IB~nrPpigyWUT4_rcSC zZ)CqV!$a^W`~(IRHhl^&Y8VOk!&}h#Fy{j5^?94`$3m&+^Q2#qxGHb}wCU;kWSv7; z&m`#Ux*siO>aPaF0?hbT_&T)ZFHhd5q_5E1z6YrMq4$_Cn*71+tJaSqpI*;=&dr-} z8T=S(y$Ez{dQyizcR8s`;&xf;If(rm*cdj4(dfrQoBmt|Z?O;-E$ zZ|7@I-y=(yeg6>Z{p*gOr}{)Elyhe5&sXUE2(Q7skD5B<9EqN;XBTxIf#;y))&7Bx z8C@sr2K^s5cJ1F2`!cu#rYn`Yzu424G4pw0`*LO+0w+M7PvYaBF#Z8)jQ=K$@nu=p zYs4>tt1NoDeu-~@&d9Q69aEw9Z|-G2(Fvw+(od~-0li=0Em*9ysUwhn`caQ3y;tyi z7w%ATI)6s$%L9wR;!ykB`YikB>AcK9_f5DNZnNZjyu9f{mkNfVFw2u>TndK5F!=p>QIc3O|OrpJGpI{wb<&(wFxv{oaFpA3Oq&!yn+)il$%JVERgCoEheV z1z>SF9qxsCz24^EL0(UKE%0jvM?)RI)l$z`{O3W*XY>CAos;lu_znCM{#DuZD_s@C z{IC#w6g~l~zzFyi)cx}|{}J+g((8_24>%L*_*hFlv+;iiN>x#V{OA`byBZK>n%oaDLepAz>_eo`D#=*b<~FSU}M+} zhQcfKQ`V#RV=;Yx8%DxUU_A7zXYx*m5e>}v7R=nxjQhf(P0U#5i&pygs<;0>>&<0- z3*Z*muKs^sZ}T8CAJ)k50@Ul*`Z=F5{`X6N7JBpHLii?p3+{)qUsAuW_seHZeV5=3 zcpGZ}hw2*tA}|2H4!1%%Pu|AQBu@K_-CO;V?5CW&C=cgY=j%v(f4GVG58-wn^&0ay zX$HH&?Uu*Sx7aVkKVinErteu{W2o!@u9@-s0cL1!#!tiAus+oBEn67>4#9?f;0Cw} zegLCkm6pa|>)k-_PnZ|~Ca^hd3w3-?bo;}pa29+O&WDjOOPkc|`GEP)U_dM5_h9S) z?7toVk9>@;fqpI6$U{Dx{|4f8e;#F>kHZSE1#AmD!tU^S=xA@&Jq&7nZ*@XNPYGmc zYxG}&Bj6aQ+#PGMzYq7qWAIa$w_WP{HJf$33OB*+@DS|Z!Nf0u!5z&w6jthVZ~EuR za~^&VJ?Y!xZWFKjwG_K;o!Y-1>v|S;f$ch*^>%{cFrd4!2SP`P8E1pO>iep-$Eda5AXx}NT0*3lG-POw_1l5^PfR?)B3a6R;-XN%iQyw2Z&dN08%Fhf_f9_`;4 zdpp=04uPX#G}Q4W*yrEjEtugslP?P_3nP0Ndn~*OGd*wY*eE?Gx!;)X$~!?uo7mi5^5>(Q_Juo3iC-=@pfi+nmOczLb%wxPgUq-fECL^f#i0|fhgyFH`tQOu@CzvU;$JlRZo<1T)Pw$D z>g|SqxCj3c_>Y2P;agDGFS;M9dL)nbe-6LVa6Fs^wSNZkR)@8q6Fv(Y!a&w1eb;)K z@XHEwz`U>kd=%>VnxSSN>cAdw1&oH5p^oo~J=4`0Z(ouQb%;3nP9C zoNdw5^-FwnbV5cLy%(YOZ|7w`(Fvq)(hsfoDSBVQZ{T@&5&i&wf_Yvtb^6oqo6u9f z;jC*mjDQlS>$CM$_RG`x7{U4{z*k_4F{b`@up@jPmKtZ~gWy`I>s#|L>D8i+&@|SQ zb(E<;WUS$fP}iRuKb=qFJgt8Y@vp%*;8GX?SHMVk42H8l$*1cZPTjL%1eE>J{?}!n z#+!9N2K!7j<0x2TvKj068fo;e>!bcVeA~8t?zW3q~55kW_%QC|FP`TTDWM6Sx*FvhH+5G z?<6i3o`mP2*3t17@ymgJL)b*;ooedR@j5O5zor&{(H$l0!B66}exqq7?`XILMnmoY z<8VQ2VE!W9G-hDKN(? z#-3@u;e$}e=fgh$mV*_c_80xkUh0c~IC=}NfI8m+^4x;y=9=}?gQH*=)bVd& zUp~*oom^me_cg=%uNw}8w)iID#;-Z-4F|!r>aVB1e@p*K>PxGBFAx2B8vVMe9}$do zKUZUqQ2Wu9{n!O%e>~-nK`-O0W`D90pB8mZzDs)!ED7X#o zgnB)m^w#^RC+9}?MfSnA@49YXpFU@v{AJyGA9WuNTKe%R_BeO}esA&r8GC6SzvW>? z%i~%6|0eD(OppHqFsmir+qegb_k;cxy^+|*z%V$?;xG1P%)bLyTKwO@?ybMgZ=*%O z!5ilOIRtiCY{q?|K2PE|0{^jaCk!KQz9l{yzeDg7_&0ojxU5Pi(xNARxtPxb8^RG# z>Jpu8*!B8iu%CdR!;9kors+?mrG`4bDf2C1Cl~?;z~wM%iP5#CQ}iuk?+b%g znDIEM`xSxT26zY_ho|6|@DjWNwVtsdJ^JUR#_K|2I|{-Lmf*E{C>$ zX}!Yq)oJPPMC!~)zRXbaOFq%j{h31i#kWm;KfF*=!js6Pqy$dD3JTAE4H2 z{D85vfSuq_sQt%c_s?u*o`p@IUshwU4|V)o#7Dv!nT$s*=!98B4@y3*Cpy`^)Unm; zZ>d-EN?bds3;)J&pe24f{_|nQEM^5&V09>cYlY4W7Cq6qjh@z#dbEzMzHiBM0bYi; zE%}49nfgz{ICvReh1a0ouZOZ5OKIqY``~F92QR@Npw|1JjxUhIRP-<`0Uv{v;IA+~ zm+|l5XLu53%dJ;t#?E_;|BCoCFiz>|tXXmz%WKpb38n5X)cZUPRqN1t6VaOrXTWt2 zYELshj6ImVI-ibOPTy9+b#MdR3U|Pd;2yXS9)vOQ7`)efI^Nehb$o_A#*zut97T&AAj`y`L9lw&g_P`kUxuxFK*tfv#a3|aiMNjAR6esx)qod=$#Qs=5 zGZP4NKBPU(xGM9V@|*DpX!Do(Q}|^>*XI8Le%|`G^*42Ng3rP6P_Lp_L1URz#BeU0 z52N8;=pSJGoR1j#KWcayno-ifq3F+n^WZ}G7L0`T3YpmEFiFW=iC!4~_p0|EIve4K zFcbN+!JIH3EC_XdHob4ry$FAVI^O1&)_Od}J;FM4-^W ze>ggCz?JYbcp9FCFBCQ%?gMqdH{o~4OMh?kev96B@ImtDf%#zz*cM8i{Tb_ee?{+4 z_!rE8|GnyaTX%i(1;Hjz*3p4+4=DREm+?HP*Y_Cq@-Prqh4(w3Z5^^NdL6PadL4_Y zF9NQCdVQkTm3h&7k?}A%6-u8LGS+&wya(_<20yjLe~tYnyan%C{B3?kcz)3JiTyF= zOTjW0f16)pi@(@y&tJM8@o$4(JJbJ!ZHaEHw=t@Y?Qse2ds_rXK(82l7E)R#?f3H}jqB~;-Kn_pV%(Q&_$_jh;~ zX5{ryRw%EJbbVS+{Bsbe{iRRd=GF06sZYoMiCz24>$y_Ao+}4i!Zxry>s4y%+Hx0!LWlZGLI3 zN5>r??@{;_yltsR$DhIPoW)=IV#}lZwF@2Xf5M{g_pnhY2us2?umkK0FT%@E=dXyp zk(d77=6w#mp70eo555LZz|W!7d4;jAcPREza4ehz?^WO1x(|>q1|EU3j?;|KL)nL1 z+_&;Ty}niWzXzk>7I?q&+14TZqSqn&qSsM~^^}06E$b7#v#dw-t}*@%W+GntRKTKV z%Ud1)+VB}mygZK#WqueOZSlAHEwuQHeGl^oV2lU#^w##eW1GYl*k{rL`Vi z_j>Z`*FExjO0UNje*ztO-E`5i9`TcW;%Cbzujjtyb>c;M6<&w=dA;Wk3&V$Dari83 z2=)4YQ2Ku{&WwLH=m$ON-LE=Zk*^Ev4qt%%pk9ye%TxH(gNZ~^ZgBE|OBY<^C z9if(bO5pc2bi(d%4SWxZ&TNZbZ~P{~sc;6I4Ohw-ohXZ*_~|<0h?ltImiVjKufyNq zZJ7RX)A2$u5biIncfyRHW4;&c2ZzB?a4eh%H^E%!R=4PH$L}M!7v?Qx@=M&amiUwS ze+AFNzo1@~%|FiKFLl(R9;u_3rJiT->jb;QQE)$ufui%WMQ;Lr)8Q<*8%o?NOT75$ zytjy#INSd7^3EYoHylGp1sYosR)Mu(5DbCOLk;n3!9?zK=Au8;{sHNYr8q1DD?sgE z`vGIA3+uxmsQp`FZwEu*OBR2bf1dgNaFE4+FZTWLFg#}QzlQxfyaDrN(v>sDo&n1H zrgB;QKgw+4GG;Mc3_pfCOCRk0;ZQij;{PZ1bXkp3CYTLseX-|Zz5pz2@vn)!HmnEb z{ZZ{0zX_vw-;%sfsUYt=3V@|x1PtW;G=4cvr4?WgsO#@ap5AZ(90hIuAzu3LllNUw zU%aYM>pSrCcD?d`B+>P>-WSjx4uh%3+jxl^Py8e}73%fa{9d>CAE2*?VJtiiwfN zvgG>|`n<=fACbiM=Rz1Q`$M@Qaw*@v-BU+=OlehE6;eAJWt z60g^%^FM^X?oUhX+J7wljDoVxIQ;c_(eWeIdDf2yIj`QH!?fx*^YWZ){jsY40Cb#b z)U&N$>q@=49({k2`-wdM_2X2>%lls)yst1Te1i9bIpIai`@=rt@h|b>zsTaR^~KMo zFa8D5^W;B*`n~mE$omJf@cZ3Rejp;h=Zj~Y!}9x%$BC2QpO0pI6H43?OT75m^6LHC zOg%f`9w@(u)c*U)dmMfa<@Y*{022`eb-dW?@B^Y?c;Qj)X~x0m^nowJX>dN=0e8VY zQ1rKw$DzJF>3NFRb=)Oydh%t0vR--LlWo0z_}7G<{P&P=Kl~Ve0?)vk@GmHJN?#<8 zLw)IZPyXrny+uZt1!jlxdyBOR=>(~^i8pBZQXCgj7ED0;YdayCns<)}<4y=k_4HYNrmU^d^G)m3rTZpB9-s(>w z-wYTIea$z7IGdhce-G9%6pn*oQ2U4a=sy$x{qSRW0{WFR8A4%p_Pv5-|3z;;@eAQ{ z_%7T6b^SX2O%=bvL%igZILRk*I-jksX!7{Fe%;5D=wy1#WX=X3gt=is7**EzNnYu5 zXG_0r`MxCYS@;9|3H}XrKRoGutMq>JpeOkyPV!5f&Tp$z`sZtZY<+G(zdORNuqTw? zTgrW7E%%QoD8HZkkntY4AI3oWec1)ZA5}0bz6^hbf51CX9%l>L_hpv-x2>-T>j;2l zVIXuu-G5JdC6rz@4|HM}jWxssgkFV*wt^c0-JDhlL_h0AJ>l;nK!eBJ_ z)f4a(JOi)5Yw#xQQQjy&#CdrbR)*DKM;Hpjp`1fqUo3TIdC-;nhCGi+-+j%e{nx7eeJtzpHoyG7 z`2@P3!P8LZ7yD)AuflkXzr3FycWI-PA5MVrUe+JZIn?zOMW-Zu5;lU~`d7lgDy#|R z{R1{VPjNbbM$Vm_LpdL|_06G3$^|}?8o7k@LP-jE$qcP&!yp$a=ujpn_oBl zLtq~-p>uEnv`K2zs?r`$VgRj9i;c}?+YrTQ^ zg~Fjw^wSzY3Y~Fq2Gn{sKU;pWds=tm{Zl3M6u*i(SHt!2L%0p@flA(yR{ag=ZGw_d z>uWzx^^D^FC-*_Q57?eBe0|;?qMo-O<@wm#=WV^dwC0~j9r|^_z3S`u&v@N>8lHvc zV1_46#Kor!Z&o!7tY+w6n=uT5(eP{?GhfANSf{SxRA@#?|9V@`x&Kl8zB2SX z0{YWuTfZYzzmqy?uA-EtnJPF-J5>hBK+!;w(?%enlX z^Z7e<>vQQ>-9!e$w4PH>`G-|B9+Th<_-tik?*Kj3H?pSj-v#|@nQ;L8x9Z+m+b1GWD(AN}?IEucPGr`FHII-9`e zupJ!7{Z1Z7J)K5jI<)B(CcYSa0+xrhp{MxbD!z(`c*!Sml277vK3iQ4$>Zz#bssyS zGaH7(1#lr;28+^f$t!(cXX&>sUkG|V;UG8^PJ+50p7i=Cy)hp2B)`N-eu>lhZFNfj zeC>~|&yR5*D-Ekd>GNg!U4!R!d7jtx+VtO{FRS22_yODxebrl|^mcmCll&4V`6W*0 zx7C@S{uG4Lzb7sI`^?M!#ZkYv`nq1JL+@i9>I#E#@DZMmo`mvzB+nCCUv%{IpXgpe z_g8oedU}48I`s2nAg>eTbw)H}IrmlQtK5hGP<{5b{vcleHieyFHy8tB;TQ00cmXD^ zm)9rxEYJV4p3LfXw72VNiGCZ{0d|IZ{h`>0!ZC1y#XlB%#u`RR`un)$^?e=u>cVGX zL#XQ+gnbAc1{YZT!>~_+v*28dztd?lJ_DP;7SPNl{cAhF_v>89>-@ymuj+iuIgs<6 zRGGQ5)t9M?_B3O$Kg4`tSj^%t_6p2bgjFs69kB0$(R@E6>+`gp9>n#7!{IEbS7q}b ziT`Lg0ZxTBJx_5u{}~_GcZ~d}U>v*zwZ4Pd!b7J=Ho680+4Nqw>p`u?foC9Xd4 zw)Bi zI!o`9-Yj%px9CYe?Ptp`>x@Cq)AOS4?*`&@{7FlF+kEuT$oG5sVO`i7%J=nC88_$k zihN(M>pjWyUZr|w#t9F@FX2_F<85)D;U5oQe%3_B!<*2PUgu|wM;|xg0S_Oq}aV&`mg(4yS}kBhaqqv z)c&E^N5IkWZHxaM>=_#vr2tq7YJKs`&HPvx($IMJguP)nTmqxuS!mPGrt}Y5>JdMw zCmz3B@NX#f+3MMCsmGIk7UC+y<}j`Kw4Sf^#!F7 zX8pGKq?MX0@o!JQj?mlqx5#%ImLk3a)cIl?8%t;t!=~)p_t1fk^g-{N*82^;Lgy&e<*Qv^L^x$0fg@fTpI0pI$>qs*W#6AOt z!^;-`i5CCYu`h+oE%7$LRrqg*w)`bpn!4MzGAz~FuoB$b#*E{j872KYhWO1e8vX!( zgLh#j@<_fUC37YA@9}e5>Ug-Fjx^)$#CeK8(AM~W4Q+a3(TjqoE%j)<%+!|)`l=sC z9_f32RqrLnx(h!OcMJXv!`VO4k$OeP+j>R+SEZlVaz32cb^gaJ^&e9E7fl}NyI$`s z=EYC1L;IIyKPtfLuqJE__qI2Yr(yO^TFHz(>9w)wMRqjtyI@-NN1-21og+IKU9EqE zJb!t}Cwgz8o2Z9wIPqHlFm~Og_2_JcJ76E`ja0g#<88g7|Gv`ygt4wC19>HXe(Ek` zsedK@avnmcSNfv$pI~16^g6VEG0s&aIG{jCFinOMEc#y`Z;xebMVf z9l>2pUadC|-NhdCMeh}KMQ;qc)ri;nL$K>EEJY_0u7ytO)w-hNZM~xZw$k6jSl4q+ z<^P;@eGj$&IO>%B4wU{{{BPqgetI2R{}=i;qnnwDfaANH@yl>BjDowN*6T=oSJ)ql zzJs`bTfdl>^=7~?8}x$`uh(~g`NQxe3=1)y;c&!rI?{}rG2aq)hY}yh*w^@R#7X>Z z#(k*6rgs6qA7MO{c!&C$g5FFxA6D&Q9E0FFsKOmz;(rzv!0&#?_dz$-Lp{xyZx6dd ziP!ai{JgOog|YBccp6@YI{rHL-(Z2B#-lI{fKR<(?7?se)OrO~{G5A?e}nk1q0YCR zI*!4wV833*GYp>Yrz6c+{NsrG(Gnln*Z2p)k?6ewz17c5J=I`fAEPV!BQ5%S$aBa; zKAV0u@&uqS`9jds>sgFlcda2htzcW|kh-DhijKGSi+&TOKS1)Rf+ECD-iz=yRPK&u z^j*$JRBz)E1GWDQ{DYOB?3>O1E8^l{MvMM<>{5sBlh*%*eHk{u%!I*rq4s|Qd$5=O z9R?bO&Ts;p_@Y)e<1Nhp1pkDO;$I$CgLR-yZ|)$YJ7Tb57(4+_!p~tGY&<0O`l^Q- zdrcSudl09i7BPPueg!k&p9>a*0nnz`Zm7vn@g?ej+u><=9qM|v5WgKBfOlYe;&Q@E z9`p(@@2MWq(e>EkJE1G zjNh7SxE5xZZpMXR9k?4lIK#~Ae`l+DbUy8GtM^0d-lpom-}UPJm;OunF0nsr*caKK zHuPWi<9Eijr;H8H;WNy0ZV}z1;sS|K)r+$(I}Y z!|G7ie}r{hgt}kdiIaZGd6zitpN_ssKSlStMPD9Y(r@wm+v2bHL;P%yzaHEddX6%c z#*8xz9Iq=kWAPtITqyJ-UPtx8J{XRKGcEq#jz)p|f|Gg@>zZKZoiIf{AN`AvNA8Pq z9}MJi{UVh6qTB};a6gpCo$QmYr`lL!nGeJ1YmBA8LHMu3lK;4&u)E~ipPwu;VU$nl1I9t5-mp(-k zcS_}z`>gaaLOl=4eN^rnw&%SLGj(P&4kuqE+z6$f^Z(^~Z26n85C3*QuCm_z;iA&$5`T}k7p3P3}s(rA6nDriExW$ zf6|jj;v+ox>%3j??+%}bZ!6;+aG#g@8SqQgqmB-idYsswfem0Ii~n2bzXv~nQm?F2xYkGi7s%TOz6dA7 z4%~2-z*d<|rtMJj3ZGN{5gz<6lczLsqI=My?~h+0SQG|8sYB?*F6&syc%P5{>n;9A zuwQ}K;4c>cOFsH*-9L#hNPmjJ0877Y@fj`t`LTPeul*h;zKKQO=Kqw%KNkBpFdkOQ zY<9R8JPc*ug?}plC=dQRPc8hrS@doG^)3Dvuvg{@JP5|Yz^tb7FevK~N_2M)j23NrI(2>ooWMd&&(LWi9-&(i>o{;!|x;sQK1G<^v4tUzn1cc?*fsFCf z@ngJ%G#7BSW+X?c23hULNZ`b!^Mk^hFj zsUWzZ;c|EoKF57^u;uQ6+O`vogdH>otG_oqSGI}fp82AKwt8RZhrLY!j=~O zap=s0a&AP|rXzjO`z8M`Sn-_nFR!6I?&O>Xb6&^Mhj6I%MCS?g%E6lOarBqN!RSUn z(UpDF`gPE02Iamax}tLwJvkT8S?Uc#XEK}x*Fl}%=D*0|FFKo;-wxw^^gnFz7oAJY zUxWGN2R&*5Hvc;oebMnpuNbUj(YN`RwfKw9Gt4)J{eARrZ}AtMAQo?^Z#YzP~}5UBHcir4<4(;L11a10y|7eHU*wZG^rLT?FN4cEcl(ARkFFFO0s zI}Crc=-d2HS^P!kI`hB7Onl#w6+R4geV*dAzvz@i?+I7~)`Q)kukqTymeTFPxEJgT z<@+po-jwG#c^;DIQLW#b*I}PPd7WYV{!si!;lG^M4K{!An}z=Z{3TA;BYw+OJrbw= z51I2PW3w_%B5X1)w;13SR6FbuAQ>!AF9C491MfH%DU)?ZW!Z9a5|g?b^IN4#}_d( z(uaA)wWk@2e~_2{rSKE~Sn9ZJsi!jj)!@^xHq=%1qR!!PESzAef2fcC6Um#YsL3pS zIgU=nlEyCoPlm2f;!mUFYy4*7Bwq40Ca=!77ykn=1|EUhKMtMi@DF&)qJPatf9YE; z`j-bzgwnTWkD1D{lrlU8b-gydu4&Y(Nu0z>{$iH=jqz^=gJEl^>y^L1zZ1Q^@PI{s z8+LE~rH}K-U-$`A!3cO1{tRbwUtMguzlzQZ^tP2YdZH^jd6do?i{2`9*24|(BN&9f z(^g-`eFwdvY!-FM^hdl1&oDSu@%HJ24zuT?#M5iBm z1L0UG|F4wfS%B_7D7vzbT3>V~p*I!22IW2_c|`XY^yQq)x%tOzSZCwvAr zfI+Y|YzI5S&afBk3kSd#;V3v3PJk2PY&aLrgA3r>a1~qw_5O7#Z!FKlesB(qfCr(D zUygkhTn#tCP0$was9+Q`z^pI_)cI_F^8ez*5;q`FNB-wL!-?}0Uje^h*nvE@d^Pc_ z1M9*1P*+(7y{fP#tOK=wB_I83JZTi3fsNrn7zUR@ov+MO#!?Oj!YZ&ftPAVICa@(8 zfm(kz_EYdYylC+sUdbq&sBBoarePDP^(W&$3(kcRQ2xH}ddBa=t#B7S4nKuoz%%eF z{2gYZzHIP8m=BhKo{7Q2xGL55_~_a5x%{g|p%7a21S#`=Ru&wxxg4XZbsy z#jBVOmxsFF-=KRQ#zQ~+^TMK*cw3%2)paH_w)rotn!0}t@&?0Ba1a~?XTxx~04|2_ z!Hw`kxD%d$C!zd3&$EoL!{6Ya@Gi_l{SU!HQ2wrGAY=J^p4AxFfX!e_*cP^j{ox=u z6b^$kVK}t)^C0t|S2O)S3vK%)_5>-t1r{5-8^CUwk*i{VEw7XA)(zLVI$fM3FM@B*~O->PL4Gt@TB z2FpPw>soQoAJ?XdbQ2&?cpMlrXnC~g_NIqS! zLw(i3zaFd)gP=owEu^k@;X1ehDtE^UAN`~8xAjBo%f94de;Kng`@EO<$ zc7QhhCg`+*W8nli8R~d%?@usxy}mE74{Km%M!^|yE<6RlhF9Qqn5CgsHe=EMnLIb4{GH5XHo`(@J;ds_P$L%zfC7!Dw%HolrAz6#j3qvrecT5R!p|-H8=b@SXD>VeW1y}t z|5kH;3d2WXQ>gt%VRvpbaiMSu)c)h}p9m+xC@Aq^mU!`#eBx)zR~DTL@JU$7qQ4OP zQn(6kg*)IbsMlwU-)!+0ySMtWCMQ|f@zpsx2X?2A;rp6aVf+%vENd>;0K1E8*VH})8K1jbwZmr&n3@IzIv)|dLL z(&u_`q~#kn|AS^9O2bO<1*r8OB){zQmyExIzrZ%^cSk7aU>f5}yG`DjQ0kU*5yAOb z55I^>{k+sfPx8IXxB$9U_nWxc(4Rg?KD|C$y`md{uCCvf$5Z_}pXdjw{%qtM?+@W| zco|mPW7Zj=*6C?|I$u-r4ud1%bm&Ph7{9i#3+xV`heM#~hcUii`u))xr1EKf?dPeU zW$4Ai(;oUQe!1AE+^`_j^~9lj0bYXf@K2aEjq!hbh%bz804xEYwB+lA{Z;rHTw(Da zgZ&ldzr=(8bmnKNc-=qkC-vC$SE08?^7A;-k0;q@olov(a)0plaavB@m*jEl>3-3N z`Ub$4;7F+J+lRfc+j4&$MBHHb5*z__zSpt80hhqd7XJwB%i&tM z!Q#J_`gg#6@MEa!iDO;g!)%94AF9A$coPQjAAY4fV*D~fKj;r%hC1KFN?-h2So|gb zBI2AL@{5iwzv#6_uM-@i^nWFfC%v}#^@Af|oR{&kF5O>Qr|#eH^r^~G)BhT<7L0)J z!ti6pFAi!wnU7?ChX;Qrx^rL{{-Ps!ezfG1_+i9*@}Gy_0(c3gM?W*n3jLterPnL* zKcVBv|0Dbk!Ox)R$U1)`ug-Vmxaof^JPBWoHTKtGy3frxE6jh&jElk_p(p(wCrsQ0 zScw0a&y)Uk^kY5bTZ`^TO1~I+D#C>@nDq>T?Y=Peb%fpEn^3Q>t$$Dyo0&eCsLXBqTIvaT>V4ZaK4!uQ}NxEt#Aj>9f`ix_*-+v20%GW_51 zps(xGak3s;e(6&n{gQP}BCoA4=a~N<{s4c1e?ncK){n){liqb7^-kgcl?Q!YpN^CD z*z(KcXeIrNgqNW_Zgz6Mn{#geA9L>kURAlgZ*K^_hF+wEDn%fI0#YQ>krqLUB0}h$ zNbg1vL9j%Tjub&b6e&s(L4yqQa6CWMe&vXZU)B5cD`VtK z+~0cEKI`Hy*X<3i-#3`+B4fUiC+{mM(aCiydGday=XrvpW@12^qIN8X?-zWU=Yd0ulng#I`@ z1<%4OP}ldWw~Rb}p4JlI4!!DM_%GGpq# zo7desxS97q>;5CYgYd77&u};rT6OjQo=4vS_JF-%KPdjPk2i?ltKR_fhbVvTr*&TS zWiE1mpURxRf$QNm7z_78U0?j(pUxAVRo~iYDdsNMoxE@8 z{Z~XM*VzcJ(=o>T#W7xAUiX!p%97jJmBuPN8#~< zZk^|lqxK(5d;**fKZRez+DHHI{PF1K8u=%P>-xD5JKslOS@;C33hTgz@LAXrwu4<@ zZ`c!p-Qh481xLZra1xvb--ol{GPnwU1NHg+p7?%v5I$vGPpye}gMHwJ zV{Tpt;Wc;*=8tp7ABG!GxV7%*LHm6{ecj(_;^*M+@J}fH>;80J684i0W`wz*?6(B# zK>f5{*Ad@J_*aK@U^Cbf`s??&q1Sa98Fe}m?*eF%#5D?eWH)Cow2Pac>b7Bu{e z5U&cuU=2fGhj<&<4t6y367R|Jm*6Bg8_tDlJ3cFatf7D3@Lf%OH#`tv{wIchC2@cK zMYoo`b#N2>9&U%y=V{iL;Z>;T6{x??KZ&ou^+b1$ykDX2GtF^#edmG&psw$)-*sdE zDe?8Up6D`@mmTUpA1D3E;#!OTyZ)E!=nB>0*9!JPmb! z+CMM(;$PG7A4&W*_&U`71%Gz+2b^}??9oq2^krdrSO+$Q&%&0l4eSW}!@+Pk90e!9 z_u(SA6fTEf!SCT-cnY3{7vM#B3#Mk@xnN#c02YGfU{%-a4?(z--j#VI(P`4 zg12CD`pOTBz-q7-Yy&&O9#HQ`_NiZQa-Oul-#NGM_Pjp3z@9J?4ux;RK>e3cNAlO= zBliQVzq6`8$&>neKNs-529sTI`wNCypwyN7(BH=Od%yV|F1h+2!BucQ{0?r1ad6}B zE>G`I{JNlfAI^gQ`ZvAo{9C|Q@FmzEE~O8vAAjrpLVtQc5|@6w>RraiU%kvl=HG^W z^?<|SN5;H$emf(774c;Fh1)3j%e?_CNp8qLy=io)?yK?vIPvX@% z-UW{O!|C6Kx&QpX^OKSH1gxz5zG58%H^77N7}Wcbeg<=V41CwnTRN$){iRRY|7Lvm z!qZUCBcm}7?f1RmSCF~(gW`Xd^=~j4b*y^QpZEouA4{E|jJkSWA@oxUhQcjS>%)lG zhT-r7xCDLySHVF229h@vj)bqlH{c8?^}l6p)suex_4o3NrQRuc8fLkA_t)2>#4Exu zSPzE7Pp{oQe?G^5fgyi6-6Jr9ywUJ&X!-REP=ABE$*7kR>OLwE*Vj)c;%~qoVN3e3 z^pWV|JoJ{YH5Xk^^4C#E&r$1_(s!xr?mUFTvQX>8h}VF1V0S~G^oH|I4s*dgFa#EY zrC=O94O888J{e(t7z*3NZt!P#4*mvH{q1~ww;Yqfv@is&h3nvEcmPgHa`%5gqyznq zhT<1Ro}T9sbb7zP6W99-^|`uHa5LNmwSPt8Pr_R8X+!Tdzhdlrw6XuUnDaaEJvbfC zf}cX&?VO{}=G>bf5kijrmF6^7@k3i}Wei&6`I4GWuE$SHbO2?`JRZ zAK?*r0_yrYKRBs-Jv{_NU@xfkA)mYZMsJQ+;ol|rjQ{_>@U-#&w_l9mGWZSLuM4rZ z^xt!QJB)>UpylV4NBTcF#ZPD8uW}@b`=Kpd28$(i@ft7=<_wa=f9DXDf7L^u1$}my z8$J#z!piX9tNDCZei1`ombkzE;pm&f7EtEYfc^D^x<7yOTN(Ls9@bEIGnDfp=b;?u zBLYq~&d+g`ALXIfb*rGS25Z39Q19R0{Q5@zhr|~^d5-f2)?<^oiU(nMa<}dQi=}Yu z=b`TJ1C<}M!w8Zo+}+s zeVME5Gn)N=3x6~ApN2Y;f51bp>rO?#+VHpZa}50)`dJ4zz|Bz4b8Ucn?Yo2gi|`Ws z1L}HK{y{@8&!_g+U+Zq-Tfmr?rB7w(JEn3w>JIzB8Bp(6o`3vuYIme=8pk*on%1pb z!%6T1sQu*m!bJ=pt*?{LMTWqU@O2msC&Qm%s`M@|1j=&;e`kFS-hfFmIDI_4k1Km%&v7fQ@9h=Ej*ar{7Bd{+0O8>f_aQvIW zR?wQaS3T|T)xTGN{n+O~I0C)`bv>EO7aV_y^VlEC^Oj{F@yschIcdK)@RH$$E8 zm8bRM^S|0(3g+VP{B*za{L-`Re=N_DodC67&Pz>oUfQ#cf_tIdk9u>yv(C z|B>j#e;RA;|H*y&%_d*^wfy9~9%YUv;LmU;??-!}Jh$;E>z|-JZ+-#iaWmBOuiqK#GtL!HWdE)<|{0iVJ&)F1z-JjNZ)wlLNo_aE8x$bqnN$6(5NY2Y( z)0}UpJOMhP(|bU+cBb%U{2)KIe6zU$!b`7Om~1zZKiU+eF;UQ7CF1>3=nP}h@vOyl@Fd|sOhKY&YMsvORt3w(j! zR|iAwABpb>7zIbcG4LIz^S$!4UVLOff%YqTE9pnBukP$W5)OvL;42bm4pX51yg zt~+^u(fc_=ALn5_yaNA%nYcc3!~C!?d>9sok3d~d`v;**554^T^-FB@2_7O{K~<5^_P7=g^$1c?ts1v><#dO0s?)Np0%iKi&A#s28Q_#t^Y1b>FwuRuN*DFw^H3a~P)4(q}OP@ZSr zf^};c0XxDTun!y#UxBZ}aZsKwK9%(h_yJr5m%=aLYPb<@fj_`q@Cf_~o`gTc-{2K^ z6<&kM^SimFhUs7im<#5GA+R7U1s{cv!Se7aSPRyLPs0|lHH?7z{Q5#%WF?G;!3A9W zKfZHZ>;ER6l)My1ep+g#*rd*NYt9QxGP zDE2)WPJ@dIDj~~gL*EqroA7?~r{gmV&V{=QpXBNNh>xyc1OM8v zF02o=z6w?(B~=S916f9FdVjpISV`88^s+b!rCR=x+Qe0#P7w)FAYPX_Fl~;dqs= zvzp@596kqMF#O{d0JWALr>`_Mvx?p4YLwuBETsrJPc(Yw^+f6+&I4BkT%$L9Ji( zsEfQ%&he*;D4x(tw_bsMHT)XxfP3MO@CZB!&%r<8b$AOVFYD@LhIwFdSPDK0E5Mqt z5o`t9!1k~!91KUn32-u;4(Gt7@GJN|+yQsPz3>#g2yek4=A06yhPhxNSPoW&&puhM&U~a1Go6cf)<~FuVva!$05+ zn34Hqf!SeBSQ3_jW#MD64y+Fw!Eo3fc81+x5BLfk1K)!Btjc+wRo)$$3*|c1dO6;g z>vs?w4x`|BI0ssOSE(m{+t43|y1w{no#h{0$vI4bli&Dq zEnEj<;Q^@oUx5E2xEM-(FMsW)`}L|HPG3!7GuYDTw=ePG@Krd@(2uR+IvfwD!B}_( zmaC_C;?{EfH*{BE{;KYHC_D(;)o}5*pmzO>{J)`(Jn4pJP84U@?5;Jjip%b^dSY z|A04CzSife=^}YxepnD{z4TF$ekwzm>-<_SqakzC`PIsS za}H~uJm*{b8cN<=_*-44Pj0+kzQN}^xD_6RpD-t>Bm0&*Tk$^%_5P0Fa~{fh72hrN zDc7O+NZ;Bof0&CDf+b)*=#D0SxAYYZz4(N4yagN_puU5l7oRAOkAaH<)K50_;q)t_ zrB7q%#iuaGOTzjA>MI(0@oCKQrm!t+5Bo#CE3bU57oTDHje;K<{+2%4(2LKf9OwTI z?28LfzuwTFA};^@7XQqYPd=x~`{u*f9!LZ0Xi9 zaKQ6!U8|L2k{2A)Lbpo%ejNYs`YzT|`43Y5iJ9&v$qR1a)`8YPK%U;8^db9`K6HJV z(;PLY{>*EHF}E>IoYUJdrm0(RhEmtxxkk0sow>E{Zxr)w^o%?H9Q5jEK?|o}0(F1S zc%1j0oc~5`osZS8^!FNlz6&S9Eihxa^Vz}mG5tBW*57xdKLmeL`C8u}pFwaW9BugL zqK`)KIT!)|AW!BqpV!46sO#q?uPl5V)`XV6D91}c$*~`xY3c;J zUu$lY@tXnXK{?M-=T+t?bJhJeCr_W3LBxl`anRrVPUyQq@f+#kr}Lz~~^j znmU2b!pcJS`Ap`_`MlTb)vLcCJ|CupT3>{C zODLbupwM=`NXl-^@frc1?BVaLe}@1KiSXx zZ_(ZB{`7vn_wcjy`^h^5PeVO#op0sIezacV+Sltm=H>hqfyH1btPbzDUM76J>Q#6^ z^|E=?)BBPBBv1Bl_2abi(&AF20sYuC{Q zsQZ(+-rpGZsr845-<0F*KLynKA;k5(a<|n=xBi)U(FnIL0ZYMXI9b;1oGu=^RpR$( z;tSwfc+k*WI?F$lK8C|6e5J1BT{G%Qen;}X^dr%|3ircH@K5*`^ify(N}P=QN%D{Q zsb7TdOSl1wkL)vfdtJ$`-{|Og2F~v6)-fo9p48Jq<6Pe3QxCAaY=CzmjL3kXVHuQOV zxeklNFjx;ZgwMjZuvMhqnOlorL3IB5?Lxl~9)uU+Wmu28R(*f#b;nQdLgLboSG|7t z_^X$>$h>8r=kWOhK55KX=g0JRJJ<$e;Xe2yJO)p~>rn4Mj{3Tvvgo5=Gz_B8w6Gks z`j`EQU!eJusJjH}x_UoD=x0AH_L5Vyg)x1|=<65>7ennAdY}2?KM?=n9`%w_FSDxK zjdd^BA4Wm1eP!&YcjneoFD*VYC$DmfqTUHbW)xR1PGI0}x3pF^vE?H6c%cKkz)x_UoX&|QPKVDLb#bZd#f z!SN+MV)hYdy&yjlc98M4zc(I#>V}fw~{bUx(igc+lwI z(n)>oFMY~>rLR2nQw-{Pzhul?`yD3#EIbb{!7K1O^x>Ps@Xw4+_t$~=awz?l$LC4d z30nQje#I}){Ji*=HtOp6&qntVTmq90amrwr73PG+p}&6n@jV2O!=K@KcmqoPf*$px zUw{3*{Jx@Ix}k1oQ-?Wz1Gm6SQ14&#pDFz@LoYh<7oFuVedVCPv+xS^Q74^IFC1MI z+y)Oo$&2>Lf6-69t|PuOC)u~mRnJrMms8J6A2HlHw1@p+k`db3t>yR%bZb5I>p8v^ z#(L;4QTGZg@Up8r1V+KEQFrfekkXI#(97{H=wdze`#63Cp779TeZ}?p7>pj});~kN zi+bqV!`bj4T=FUzqjjcRuj2UEa3kC*c`8kOn~!t{o`(@|33bxHrV?5HMPEtKCxdBW zMwka`m-58D>L)Sk*Yn6cO`i`XkaMh<=H`$EySAe|2@L^aL>iyIt zUKciiO`xu?^A{6X2ET#(wUD)?|DNO9VJzGOEkCb3(*MC(ep-^m{SXJQz<-4Nkd%0c z(#IS6w~5b!bKpEfKO;bW@}#b@R4@Y!fl+W5tQ+L??V&H3TUUf^^B+u8`8hrGKcL?M zA4Xro@Gpn1Jgfwp!Isc^UXGPti@bj}!VgUX^p8Z}7Y>9n*B;FAO_`@E=x_c|BVW$j zVd|cNavtTpHD$i9z?H_llTt_WXM5-?psxfMqhDtD>povV*9NwS!|yY{yOF<<_-42r zo`g%1yAzZ&g<}+)45z_)utiFz8v$Q|r_!CcbtRP_ zjLy=(guWjvfWENdulw9dKbMU2Vd;M~^s)324==+%q2BfR0QK7UZ}RiePXSoO=+DYe zZRm3o_t#(R9w9&6@VE3841J2!Zi4yY85otu#bco_ty@0rUk&}X93 z{UmX{pOo~Q9%h1@v${Op?MQ9EQR$D9?+j z$v*4ChEVpo)!47U_2oGpdR~&BhJIx3y8pZ!F9VD+B z)McMz;k!^?Pm6e6wdM5`tzKVpUbX)-_;|g3hoBn{r4Q{VuXn8#!Nu zXD2BCzP{Xtq*|<)JaO;dLOxo7l5HK75i%fo5Kj$750L%?}5ht{p}|^_4Pa?|GLM1GLWAIT7Kdq z^U-;_zWDsF=2e)v`a6H!ubjsxn1ehYXcX(&@G6woTO6-D{rb}W;`=1|b)eVl{Uvn$ zp!DVCFZT_e^_xo>Dcxo>n-_l+pl(NOMl^5+amvpc6Pa2wnOgN^sIdCKpM zhyDQiL+~&3H=y3ntLUWfwCFOzEHD=gffZmSSOwOG^33e)*SQT>+js;xh37m>j~w#E`3<*{mOZhdU9UmxjEtF>GNdOdrDW8k-N8;pg!;a(UAPe6Gd&l%R2;UDlXcpaw5 z=jNUorh^$^E|>@AhXr71sGskn&+oVBX z{VkZ6F(0n@`Q&>cl_{1b*=*Ad^~hn;g)m;>s1R(_D7f1G$FSQS1CTfpa`u4m=f zG4$fohU4wvKsW>rhq|7X-_6kLIy2DCf*-)O_nE)Q$d`Rg;kd4=_a}XPi+&UQ$?&)I zyA8egoZ|Rdcnc=wK9CIR{;d4VMm_P#i(dg)*zlLQzAx2A-xxLtFu#(af7-QT^|TWa`Ob&sR} z72XIif4`v@-%}hv3w0fNpUkV?C-pr1^-GD5&d*HT-+Fb*IEVglFdPB3|JqO&8D7@$ z(I*_kq1Jzoeh1tQPr`HXS9l5j1#iK0k2>dIm=)%RCE#PQCaen^z$UOA>;Z?tDEKNI z2dBY#@C&#Ku7TgeUGOkG598tQ@DG@Rc?QD}SQr+AC15336Fv)Hgngi#pBl#bk@L6- z{T8?f>ht+J@i*ZVxC(v+H$a_l)p@GCR=TyNU-a1B^(U!&6(%j`bXj3OSQ?gvkHgBa z5o`jR!B(&r>;wD5q3|vEE}RIb!_VL{xEy{3x5Hg^LPeY*Nen2b3OeD z<#nL18ackcqC2t~%JcAISs#MGLCY_^g7Xo-yu9unFjc$lk_tD6rN zfO@|=uQ$4H4ZY-~lP zCQL&8U}*It`?KoFe(u#j8U3V&nP3)}14@0bdU~o8$e#u0z|o zKY@8qf-~Sx&`UoD-KU0rJ@IXD58P+ylT~#UQ^9PoG7N`R>*&thdij%%o8b3wJCx@U z{>56JBemrzm$x16grU`4{4rP=#=xqz-0_#-q}p!13c6L|cd47Mnu}$s>3Bd$-ZdjX zBY6d3ry6(nNk>28lgF4>5qv5@*_Zf=kL*+Sb-?JaK0du*UpNpZPQv}vJwUzo9Y+4k za3q`wKZFaQu5abPY3M&Ne3uaa9BzX<;U2gj9)`LffBjb&`MORVx-;+>m?X?qP7YH+ z7ft-`uivjmKe}#t>NGd%S^B38z4V=y;~8K{SO%7ZKJ{hg=Q8xIh_`|5;H%K5zASxD zLqC@Ic=#6l7%qZKpig~S`BM$O_ulq3lBh_`m*xZ8G7+K#POr>FL+b(xc~ap zmz953@|50(PY_H23&UctB-H6vekMa-fVjW@rO`Kl@?6KJteeADQ1>VCp&TCxN5ePa zyHMxLa~!?$CXzn|E`m$oDtN#CI$xdx>2E!)i=i)hpO^RbUx;6Xm!UlGF)8owx*vc2 zf5cDc|Do!nq>elfvN-EfP=D_5*DtftfBm}7p&jfDyF%?>>1h`k1c$&`Q0wavAE5NF zuznSehi}6fQ2RBg?;;Ce+lFo(1vkUsMmqD~^NKcbYtijC^r8zzKL;*=i{T2m3Vs8> zg|Sd~C2?yXTED)ri^RgcFb-;cXt;}1gVo_kcmSq*MrXRUrMr$kRTHNRe)jJ9)A0Wg zE`Xa<-TwFug2R-5gQj=)m*=5be$tg{UmwvT(Us-zowbt&ua$epfKN`-0dcUHN;`k^i`I9~Jb)M9dJY7$G zMJIj9JObUX2ikNf$5tKVq) zles;petj>xxdg$qFe7wF6Tden{vMRig{_!NFDUa$%yd8PAa5t!1NTF%|AY8n@CMA< zMk`qp4}xi7Mnk`V^Zg0@9Ik}A%5kpabMQC#psq_fuljW%_g(osA=jb5{oSNbt(SQE zR_;hp`p#EWvg1T1an8xj8!#=y-`UYB)wUF!XN?YB8TFTyS`3hMqXeK+)dJ@R*R z{0KY^FF{=|6ZfBoU=bJ!wLYA96DV_$`05Tzow&6em*-{{L08?-TRO`> ziaxf%GxQ^MC9j=PPx6+iP!Dw`d`}J@vWYB-_Iy?qv2Ti2Gm&}6aN%0hCdkkRm8u7 z>*05XKAL_f!0B)n)cu@dU%$%!dO1aMsP#pNmw;tq1w-G1_+S_fuSaTUx9+3#^?JMb zbFee~(9itFM*dvl=b=3RG6%jPunMdRr$MW}^e6s-=BLD8`l!RYAq%0`bG}7nl;? zVE71p432|Ved$m91I_wMOaAM!_jC-5@XCAZf3t&IG|#IM4X_!WYoumOA)E`(Nn=}-Iv&Ci0r^wENKYuFZ! zgnA0)iPuy*>09gNe9QAQpJ5()?p=uYgtE^V>R8vy*PPD{a2u3*Uite1)Cbd__P=V( zOP&K+g6q8u3^T5OiLc>!9LzS@O*S9QJ5*2FttGz(c~jtGsIw$~ul3~ljFLZ^{wBid z@W>Ed*{yZ{cq3n)M_2%#P}l&rg#F+!xC>hKr9bfxG`|Y|(#L$(i{KI{edsB?$nhRf z`ku#H>#gfyAaffE%MR>l;;}i zd0D!BiE@{^PArI$Lz&_}~qxED$vGwG-G%et~# ze}w*1m7jq;ulyg-WyU`rwCb(n_U^5UXB;XFBDq-B@lR&>2kO^X^>?rN1MwLXpr7nR=B4M8 zI0^TY%y|>@mO1*2`Mrs55_}&ngrC8sP}e)ac{&5n!;4VsrLW|iC+SP(s`bB8SM;Im zqlvMfboe%bm$?p7sks+3`dfu=J=_X+z};{k)cs1`1Jsu~a^AFl0J_0&7#s!1!1tie z|BSe<7iH{kozj;^R~}Y^RgL^X%yXTZtGteNe^Kbh!twB3I0;ULGvFMk{W22B4D-VZ zTFBbcm*jY97z!VQmY-K1>Hk3f*tuvD7itTK!+(VQ!0p#Jn{`fflCSj{IW9W!Z%@1< z>;iTEXyRkxTX44Mle!Ff&gN#;Cty^NJH85P|J*8niif@d`o?fR`h|x7=k&P(u7-Qz z0eA@N{;d3shJF`ufBoe-vX{yGQ+(Oqd6?eVpTGIn@zMK}^Y#MAyFfXQa^7}RKS?s@ zPyuSckt#pLL%$OJYFGw+S;JrVd6>SA!4vQ%)cyIJ|BI1di@xf@Mz8}6P3|g=guWDR zodf2Eh2efE&s|KC(&^Jc-T%|dU!K3{Z~idyUxvx3lhUZSnSG^3pA%g%Lw^d5-c}BmaG)&d0<*g^S@Aa0A>7w?f@tp#C~v*S$oYEAT4JoYrM#hq>Ur z`UM&Jvd>-A-3t%HIQR>^2&He`kL7oOJgwJtWS{xyw*V{(i^IyW8hi>qDE~(cKi!Aa zt&P4OjDX!>52*X|*Y6o4U+Q#Hb^5UG58r?j;3TN~@z-y-kuP=L=J-sw3a)|QKwaO; z|JcyWKBjX#E$>&EU{+WXmW7YQ%CH*L{rKyj$Hs$G;hF*M5 zbNn1kPJgMP%t_a?@~;^6WUi(0tqAp8n-hN?wuYTxFR1(R*T0UDFVC-+I(^Z-0i)r& za4MV)bw8Hha6>;y`F_Y+K6lA;?6(ltpSyH@%Wtt!zhtm;m=5K0vi19b=&PdtmCu!y zUUc&O?ZfCLPxmjno~j?o)B0ptT_hFE2J67KaD0f)bn8MKFA9sp$Dz*ZO#Bu28k}b6 zXA%Dl%JZMs8T#beT*ch*QeL-iozHPFf57}9tQF$o&7fN)eixrI_`Qao_=?X?{IcOw z#PE9*pNg;&d>TrhvGm&zU+G(XA2{3;gH8s_H~~9ar7UEZ^P^^6b0XbYjV4I1iqQ*yA1rth#$nKz4DpJ zdJ6mqhT*U4yh}Z)Bj-ielRDD3_%6as&GF+#;SVhpU(a=7IU4?&n_p(-?lbkK9Hd6^K`cPu{0r z2_s*8TXMV|l;^hVd5G=>bn@JJo!5=yz2Qjsp#1t6e!A`gbPM4kxDjrFKS14|zkaKX zeA)LHj*o-$pzcf8xAG?&^<*EXIDQdcGW4Q5&2f1Sy3R|U({+{xrh~d)fBmi*{YagZ zs!mCxo~6%Y=*1_LbK3-*Oy!f)XQsOyQ}vmCelmK*t2oq|PlCAXG%QI3~_O$`0-#Q%aJ z#hgz$SRF2htKoO>dw2^jDdF;SKH?Y#qf5E+Kbm@a|B~O0<1%;M|8S0vgqEN9$UJnOt}j0St@(8Kn6K_f z&dUP!c?MpB@_I^N(q+cLcqp$gIj`D(DL!7W--GCmKT#{GUOeY_9n!1?eK_zheS zH^S|3C)@+~L%kpE|0Ozq{k{Cw-lw0QgTMPcL*00I8Ty!ekTHL2Zl^3iH2^FB2>Cxj zfBj@1=gG775zKzF!aT4vd=#p7ed6~A{xTQsmn*>hbm*lYD__?MAy4-mM!W`;*Qwr* z>{ILOlV|BGar{ZB^J^RVI!|;~y<2?lOjFc#5&{dua!{TVyoB{yxDJN!{vn@3gNi%9 zoKW6J^Kic^1Vf>`kLu5TUY}F-{H*?4(r0Jb9oFUZvh*kQS8;#T{xXlsBaHB z(DD->nUBuX^~L9ZHLq8htH1Nt{mOZq${dcuf1cO;u*+x(<@FXuUH$sf{^C0m-}%t% z^}Z9`9w>cz`OE!6=O4ny-~0{ezk}Luzu~9zL}%5HV6O5!=nj)!uellyl&exEJQ`+6x@ z7Ru*dT`w8;XYs4+p|6O(GJFpG3sC!?RDE}+&)%>v91Nr2WH=4ZfFHq+;pcD#^s2Ax zjU(?(_zv{9{{7aQc%SvGIS#|u-??o-zZGtSJK!#;_bca3>dATe6#WvY&y!W}LHVzu z?i%QCz3^0 zi@dxlU+cBb%l~ISw`Y9RIhBUx;gj$wsO!mdm)|byj=TeRL9M?){35&rGndm!xBi{u ze?WQ8vCq(tdd%gmh3nvEcmUR|aQAv`A9wL#a0FZlb=E=Rhv6|8R9-9Hx(RikgB@UJ zI1mnoW8gBVT~`s`50AoA@I3q*TKQ`@z7Bq8=rdMy4iCWsum~&$%fQFs6R--b0qenr zFdVjktzj3~9rl9#;LC6%91X|ANpLEh0Y89?;1c*b{1R?}-@&c$2e=;|gh${pcn)5G z@$eG734Kqv`3Aw{Fc@Zm*%%L%$g{8od>*!i z3!qoN=(`wt?Jv5%hJGCJ_uzE+zM(%%{4|v3Qm3xsIuC(;q25m+;zi*j0qV=5uL`w) zGvaMvC)mZ%UnYJ9UWI?dTTq^FoTaL(m=o&y9f%KuW8hncemn7`PdcX*FfG*n^4#3I zPq`z7>N-}3S})JVeXp83G6jAN7ejeI?P}KR;CE1-gL{beF?b4Ifwy4BFjp@N%mMSl zQm_K71M9=auo>(Kd&8IEC^!bb3EzkF;Y#=w{06RrJK-Vt3%m#~!z(a3^UnzL!{V?i zl=Jhraem}HzKwnYoC$sE>sjJ0VOtmlN5MCtO7~fHffxPm2NH1|CHx=Zl#Z7 z@GQI`{Z)5)$zU3o4;F+);UlmbtO4u5hA;wlgk528I1;`F$H8bg4=#Y8!ewwH+yb}3 z-S9O01zv<#U@GR94hF-_uqZ46OT$oD1J;4{VM7=JJHoE8JB)&_LF;_3;`q{s28B6s!xMHuRqpUj@I085(G3x3=_K41bAl0Nf-*NLhUylzq7Cyeh=FJ5dO#D2|w#e|BoB}&%?*^ z*Yol6TZP|RxDkE_e}Gm!J>Of*GcEn6hrz~vQW5u8AB2B;m=$J+x}O(`N5DRCu%RDG z{0%q(PBiq5>Ax9_g#DrJCv~`sbb{SsB((H(IbI(&hD~5gsPm5y@B559G7!E7$3b~c z?>5$lVH}Kymj7ky+<=AAKMWr+^8L*#L;j=iF<2gY?N6SAto!dkeka&f`3+&M=Vkdx z{9flN&rx1O-EW}y#j@7t=L~VZzg|sTBxO^_v{0U7osV?{>I?pE5dlt5U;@5?>K0i^! z^?bJxU((DS`3WXzp_Ohe`V-`xhgqL>$8*3uP`mh>m!JFsP@c0L&szGuK)?N)>q>6@ z5XTF{hvEI^N8)p@_1lo&9(MNAPx`;t{JPY!^7Ry(GS}8H0(Lg$EpzEhol$Tc)czBQ zzYpg@IX_xog7Yuefn2ZpJXqInGx9sZt`Eqs6+SP*wovzP>2yEK$PaXX{`$|s|3mmO zl-H3pKbhCP=8vO}m9OV@i0kYuyZ|p7*WdT}?}dlqaj5-UwseuV;4~=DE#ATUPpI=> ze9lEGKkpa@`$4Uri2i%HANtUzgO3^c1zS1Co=~2bIphUh$*n~%&jt0;i|-KpGx@)U!Cuz7hjo&_*(O*(#B0_3XF)G%O3~i?;`R{q*ZQWXQ@?3JAH3!`SxEQX0 ztKc{ATet`M)YnzwR^LkQJJH@n;^7tO>!6ix9ZtLrYzIGpSK%Wa@1Aez9_r-u#b9Wc zyXUXQexnX{Z{nd>a(y9n-t@?6pVtYz-j`CX#&z4XbrPJ&@J zX!%PY2dQ%u{tVATxgPcYYvNxQHiXS#Ul`xbW!`|dVA}33o(YEbbn8eM1%rFJ_{{;1 zTLwD*HrVkxbgRVgf%f;H>rZC?S7Er?-)6PH#LnGMk{8?6tpnYE9_CyK7Kfp*9IOn( zU@NHm=|Wu2r_4+CCFe`_rSm1v>pJ+6>*34;x-KdY(Ve(;GW_K_EXle&42RFc=U^}R zDjWlKmEnd??{8iYm$?MSL%Hs!as7Yh=j*`QukL3y{cSS(mwioCy1j;8;(CAC*iT+q z92VjAJPBUrbzN|fo}^pL^>Cf~f5Y_n=`7Lzg6>x+`N_$%@^zlnlRRBdd_^a|vj0H$ zD|vcu&FT9^*b#Pzec>1VU58Rv?wc}a-M^fdD&$v#ji9A3!|}3E@@sqK>pZC^dAgqX zicb2Hc?7y&$@7|{ygudiIFQ%j5cm$);vb4d7Nb~+P^Y7Jx|HkdaaYW%l$D9zp&me^ISj2WV{ZF!9h^_ z%l%b+n&9(1jPTP}u0y%6$#wAv_ci^w#kybXepW`hj@Q5vFS&IYSP52#wcyjR3G4zx z`ndc!V^03|U)*?qt&4ws*c7&adj2zs$3dCbQS~`bo|CEbAL{EOd0_}F1hrnC?_7)H zbzuiXA5Od}Yz5mG`q^CfbKxgYo^PxBv97oO)_IrLfn2Y09aQoS3%wtS59jzu z_?n@Y_#}=`gEI~NTH@cp_3&X{A99_~=6aX?j^*|5wcjoH?SzNmWvKUS>5riQ*(1L^ zeTTuPVKb=f^&ma~M#1rhejf3Upv*Yr*&3++2_5! zZ|V8up)bqN(&s0y04!$g&(ir@Zv^j)Z^7A6o}aAyljkX~M7J9L1hu}_P#1X`Hi0dm z*2{Bur*eD=95O;H-CB-c6u)6E-Up6@6JSn!%6a%5Q+~zB*Ih`wF~?iMu71|jefKAC zG@J}G4tJGvK)nxZKH7g1`McqMcmc-4EATIP1EyncQeV$wsOo<%>%;IJd}c#A4^QG- z8-^R_NBaeupHJ#SsbkG=GJO<%+3j=?Tm>sex%gN(=@qx$3*9R5d(?gA*P~7Y*a}Mh z_o+MAs4x9#oz?HN^cxXi-$_R$?)hJTlDYkjUoidaImQt`3D3Y=FctfoGtzhGYZG*< z#P3q?XQN)C()}cJTo>im+u$MSRZs8h0(sY9diov@W$xP7nxFPh&b~9l9I!Ba7(N2! z`OM{5UxQY?c-4O{`j+QRw<5nAl=HBY^^fo<^rz63gXj`3#JI!ogI`b+)NuR5nN7y~!J&F~mJ1Le879gY6Q zKcDDf7pVJ}_#BQef?q+edU_w*$U6uxz)quG<(HuLwa$n3zeRpJ_MZvng9TtwSOS)U zWuVm8^N3ab|HZm2b&tXeP|ianzHh_#p!U~(f#wgU?psjmjxp-(As#%&by5#DgCpQL zxD0*?ePdl-Iv5R?L%qLyt#gEWIgL8I#_39KE%mafzGJ0-l_olW=b+Mk;(M?C%Kn+?<}}@ym(;Ig)Yto!^L!Zn8TbeEs+XF+vcqC<7F-Cm zk9EG&p%=f$SoeZi*>@o|hmTn=g=?VRulAEU`kTLuKE8ue$J$Rd`e*>3fvsU%*b#Px zJzy^=^V9o@=Xm6JcVrM83SWWI(8|xp@rtmmp|^a0LVpUL4={fr`nB)~^zxH^N&fl( z^C!RICK3-1yyezq-gev#k3c;?(P#2gpZiVcFa^e_Jndf`T~9yt7trNrAMyC-j@Hg@ z9f?l%A^VoTWZ&YWT`J?>7<%c0-f<4eVK%rBYJWN2hU1+)^z}LZ3~T{)eh=b(;b5ri zTRN-Xv-m_$a89#fER2Wg-*d+^!7MNYPKCOk;CEf5BrFa0K=Hps9i1=w4(KD{RQMs> z26g@=;v@0b`;)r&x<4;}uleYH{=e22<2;szRbd#c2^+)aunp`9`@-Qc3hMK7uRhlG zHlI2l!zFMz{0erSAv@pcMARqdrx(h2SV*@&0qT$W#8ps1=tXV!)EX~_yX(!rM{lWA=Q5n{Z@g$K_5PH z9%i!s2rhxzU;71`KZ5!bpwzMEcZ)v8&2XJW!;tsgx-cvXL*X>2`+sA)i@Xg#hr{vJ z`u!YF&+%NaI&1_7K%E~;d=ESXrGKq2Gt))N!e%hvEUk3wK^*@Ku7)SzMVQ0Lk0jm? z4uvs>e%WkScR4%>d(Uz4K``3~Zmqxfp zfTdw&sPjuw=TTS%hC!{b$i8F_(&s?uI|YI4I{?&hOvEgE&9+U=#Q(MD`Ej_~|6H z{AwBbR-I*rUg9e_z7`%g^h3FBp5{7f3a7&*a3zf6I=BkcC3hPe0^^}CrHjYIuvBhc z7dD1V;7S++gVQ>FvGk6w!Z;X`!NuFcC>R4b!&tZ*9)Rbeb4mOz^Vq~(0^Psf-!Rp` z?kBOb`$_7rqW&5f^Iz&m*PlY2_u+?dKKv9~{ptN{|E63Y?O`|AQ?478&^AE5oVP#d z^9J;B{mxE90d;+abDmUr-&XQ}hG*dg@l^@O1Jql-(tn`+N}k?#ZuTF`>nm$&*LW^i z7#4%2pvrHeIo17C&b#&(->Uf4hWh=WFYy6T`tb650eu_T0cyYD#P8Lw8~MEq z|G|d8&J&&0PaWpm0=9x}V0+jF9)$9KsQ2e@{xN(`z>DIm488o6+-LRO4gV-8`;hBO z-p_s9N9Fxm`&;*wa@=34!G^FkTnJz0el7PwulIjlf3Wh?pBJ>giRxn+>o4IN_zheS z55nW{6#NrjgI@JkptI`z!14W1*LzU@JMiBXU_IUUQKL^imt>shbTBi_4fDc6uqf2~ z)%E2(T%eDC&I9>g_3qXG2K55fG9`?`gUGfJ3YM`>8jDG%}xs%VO^1h(#eBze`zvM9W1M>fg`Tq=m zh4JtTwCd}A*YW$#PIv$wf?B^lK)uXW{`{vFbEpSn(zyz9zLzs!nX|5+g}gGbEG!ST zzGQ%U%UAjjv|q{7`)*HPA$*?c0K39|a1b00<tDjh>RYY@dB0o5p9{*Lb7(*7J}`#scp8j@bGe_&eJTWh{dFbB*Hb^Y^(-yg)2k!Se_>X$M=zl`X;_EiC0WmpwH1#7}* z;ImNo?^Q1!c?DoeSPpvCzu$VH_gPQR$(la>CFU?=+9j}GAIC28Q`cVoI`)b5mG0HMU&qo< zRj*g7UqokCoqP2!-L8Mv9vw<`?cghYd(|Z(;h&-_-C=OA1V6`qecX`&9sBg{+Pl}k zE!rd;>(jAE#O(xritN$PSGsGjt_k1!bsUuNy;H&>;aKku5&a^3r8{D~#33ExWJsp1^l_3i5`-M)9vo*jE7 z9PH4sUH{Gr$q~IecT9-8W!GMvddtrd?b`L}I6ziCy7uZQKPSvfe(cdZp_t3X=|2Yk zpMfA>5?{rnVgGR;$d}aj>|NtQzJ7O&C-Y6YYdpCx=C1J+zB9?feEKUW;X~tsTddUzVN%o)BBdBzMDRS@An-weflfNm+&Hpy<0rk zH#=J`El>DQW0`#658o}G*%v&s<6ZPwd^ztL&*}@gYdo8;*j?kvDae=SUsj1f2Kn+PF8<}GAYZ7v?~AMzcIQyrJKueG9B#*B ziO(TkI)S(2QMJSFM7~Yt4%8+dUf;##uQNVFT&_2nNKfMN=(FB#FyZU8UXz!2Tvrzl zM_+|_d{Y-+#eW}pLG*9A_$Ks2i7$EH#pP{i67d+~{PBTrG4WUx|4#gg*GO33{@P1C z^o6iHYx#R9=ZHtQcJZ0)D@DS8XT3AG2p4b2^-zL%R7V%@e#gz{dxm&C@d^0!A|9>g zGg17zIiG|0EF>P>-Nm=0bO&}3kL%;&GM_8N^} zj~VRrPrI_d5uzXB;^ooLB_1B-;Efjy^TqeHniN_JYknk39`|F(OkB8k}`ot$$!uh=2 zf86n~+hO@*AYWeMOMZ26`TH#uiAP^@@t3pS&hq&hC&cgEf6mstv#xvRyU+I`aUb`S zO2m5-4c%D97jA#uBK`xPD9YqUq#|E#Gk~c z9&z6nPQQ%np(XJuX~OO(`_NB!@mb;YCGi1;V|2Wp2R04eb^n}|JT{Kz{gcp zd!N3blmaQxPe8yal!pR!@@Uc)fzv!nOVc(sfff}f(`1sS&11++N{e_6kN!SD@QVty zK+%YZ<)H@kf^hYA072x!B|O9;mur-lf|Q_q2#Rvo00m^iScO{`r`WbQo>7( z>ucciJoq3yc>J6K%lRho|Am3^eSF6EJy6FBwR!gepAGy>(DVIvrvD7^*}$Izem-#X zi$4P20elYV|4z8rE$V+#uN;R{{9ydV7|!=w8BfOW<-ixjaK1mu^gi&J;NK6t6?g*p zR^Wp%d^_;r82>A%9rG^$9|9kX!>v!Kzbpm*UGUind@Asu-UDv+p2g?kCza0z(C-4D znNKNhd@Rl@4$z!>KUz zqk%Ucs(2IlF9QB8;9tSSuOWPI_Wa0E%E#=v2Ymh;cp3PcJ?{Z-_RROgc)afcw|HXq zTy?bakK6My;3t8<*>fIv+@1%apSV3=9>dL^hhn(d^Y$2S_B;$cZqEha3CL;oJQl;v zp1r5leoRiY=LGP$Jr{t-?RokyqjqTaoB$rTXYU!M-vBwyp1Xm^?RgM*+@60Dqc?lL z7V{Ff=K|>C_PjgBXDjSv7x-+3oZlq8lzmzp8v`HHr^T_~$8ekX@fdFN{!R=xzw>?- z)$@-b=f;zKk?;)2`5^GyfHwkv0{Gj&dw@R=yzOLPkTt+x1%5B^g~0y;{LjD-LcX3s z2Fd<4IjMZMgZ>!8OX;nGaE_m?&*``dpg(m2dfs1W{;lQ8e_*mU;C*(+$0`-Cg*+Dk zPu3{@KH}0g!nxnXe?|4VgYZ&v-ZlY#kMX~2u?X@s{bB zW6Fou)pF2+k;gUqTg7*S|IxrlUQ~SfR3)z>ockSoL;FoA?9~Hrjo~feM#(XaXM z4d7Guy3$+xxBTvdPXc@_9wuTqKMx^x_-Exa75cy1{zsL^^n5AdrToIyM}r%b&)t)?<7>d*?@@dh`n2`YSg+#tJoCSS zzNugF5zzk*xR+PFVwUnRJ4ogCw<>-Fa9gL0y{qyMg3swh&w4HZH#=ELc&Txn3;x;3 zb<*Ju=(87i*%ZZX{dfuRCg7%r?MAQaW3_q7} zwyW`V%D*sK8(c(qDLwyh(2rfN^rq(>zz45T-1IyGya3$R%MTczE0x~#|19v~s}=8> zqVl|9^xst6^nWPX5$ho_J*xi?6VBuE?^pVZV)}0dz3Kns;Nt-|{htARaHjG%eOjHL zn4!4UMYfI@++T64yR6P1+fQ+;i_V6ez--XG`VFyP)HN^f>D4|rj=;$|oHz>|k7z5)qp0Qlf>ihmLKEx^ZW z6@LNt|L=s0f0e0z%={ph($|6I(jh7VDCv#V#p#|IxwuV?vj7`WNnA3#3{J_*qR z?WdVMG2HCXo2C86?XdM=#p8BZK)+^(vky`Ik8II#?N9yjVRl#mAF~s)!$I_Gc6B=O zVZ9X&(tgbjmjfR=P;sMQ3)}~8{4>B4z!M^k=M4b&V)$0VOWE7D32@t2OzhI}njKEG z@AMt%cb1pEPUVL<-oFZ7ydTBaOUZK|(R<#(MfW$K6h6iGF`ol{{66U(p{Gsi*nP>V zhcb6AUeq7W-evv^{WPhO+_&H=s)_}CMQ(>AgkPXZtPyW&SqQTz|U{ZB3y!06ut zo>=4yZhDwS8wH}DKdJmPl#O=`@Sz)3{?$|U!fC)0oyyX9LiF>X^0^Oujs`xy&==$g;2#A(`Vpl!J_~@i{#^BTEa=YyKDJ!@{RZ%j z!2N0+*K1Ss!dBz|mf~kYZ`*;FjVQhuFBP9q&g#Utu`(K&Nti0(jHTDF2nfTYHuN_aM&( z;6vbZ4ETJ`_-xbhK7t9j4*2Mkirc*02D}V*V)#9TC&1JF_@(iIotRy{2E3_4gNIQc}Z}b zI3IrjeF5@Y96RK2=O*1AMu6o0L<4+3PfyI!{60;d`#JR9|rC>C_OEsg0S5@J`^~^D6&akbl7Ve@FR$4fIz5AFEdS-vYk@_z>*M^3->MzjB~2ln;ac z7r=*O{`lX7mwAVIKa14^FM)pKB9-6nlky`KMeSAm*PiG(hF6<%P{ZL zLEiw}>sI=cfS+UZ<*En1ug^z^;Vr6%4CZkVc=APG==$WZ=Y2_VoH!phfqwXh+V9=C zb}w*mgYx$=FaH6&>=j=qzp}43c-Htp|7U{!Eu&wf^3MhT1CCU^jnye`<2@eu$mg}+ zE8)OZz`Yxl{;Bnu(l2HZ<1 zeG|rYGVsJZsvonrM#JlUk%r7$j?V%w{7U&)os$PXggW2i&!xae`<4FtQ_l+pNn~UN^l6_e7pks#0jb&`yKp0jL$+}s7?Q~kA{ApQhCk=uakiL@UNpF zPp#oMD*vNsf1i)Fz`d0!=P3$%TY#6fD*ewf-md^JK%XYhw}B_WrF`tYB=-X!M4fy9 z_&;Oxh}(03zY07ts^i)U{C(g)`lW3FIgX^liS1+zb&J)9mB2?b%E#=d4tVlz#jQ?T z3%uz>wI7qS3%HNE#rSMCdi2`^dAruci4 zwBsS*t#A4QoTIS!ZQu#)Pu>aqZo^+t`V{b=10VmK+U>2tUjaUd`r$a>(~eO+B>q$N ze>?EQftTH?_*ql*LJjcj7nJ{`Ns6Bf+?%TO{{cQ*fY;6S1+j5mWBeB>ZuW3HaQ{<_ zh5ixf?IGZa#}%)TtDg63;7zA0pApFUN8p37L$h=5SRHQx_HXNpS-=y^mH&Q|wS!{> zM`zB*V$hF$M#uX-T>CWerX96{+yuNIc%eb@1sK=Wz=vP-g_6nxIlc~j{F};u5%gRD z?jKkubSM)`I?sxzNu!DyMSuypw@9y{-B= ze~L0%0zCVk;*G#lgmc{(f9|~-^kc|JPl5h&;04(KV~}&Z@rQqH0zLxVhd-MA>;gXI zt2|fZ!fxQ>sBg_4UIU)MKCSuJ-+>PvtNOVS`ahT)hx16;ef1&u%>B@gAc&!HBx>@-w zpQabufe#KU{mr1?EI3Y_k1GjJIOq2DI`A2*_XRS)z8!cO^n4Q<+z&jN)qZb)o?ijp zgnc}FuHYZQM>>_>?DmKgR6k><=(wJNKC1|yHTfWK`l#yPhb0h&VKZ_gM6XU*?{3j{?3!_I~Grc_zd~l80&yUdW zyT<1y%I7ZN`<|rwNj|LfR{!(!=j?aGShw0f*oQ%1s8IUJ;BzwYG5Ga9kpB$hAFJQa z0p5!Bi`hvAc+-9==gr`=#c=4&))AKi_pmOu`utkL%jh}k-Ld_s>p|a&b)xD24&a3s zed$qq_q_X!9`~&-fjqwk{>n_H2kiaU=%KecxbP?7BiP5wPtpt1PF6ii<^oX#_q9f4AQkcLHz1I>qMYbH)ejHQRUH4!nT4YH{gS;A6LGzent^ z^6xS}_b6`l@Y6^3Oa)^^kbU7sj2+-aA@woH!qqpl_}B1*O|b zax@tp(`OIxF|0ezo1*lW10RaTpBsVupHcl^3;zF3INMcxpYsvWXR%IP1Nz?p|JyOX zAk`Qz|9(XF86fAsn4g0F`lQmE{r>~_ShdPy`kXya`Ltr6;V|OM$0@*@;CFe2z0-k@ ztf>{`VH9L7hJRZ5Ukv&##x9**k4)?`a{X_S)a+v7YkzcFc

    dXgbYJ&#gLzgIj^Xv0~mtmb^ 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 K+TQ-g`~M$a`H%Vl diff --git a/contrib/seekable_format/zstd_seekable_compression_format.md b/contrib/seekable_format/zstd_seekable_compression_format.md index fd0593f9a..cc98150ce 100644 --- a/contrib/seekable_format/zstd_seekable_compression_format.md +++ b/contrib/seekable_format/zstd_seekable_compression_format.md @@ -42,13 +42,16 @@ The structure of the seek table frame is as follows: __`Skippable_Magic_Number`__ -Value : 0x184D2A5?, which means any value from 0x184D2A50 to 0x184D2A5F. -All 16 values are valid to identify a skippable frame. +Value : 0x184D2A5E. This is for compatibility with [Zstandard skippable frames]. +Since it is legal for other Zstandard skippable frames to use the same +magic number, it is not recommended for a decoder to recognize frames +solely on this. __`Frame_Size`__ -The total size of the skippable frame, not including the `Skippable_Magic_Number` or `Frame_Size`. This is for compatibility with [Zstandard skippable frames]. +The total size of the skippable frame, not including the `Skippable_Magic_Number` or `Frame_Size`. +This is for compatibility with [Zstandard skippable frames]. [Zstandard skippable frames]: https://github.com/facebook/zstd/blob/master/doc/zstd_compression_format.md#skippable-frames @@ -59,6 +62,12 @@ The seek table footer format is as follows: |------------------|-----------------------|-----------------------| | 4 bytes | 1 byte | 4 bytes | +__`Seekable_Magic_Number`__ + +Value : 0x8F92EAB1. +This value must be the last bytes present in the compressed file so that decoders +can efficiently find it and determine if there is an actual seek table present. + __`Number_Of_Chunks`__ The number of stored chunks in the data. diff --git a/contrib/seekable_format/zstdseek_compress.c b/contrib/seekable_format/zstdseek_compress.c index 2504287cd..44dd4afca 100644 --- a/contrib/seekable_format/zstdseek_compress.c +++ b/contrib/seekable_format/zstdseek_compress.c @@ -247,7 +247,7 @@ static size_t ZSTD_seekable_writeSeekTable(ZSTD_seekable_CStream* zcs, ZSTD_outB } \ } while (0) - st_write32(ZSTD_MAGIC_SKIPPABLE_START, 0); + st_write32(ZSTD_MAGIC_SKIPPABLE_START | 0xE, 0); st_write32(seekTableLen - ZSTD_skippableHeaderSize, 4); while (zcs->chunkDSize < zcs->chunklog.size) { diff --git a/contrib/seekable_format/zstdseek_decompress.c b/contrib/seekable_format/zstdseek_decompress.c index 3cb851ea5..2b97480f5 100644 --- a/contrib/seekable_format/zstdseek_decompress.c +++ b/contrib/seekable_format/zstdseek_decompress.c @@ -39,7 +39,7 @@ static U32 ZSTD_seekable_offsetToChunk(const seekTable_t* table, U64 pos) U32 hi = table->tableLen; while (lo + 1 < hi) { - U32 mid = lo + ((hi - lo) >> 1); + U32 const mid = lo + ((hi - lo) >> 1); if (table->entries[mid].dOffset <= pos) { lo = mid; } else { @@ -139,7 +139,7 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, if (srcSize < frameSize) return frameSize; - if ((MEM_readLE32(base) & 0xFFFFFFF0U) != ZSTD_MAGIC_SKIPPABLE_START) { + if (MEM_readLE32(base) != (ZSTD_MAGIC_SKIPPABLE_START | 0xE)) { return ERROR(prefix_unknown); } if (MEM_readLE32(base+4) + ZSTD_skippableHeaderSize != frameSize) { diff --git a/doc/README.md b/doc/README.md index 6f761b33e..47cfe3617 100644 --- a/doc/README.md +++ b/doc/README.md @@ -17,3 +17,4 @@ __`zstd_manual.html`__ : Documentation on the functions found in `zstd.h`. See [http://zstd.net/zstd_manual.html](http://zstd.net/zstd_manual.html) for the manual released with the latest official `zstd` release. + From 5ee1135f306a94df1c65889e13b3c70d4d4e6ef7 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Wed, 12 Apr 2017 11:06:00 -0700 Subject: [PATCH 178/305] s/chunk/frame/ --- .../examples/seekable_compression.c | 10 +- contrib/seekable_format/zstd_seekable.h | 36 ++-- .../zstd_seekable_compression_format.md | 20 +-- contrib/seekable_format/zstdseek_compress.c | 160 +++++++++--------- contrib/seekable_format/zstdseek_decompress.c | 54 +++--- lib/common/error_private.c | 2 +- lib/common/zstd_errors.h | 2 +- 7 files changed, 142 insertions(+), 142 deletions(-) diff --git a/contrib/seekable_format/examples/seekable_compression.c b/contrib/seekable_format/examples/seekable_compression.c index 8d3fe8576..a33952d93 100644 --- a/contrib/seekable_format/examples/seekable_compression.c +++ b/contrib/seekable_format/examples/seekable_compression.c @@ -59,7 +59,7 @@ static size_t fclose_orDie(FILE* file) exit(6); } -static void compressFile_orDie(const char* fname, const char* outName, int cLevel, unsigned chunkSize) +static void compressFile_orDie(const char* fname, const char* outName, int cLevel, unsigned frameSize) { FILE* const fin = fopen_orDie(fname, "rb"); FILE* const fout = fopen_orDie(outName, "wb"); @@ -70,7 +70,7 @@ static void compressFile_orDie(const char* fname, const char* outName, int cLeve ZSTD_seekable_CStream* const cstream = ZSTD_seekable_createCStream(); if (cstream==NULL) { fprintf(stderr, "ZSTD_seekable_createCStream() error \n"); exit(10); } - size_t const initResult = ZSTD_seekable_initCStream(cstream, cLevel, 1, chunkSize); + size_t const initResult = ZSTD_seekable_initCStream(cstream, cLevel, 1, frameSize); if (ZSTD_isError(initResult)) { fprintf(stderr, "ZSTD_seekable_initCStream() error : %s \n", ZSTD_getErrorName(initResult)); exit(11); } size_t read, toRead = buffInSize; @@ -116,15 +116,15 @@ int main(int argc, const char** argv) { if (argc!=3) { printf("wrong arguments\n"); printf("usage:\n"); - printf("%s FILE CHUNK_SIZE\n", exeName); + printf("%s FILE FRAME_SIZE\n", exeName); return 1; } { const char* const inFileName = argv[1]; - unsigned const chunkSize = (unsigned)atoi(argv[2]); + unsigned const frameSize = (unsigned)atoi(argv[2]); const char* const outFileName = createOutFilename_orDie(inFileName); - compressFile_orDie(inFileName, outFileName, 5, chunkSize); + compressFile_orDie(inFileName, outFileName, 5, frameSize); } return 0; diff --git a/contrib/seekable_format/zstd_seekable.h b/contrib/seekable_format/zstd_seekable.h index 959b1bd0f..f389ec973 100644 --- a/contrib/seekable_format/zstd_seekable.h +++ b/contrib/seekable_format/zstd_seekable.h @@ -9,17 +9,17 @@ static const unsigned ZSTD_seekTableFooterSize = 9; #define ZSTD_SEEKABLE_MAGICNUMBER 0x8F92EAB1 -#define ZSTD_SEEKABLE_MAXCHUNKS 0x8000000U +#define ZSTD_SEEKABLE_MAXFRAMES 0x8000000U /* 0xFE03F607 is the largest number x such that ZSTD_compressBound(x) fits in a 32-bit integer */ -#define ZSTD_SEEKABLE_MAX_CHUNK_DECOMPRESSED_SIZE 0xFE03F607 +#define ZSTD_SEEKABLE_MAX_FRAME_DECOMPRESSED_SIZE 0xFE03F607 /*-**************************************************************************** * Seekable Format * -* The seekable format splits the compressed data into a series of "chunks", +* The seekable format splits the compressed data into a series of "frames", * each compressed individually so that decompression of a section in the -* middle of an archive only requires zstd to decompress at most a chunk's +* middle of an archive only requires zstd to decompress at most a frame's * worth of extra data, instead of the entire archive. ******************************************************************************/ @@ -37,15 +37,15 @@ typedef struct ZSTD_seekable_DStream_s ZSTD_seekable_DStream; * compressor. * * Data streamed to the seekable compressor will automatically be split into -* chunks of size `maxChunkSize` (provided in ZSTD_seekable_initCStream()), -* or if none is provided, will be cut off whenver ZSTD_endChunk() is called -* or when the default maximum chunk size is reached (approximately 4GB). +* frames of size `maxFrameSize` (provided in ZSTD_seekable_initCStream()), +* or if none is provided, will be cut off whenver ZSTD_endFrame() is called +* or when the default maximum frame size is reached (approximately 4GB). * * Use ZSTD_seekable_initCStream() to initialize a ZSTD_seekable_CStream object * for a new compression operation. -* `maxChunkSize` indicates the size at which to automatically start a new -* seekable frame. `maxChunkSize == 0` implies the default maximum size. -* `checksumFlag` indicates whether or not the seek table should include chunk +* `maxFrameSize` indicates the size at which to automatically start a new +* seekable frame. `maxFrameSize == 0` implies the default maximum size. +* `checksumFlag` indicates whether or not the seek table should include frame * checksums on the uncompressed data for verification. * @return : a size hint for input to provide for compression, or an error code * checkable with ZSTD_isError() @@ -61,14 +61,14 @@ typedef struct ZSTD_seekable_DStream_s ZSTD_seekable_DStream; * value will work fine. * Note 2 : size hint is guaranteed to be <= ZSTD_CStreamInSize() * -* At any time, call ZSTD_seekable_endChunk() to end the current chunk and +* At any time, call ZSTD_seekable_endFrame() to end the current frame and * start a new one. * -* ZSTD_endStream() will end the current chunk, and then write the seek table -* so that decompressors can efficiently find compressed chunks. -* ZSTD_endStream() may return a number > 0 if it was unable to flush all the -* necessary data to `output`. In this case, it should be called again until -* all remaining data is flushed out and 0 is returned. +* ZSTD_seekable_endStream() will end the current frame, and then write the seek +* table so that decompressors can efficiently find compressed frames. +* ZSTD_seekable_endStream() may return a number > 0 if it was unable to flush +* all the necessary data to `output`. In this case, it should be called again +* until all remaining data is flushed out and 0 is returned. ******************************************************************************/ /*===== Seekable compressor management =====*/ @@ -76,9 +76,9 @@ ZSTDLIB_API ZSTD_seekable_CStream* ZSTD_seekable_createCStream(void); ZSTDLIB_API size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs); /*===== Seekable compression functions =====*/ -ZSTDLIB_API size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, int checksumFlag, unsigned maxChunkSize); +ZSTDLIB_API size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, int checksumFlag, unsigned maxFrameSize); ZSTDLIB_API size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output, ZSTD_inBuffer* input); -ZSTDLIB_API size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); +ZSTDLIB_API size_t ZSTD_seekable_endFrame(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); ZSTDLIB_API size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); /*-**************************************************************************** diff --git a/contrib/seekable_format/zstd_seekable_compression_format.md b/contrib/seekable_format/zstd_seekable_compression_format.md index cc98150ce..bf3080f7b 100644 --- a/contrib/seekable_format/zstd_seekable_compression_format.md +++ b/contrib/seekable_format/zstd_seekable_compression_format.md @@ -18,7 +18,7 @@ Distribution of this document is unlimited. ## Introduction This document defines a format for compressed data to be stored so that subranges of the data can be efficiently decompressed without requiring the entire document to be decompressed. -This is done by splitting up the input data into chunks, +This is done by splitting up the input data into frames, each of which are compressed independently, and so can be decompressed independently. Decompression then takes advantage of a provided 'seek table', which allows the decompressor to immediately jump to the desired data. This is done in a way that is compatible with the original Zstandard format by placing the seek table in a Zstandard skippable frame. @@ -31,7 +31,7 @@ In this document: ## Format -The format consists of a number of chunks (Zstandard compressed frames and skippable frames), followed by a final skippable frame at the end containing the seek table. +The format consists of a number of frames (Zstandard compressed frames and skippable frames), followed by a final skippable frame at the end containing the seek table. ### Seek Table Format The structure of the seek table frame is as follows: @@ -58,7 +58,7 @@ This is for compatibility with [Zstandard skippable frames]. #### `Seek_Table_Footer` The seek table footer format is as follows: -|`Number_Of_Chunks`|`Seek_Table_Descriptor`|`Seekable_Magic_Number`| +|`Number_Of_Frames`|`Seek_Table_Descriptor`|`Seekable_Magic_Number`| |------------------|-----------------------|-----------------------| | 4 bytes | 1 byte | 4 bytes | @@ -68,9 +68,9 @@ Value : 0x8F92EAB1. This value must be the last bytes present in the compressed file so that decoders can efficiently find it and determine if there is an actual seek table present. -__`Number_Of_Chunks`__ +__`Number_Of_Frames`__ -The number of stored chunks in the data. +The number of stored frames in the data. __`Seek_Table_Descriptor`__ @@ -87,13 +87,13 @@ for example the addition of inline dictionaries. __`Checksum_Flag`__ -If the checksum flag is set, each of the seek table entries contains a 4 byte checksum of the uncompressed data contained in its chunk. +If the checksum flag is set, each of the seek table entries contains a 4 byte checksum of the uncompressed data contained in its frame. `Reserved_Bits` are not currently used but may be used in the future for breaking changes, so a compliant decoder should ensure they are set to 0. `Unused_Bits` may be used in the future for non-breaking changes, so a compliant decoder should not interpret these bits. #### __`Seek_Table_Entries`__ -`Seek_Table_Entries` consists of `Number_Of_Chunks` (one for each chunk in the data, not including the seek table frame) entries of the following form, in sequence: +`Seek_Table_Entries` consists of `Number_Of_Frames` (one for each frame in the data, not including the seek table frame) entries of the following form, in sequence: |`Compressed_Size`|`Decompressed_Size`|`[Checksum]`| |-----------------|-------------------|------------| @@ -101,12 +101,12 @@ If the checksum flag is set, each of the seek table entries contains a 4 byte ch __`Compressed_Size`__ -The compressed size of the chunk. -The cumulative sum of the `Compressed_Size` fields of chunks `0` to `i` gives the offset in the compressed file of chunk `i+1`. +The compressed size of the frame. +The cumulative sum of the `Compressed_Size` fields of frames `0` to `i` gives the offset in the compressed file of frame `i+1`. __`Decompressed_Size`__ -The size of the decompressed data contained in the chunk. For skippable or otherwise empty frames, this value is 0. +The size of the decompressed data contained in the frame. For skippable or otherwise empty frames, this value is 0. __`Checksum`__ diff --git a/contrib/seekable_format/zstdseek_compress.c b/contrib/seekable_format/zstdseek_compress.c index 44dd4afca..df4feff28 100644 --- a/contrib/seekable_format/zstdseek_compress.c +++ b/contrib/seekable_format/zstdseek_compress.c @@ -20,24 +20,24 @@ typedef struct { U32 cSize; U32 dSize; U32 checksum; -} chunklogEntry_t; +} framelogEntry_t; typedef struct { - chunklogEntry_t* entries; + framelogEntry_t* entries; U32 size; U32 capacity; -} chunklog_t; +} framelog_t; struct ZSTD_seekable_CStream_s { ZSTD_CStream* cstream; - chunklog_t chunklog; + framelog_t framelog; - U32 chunkCSize; - U32 chunkDSize; + U32 frameCSize; + U32 frameDSize; XXH64_state_t xxhState; - U32 maxChunkSize; + U32 maxFrameSize; int checksumFlag; @@ -56,11 +56,11 @@ ZSTD_seekable_CStream* ZSTD_seekable_createCStream() if (zcs->cstream == NULL) goto failed1; /* allocate some initial space */ - { size_t const CHUNKLOG_STARTING_CAPACITY = 16; - zcs->chunklog.entries = - malloc(sizeof(chunklogEntry_t) * CHUNKLOG_STARTING_CAPACITY); - if (zcs->chunklog.entries == NULL) goto failed2; - zcs->chunklog.capacity = CHUNKLOG_STARTING_CAPACITY; + { size_t const FRAMELOG_STARTING_CAPACITY = 16; + zcs->framelog.entries = + malloc(sizeof(framelogEntry_t) * FRAMELOG_STARTING_CAPACITY); + if (zcs->framelog.entries == NULL) goto failed2; + zcs->framelog.capacity = FRAMELOG_STARTING_CAPACITY; } return zcs; @@ -76,7 +76,7 @@ size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs) { if (zcs == NULL) return 0; /* support free on null */ ZSTD_freeCStream(zcs->cstream); - free(zcs->chunklog.entries); + free(zcs->framelog.entries); free(zcs); return 0; @@ -85,20 +85,20 @@ size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs) size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, int compressionLevel, int checksumFlag, - U32 maxChunkSize) + U32 maxFrameSize) { - zcs->chunklog.size = 0; - zcs->chunkCSize = 0; - zcs->chunkDSize = 0; + zcs->framelog.size = 0; + zcs->frameCSize = 0; + zcs->frameDSize = 0; - /* make sure maxChunkSize has a reasonable value */ - if (maxChunkSize > ZSTD_SEEKABLE_MAX_CHUNK_DECOMPRESSED_SIZE) { + /* make sure maxFrameSize has a reasonable value */ + if (maxFrameSize > ZSTD_SEEKABLE_MAX_FRAME_DECOMPRESSED_SIZE) { return ERROR(compressionParameter_unsupported); } - zcs->maxChunkSize = maxChunkSize - ? maxChunkSize - : ZSTD_SEEKABLE_MAX_CHUNK_DECOMPRESSED_SIZE; + zcs->maxFrameSize = maxFrameSize + ? maxFrameSize + : ZSTD_SEEKABLE_MAX_FRAME_DECOMPRESSED_SIZE; zcs->checksumFlag = checksumFlag; if (zcs->checksumFlag) { @@ -110,58 +110,58 @@ size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, return ZSTD_initCStream(zcs->cstream, compressionLevel); } -static size_t ZSTD_seekable_logChunk(ZSTD_seekable_CStream* zcs) +static size_t ZSTD_seekable_logFrame(ZSTD_seekable_CStream* zcs) { - if (zcs->chunklog.size == ZSTD_SEEKABLE_MAXCHUNKS) - return ERROR(chunkIndex_tooLarge); + if (zcs->framelog.size == ZSTD_SEEKABLE_MAXFRAMES) + return ERROR(frameIndex_tooLarge); - zcs->chunklog.entries[zcs->chunklog.size] = (chunklogEntry_t) + zcs->framelog.entries[zcs->framelog.size] = (framelogEntry_t) { - .cSize = zcs->chunkCSize, - .dSize = zcs->chunkDSize, + .cSize = zcs->frameCSize, + .dSize = zcs->frameDSize, }; if (zcs->checksumFlag) - zcs->chunklog.entries[zcs->chunklog.size].checksum = + zcs->framelog.entries[zcs->framelog.size].checksum = /* take lower 32 bits of digest */ XXH64_digest(&zcs->xxhState) & 0xFFFFFFFFU; - zcs->chunklog.size++; + zcs->framelog.size++; /* grow the buffer if required */ - if (zcs->chunklog.size == zcs->chunklog.capacity) { + if (zcs->framelog.size == zcs->framelog.capacity) { /* exponential size increase for constant amortized runtime */ - size_t const newCapacity = zcs->chunklog.capacity * 2; - chunklogEntry_t* const newEntries = realloc(zcs->chunklog.entries, - sizeof(chunklogEntry_t) * newCapacity); + size_t const newCapacity = zcs->framelog.capacity * 2; + framelogEntry_t* const newEntries = realloc(zcs->framelog.entries, + sizeof(framelogEntry_t) * newCapacity); if (newEntries == NULL) return ERROR(memory_allocation); - zcs->chunklog.entries = newEntries; - zcs->chunklog.capacity = newCapacity; + zcs->framelog.entries = newEntries; + zcs->framelog.capacity = newCapacity; } return 0; } -size_t ZSTD_seekable_endChunk(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output) +size_t ZSTD_seekable_endFrame(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output) { size_t const prevOutPos = output->pos; /* end the frame */ size_t ret = ZSTD_endStream(zcs->cstream, output); - zcs->chunkCSize += output->pos - prevOutPos; + zcs->frameCSize += output->pos - prevOutPos; /* need to flush before doing the rest */ if (ret) return ret; /* frame done */ - /* store the chunk data for later */ - ret = ZSTD_seekable_logChunk(zcs); + /* store the frame data for later */ + ret = ZSTD_seekable_logFrame(zcs); if (ret) return ret; - /* reset for the next chunk */ - zcs->chunkCSize = 0; - zcs->chunkDSize = 0; + /* reset for the next frame */ + zcs->frameCSize = 0; + zcs->frameDSize = 0; ZSTD_resetCStream(zcs->cstream, 0); if (zcs->checksumFlag) @@ -175,9 +175,9 @@ size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* const BYTE* const inBase = (const BYTE*) input->src + input->pos; size_t inLen = input->size - input->pos; - inLen = MIN(inLen, (size_t)(zcs->maxChunkSize - zcs->chunkDSize)); + inLen = MIN(inLen, (size_t)(zcs->maxFrameSize - zcs->frameDSize)); - /* if we haven't finished flushing the last chunk, don't start writing a new one */ + /* if we haven't finished flushing the last frame, don't start writing a new one */ if (inLen > 0) { ZSTD_inBuffer inTmp = { inBase, inLen, 0 }; size_t const prevOutPos = output->pos; @@ -188,31 +188,31 @@ size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* XXH64_update(&zcs->xxhState, inBase, inTmp.pos); } - zcs->chunkCSize += output->pos - prevOutPos; - zcs->chunkDSize += inTmp.pos; + zcs->frameCSize += output->pos - prevOutPos; + zcs->frameDSize += inTmp.pos; input->pos += inTmp.pos; if (ZSTD_isError(ret)) return ret; } - if (zcs->maxChunkSize == zcs->chunkDSize) { - /* log the chunk and start over */ - size_t const ret = ZSTD_seekable_endChunk(zcs, output); + if (zcs->maxFrameSize == zcs->frameDSize) { + /* log the frame and start over */ + size_t const ret = ZSTD_seekable_endFrame(zcs, output); if (ZSTD_isError(ret)) return ret; - /* get the client ready for the next chunk */ - return (size_t)zcs->maxChunkSize; + /* get the client ready for the next frame */ + return (size_t)zcs->maxFrameSize; } - return (size_t)(zcs->maxChunkSize - zcs->chunkDSize); + return (size_t)(zcs->maxFrameSize - zcs->frameDSize); } static size_t ZSTD_seekable_seekTableSize(ZSTD_seekable_CStream* zcs) { - size_t const sizePerChunk = 8 + (zcs->checksumFlag?4:0); + size_t const sizePerFrame = 8 + (zcs->checksumFlag?4:0); size_t const seekTableLen = ZSTD_skippableHeaderSize + - sizePerChunk * zcs->chunklog.size + + sizePerFrame * zcs->framelog.size + ZSTD_seekTableFooterSize; return seekTableLen; @@ -224,60 +224,60 @@ static size_t ZSTD_seekable_writeSeekTable(ZSTD_seekable_CStream* zcs, ZSTD_outB BYTE tmp[4]; /* so that we can work with buffers too small to write a whole word to */ /* repurpose - * zcs->chunkDSize: the current index in the table and - * zcs->chunkCSize: the amount of the table written so far + * zcs->frameDSize: the current index in the table and + * zcs->frameCSize: the amount of the table written so far * * This function is written this way so that if it has to return early * because of a small buffer, it can keep going where it left off. */ - size_t const sizePerChunk = 8 + (zcs->checksumFlag?4:0); + size_t const sizePerFrame = 8 + (zcs->checksumFlag?4:0); size_t const seekTableLen = ZSTD_seekable_seekTableSize(zcs); #define st_write32(x, o) \ do { \ - if (zcs->chunkCSize < (o) + 4) { \ + if (zcs->frameCSize < (o) + 4) { \ size_t const lenWrite = MIN(output->size - output->pos, \ - (o) + 4 - zcs->chunkCSize); \ + (o) + 4 - zcs->frameCSize); \ MEM_writeLE32(tmp, (x)); \ - memcpy(op + output->pos, tmp + (zcs->chunkCSize - (o)), lenWrite); \ - zcs->chunkCSize += lenWrite; \ + memcpy(op + output->pos, tmp + (zcs->frameCSize - (o)), lenWrite); \ + zcs->frameCSize += lenWrite; \ output->pos += lenWrite; \ - if (lenWrite < 4) return seekTableLen - zcs->chunkCSize; \ + if (lenWrite < 4) return seekTableLen - zcs->frameCSize; \ } \ } while (0) st_write32(ZSTD_MAGIC_SKIPPABLE_START | 0xE, 0); st_write32(seekTableLen - ZSTD_skippableHeaderSize, 4); - while (zcs->chunkDSize < zcs->chunklog.size) { - st_write32(zcs->chunklog.entries[zcs->chunkDSize].cSize, - ZSTD_skippableHeaderSize + sizePerChunk * zcs->chunkDSize); - st_write32(zcs->chunklog.entries[zcs->chunkDSize].dSize, - ZSTD_skippableHeaderSize + sizePerChunk * zcs->chunkDSize + 4); + while (zcs->frameDSize < zcs->framelog.size) { + st_write32(zcs->framelog.entries[zcs->frameDSize].cSize, + ZSTD_skippableHeaderSize + sizePerFrame * zcs->frameDSize); + st_write32(zcs->framelog.entries[zcs->frameDSize].dSize, + ZSTD_skippableHeaderSize + sizePerFrame * zcs->frameDSize + 4); if (zcs->checksumFlag) { - st_write32(zcs->chunklog.entries[zcs->chunkDSize].checksum, - ZSTD_skippableHeaderSize + sizePerChunk * zcs->chunkDSize + 8); + st_write32(zcs->framelog.entries[zcs->frameDSize].checksum, + ZSTD_skippableHeaderSize + sizePerFrame * zcs->frameDSize + 8); } - zcs->chunkDSize++; + zcs->frameDSize++; } - st_write32(zcs->chunklog.size, seekTableLen - ZSTD_seekTableFooterSize); + st_write32(zcs->framelog.size, seekTableLen - ZSTD_seekTableFooterSize); - if (output->size - output->pos < 1) return seekTableLen - zcs->chunkCSize; - if (zcs->chunkCSize < seekTableLen - 4) { + if (output->size - output->pos < 1) return seekTableLen - zcs->frameCSize; + if (zcs->frameCSize < seekTableLen - 4) { BYTE sfd = 0; sfd |= (zcs->checksumFlag) << 7; op[output->pos] = sfd; output->pos++; - zcs->chunkCSize++; + zcs->frameCSize++; } st_write32(ZSTD_SEEKABLE_MAGICNUMBER, seekTableLen - 4); - if (zcs->chunkCSize != seekTableLen) return ERROR(GENERIC); + if (zcs->frameCSize != seekTableLen) return ERROR(GENERIC); return 0; #undef st_write32 @@ -285,11 +285,11 @@ static size_t ZSTD_seekable_writeSeekTable(ZSTD_seekable_CStream* zcs, ZSTD_outB size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output) { - if (!zcs->writingSeekTable && zcs->chunkDSize) { - const size_t endChunk = ZSTD_seekable_endChunk(zcs, output); - if (ZSTD_isError(endChunk)) return endChunk; + if (!zcs->writingSeekTable && zcs->frameDSize) { + const size_t endFrame = ZSTD_seekable_endFrame(zcs, output); + if (ZSTD_isError(endFrame)) return endFrame; /* return an accurate size hint */ - if (endChunk) return endChunk + ZSTD_seekable_seekTableSize(zcs); + if (endFrame) return endFrame + ZSTD_seekable_seekTableSize(zcs); } zcs->writingSeekTable = 1; diff --git a/contrib/seekable_format/zstdseek_decompress.c b/contrib/seekable_format/zstdseek_decompress.c index 2b97480f5..add62c762 100644 --- a/contrib/seekable_format/zstdseek_decompress.c +++ b/contrib/seekable_format/zstdseek_decompress.c @@ -29,11 +29,11 @@ typedef struct { int checksumFlag; } seekTable_t; -/** ZSTD_seekable_offsetToChunk() : - * Performs a binary search to find the last chunk with a decompressed offset +/** ZSTD_seekable_offsetToFrame() : + * Performs a binary search to find the last frame with a decompressed offset * <= pos - * @return : the chunk's index */ -static U32 ZSTD_seekable_offsetToChunk(const seekTable_t* table, U64 pos) + * @return : the frame's index */ +static U32 ZSTD_seekable_offsetToFrame(const seekTable_t* table, U64 pos) { U32 lo = 0; U32 hi = table->tableLen; @@ -61,7 +61,7 @@ struct ZSTD_seekable_DStream_s { ZSTD_DStream* dstream; seekTable_t seekTable; - U32 curChunk; + U32 curFrame; U64 compressedOffset; U64 decompressedOffset; @@ -107,7 +107,7 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, { const BYTE* ip = (const BYTE*)src + srcSize; - U32 numChunks; + U32 numFrames; int checksumFlag; U32 sizePerEntry; @@ -129,10 +129,10 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, } } - numChunks = MEM_readLE32(ip-9); + numFrames = MEM_readLE32(ip-9); sizePerEntry = 8 + (checksumFlag?4:0); - { U32 const tableSize = sizePerEntry * numChunks; + { U32 const tableSize = sizePerEntry * numFrames; U32 const frameSize = tableSize + ZSTD_seekTableFooterSize + ZSTD_skippableHeaderSize; const BYTE* base = ip - frameSize; @@ -148,7 +148,7 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, { /* Allocate an extra entry at the end so that we can do size * computations on the last element without special case */ - seekEntry_t* entries = malloc(sizeof(seekEntry_t) * (numChunks + 1)); + seekEntry_t* entries = malloc(sizeof(seekEntry_t) * (numFrames + 1)); const BYTE* tableBase = base + ZSTD_skippableHeaderSize; U32 idx; @@ -163,7 +163,7 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, } /* compute cumulative positions */ - for (idx = 0, pos = 0; idx < numChunks; idx++) { + for (idx = 0, pos = 0; idx < numFrames; idx++) { entries[idx].cOffset = cOffset; entries[idx].dOffset = dOffset; @@ -174,11 +174,11 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, pos += 4; } } - entries[numChunks].cOffset = cOffset; - entries[numChunks].dOffset = dOffset; + entries[numFrames].cOffset = cOffset; + entries[numFrames].dOffset = dOffset; zds->seekTable.entries = entries; - zds->seekTable.tableLen = numChunks; + zds->seekTable.tableLen = numFrames; zds->seekTable.checksumFlag = checksumFlag; return 0; } @@ -197,7 +197,7 @@ size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, U64 rangeStart, U64 zds->stage = zsds_seek; /* force a seek first */ - zds->curChunk = (U32) -1; + zds->curFrame = (U32) -1; zds->compressedOffset = (U64) -1; zds->decompressedOffset = (U64) -1; @@ -271,7 +271,7 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer /* need more input */ return MIN( ZSTD_DStreamInSize(), - (size_t)(jt->entries[zds->curChunk + 1] + (size_t)(jt->entries[zds->curFrame + 1] .cOffset - zds->compressedOffset)); } @@ -281,7 +281,7 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer { U64 const toDecompress = MIN(zds->targetEnd, - jt->entries[zds->curChunk + 1].dOffset) - + jt->entries[zds->curFrame + 1].dOffset) - zds->decompressedOffset; size_t const prevInputPos = input->pos; @@ -305,7 +305,7 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer if (ret == 0) { /* verify the checksum */ U32 const digest = XXH64_digest(&zds->xxhState) & 0xFFFFFFFFU; - if (digest != jt->entries[zds->curChunk].checksum) { + if (digest != jt->entries[zds->curFrame].checksum) { return ERROR(checksum_wrong); } @@ -323,9 +323,9 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer /* frame is done */ /* make sure this lines up with the expected frame border */ if (zds->decompressedOffset != - jt->entries[zds->curChunk + 1].dOffset || + jt->entries[zds->curFrame + 1].dOffset || zds->compressedOffset != - jt->entries[zds->curChunk + 1].cOffset) + jt->entries[zds->curFrame + 1].cOffset) return ERROR(corruption_detected); ZSTD_resetDStream(zds->dstream); zds->stage = zsds_seek; @@ -334,29 +334,29 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer /* need more input */ return MIN(ZSTD_DStreamInSize(), (size_t)( - jt->entries[zds->curChunk + 1].cOffset - + jt->entries[zds->curFrame + 1].cOffset - zds->compressedOffset)); } } case zsds_seek: { - U32 targetChunk; + U32 targetFrame; if (zds->decompressedOffset < zds->targetStart || zds->decompressedOffset >= zds->targetEnd) { /* haven't started yet */ - targetChunk = ZSTD_seekable_offsetToChunk(jt, zds->targetStart); + targetFrame = ZSTD_seekable_offsetToFrame(jt, zds->targetStart); } else { - targetChunk = ZSTD_seekable_offsetToChunk(jt, zds->decompressedOffset); + targetFrame = ZSTD_seekable_offsetToFrame(jt, zds->decompressedOffset); } - zds->curChunk = targetChunk; + zds->curFrame = targetFrame; - if (zds->compressedOffset == jt->entries[targetChunk].cOffset) { + if (zds->compressedOffset == jt->entries[targetFrame].cOffset) { zds->stage = zsds_decompress; break; } - zds->nextSeek = jt->entries[targetChunk].cOffset; - zds->decompressedOffset = jt->entries[targetChunk].dOffset; + zds->nextSeek = jt->entries[targetFrame].cOffset; + zds->decompressedOffset = jt->entries[targetFrame].dOffset; /* signal to user that a seek is required */ return ERROR(needSeek); } diff --git a/lib/common/error_private.c b/lib/common/error_private.c index 4084edca6..f32c6abda 100644 --- a/lib/common/error_private.c +++ b/lib/common/error_private.c @@ -38,7 +38,7 @@ const char* ERR_getErrorString(ERR_enum code) case PREFIX(dictionary_corrupted): return "Dictionary is corrupted"; case PREFIX(dictionary_wrong): return "Dictionary mismatch"; case PREFIX(dictionaryCreation_failed): return "Cannot create Dictionary from provided samples"; - case PREFIX(chunkIndex_tooLarge): return "Chunk index is too large"; + case PREFIX(frameIndex_tooLarge): return "Frame index is too large"; case PREFIX(needSeek): return "Wrong file position, a seek is required to continue"; case PREFIX(maxCode): default: return notErrorCode; diff --git a/lib/common/zstd_errors.h b/lib/common/zstd_errors.h index 13a5608e0..d11c1ba21 100644 --- a/lib/common/zstd_errors.h +++ b/lib/common/zstd_errors.h @@ -58,7 +58,7 @@ typedef enum { ZSTD_error_dictionary_corrupted, ZSTD_error_dictionary_wrong, ZSTD_error_dictionaryCreation_failed, - ZSTD_error_chunkIndex_tooLarge, + ZSTD_error_frameIndex_tooLarge, ZSTD_error_needSeek, ZSTD_error_maxCode } ZSTD_ErrorCode; From 2785b28e05d715a99fa66beee12bf47422fb8601 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Wed, 12 Apr 2017 14:09:13 -0700 Subject: [PATCH 179/305] Reduce the limit on frame decompressed size to 2 GB --- contrib/seekable_format/zstd_seekable.h | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/contrib/seekable_format/zstd_seekable.h b/contrib/seekable_format/zstd_seekable.h index f389ec973..2e148ebf3 100644 --- a/contrib/seekable_format/zstd_seekable.h +++ b/contrib/seekable_format/zstd_seekable.h @@ -11,8 +11,8 @@ static const unsigned ZSTD_seekTableFooterSize = 9; #define ZSTD_SEEKABLE_MAXFRAMES 0x8000000U -/* 0xFE03F607 is the largest number x such that ZSTD_compressBound(x) fits in a 32-bit integer */ -#define ZSTD_SEEKABLE_MAX_FRAME_DECOMPRESSED_SIZE 0xFE03F607 +/* Limit the maximum size to avoid any potential issues storing the compressed size */ +#define ZSTD_SEEKABLE_MAX_FRAME_DECOMPRESSED_SIZE 0x80000000U /*-**************************************************************************** * Seekable Format From afa48518e2fbd5dfce3982e4ed003f4c707d1801 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 13 Apr 2017 12:28:28 -0700 Subject: [PATCH 180/305] -T0 detects number of physical cores --- programs/util.h | 172 +++++++++++++++++++++++++++++++++++++++++++++ programs/zstdcli.c | 5 ++ 2 files changed, 177 insertions(+) diff --git a/programs/util.h b/programs/util.h index 9992d79da..0b90cda19 100644 --- a/programs/util.h +++ b/programs/util.h @@ -25,6 +25,7 @@ extern "C" { #include /* malloc */ #include /* size_t, ptrdiff_t */ #include /* fprintf */ +#include /* strncmp */ #include /* stat, utime */ #include /* stat */ #if defined(_MSC_VER) @@ -488,6 +489,177 @@ UTIL_STATIC void UTIL_freeFileList(const char** filenameTable, char* allocatedBu if (filenameTable) free((void*)filenameTable); } +/* count the number of physical cores */ +#if defined(_WIN32) || defined(WIN32) + +#include + +typedef BOOL(WINAPI* LPFN_GLPI)(PSYSTEM_LOGICAL_PROCESSOR_INFORMATION, PDWORD); + +UTIL_STATIC int UTIL_countPhysicalCores(void) +{ + static int numPhysicalCores; + if (numPhysicalCores != 0) return numPhysicalCores; + + { LPFN_GLPI glpi; + BOOL done = FALSE; + PSYSTEM_LOGICAL_PROCESSOR_INFORMATION buffer = NULL; + PSYSTEM_LOGICAL_PROCESSOR_INFORMATION ptr = NULL; + DWORD returnLength = 0; + size_t byteOffset = 0; + + glpi = (LPFN_GLPI)GetProcAddress(GetModuleHandle(TEXT("kernel32")), + "GetLogicalProcessorInformation"); + + if (glpi == NULL) { + goto failed; + } + + while(!done) { + DWORD rc = glpi(buffer, &returnLength); + if (FALSE == rc) { + if (GetLastError() == ERROR_INSUFFICIENT_BUFFER) { + if (buffer) + free(buffer); + buffer = (PSYSTEM_LOGICAL_PROCESSOR_INFORMATION)malloc(returnLength); + + if (buffer == NULL) { + perror("zstd"); + exit(1); + } + } else { + /* some other error */ + goto failed; + } + } else { + done = TRUE; + } + } + + while (byteOffset + sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION) <= returnLength) { + ptr = buffer; + + if (ptr->RelationShip == RelationProcessorCore) { + numPhysicalCores++; + } + } + + free(buffer); + + return numPhysicalCores; + } + +failed: + /* try to fall back on GetSystemInfo */ + SYSTEM_INFO sysinfo; + GetSystemInfo(&sysinfo); + numPhysicalCores = sysinfo.dwNumberOfProcessors; + if (numPhysicalCores == 0) numPhysicalCores = 1; /* just in case */ + return numPhysicalCores; +} + +#elif defined(__APPLE__) + +#include + +/* Use apple-provided syscall + * see: man 3 sysctl */ +UTIL_STATIC int UTIL_countPhysicalCores(void) +{ + static S32 numPhysicalCores; /* apple specifies int32_t */ + if (numPhysicalCores != 0) return numPhysicalCores; + + { int const ret = sysctlbyname("hw.physicalcpu", &numPhysicalCores, sizeof(int), NULL, 0); + if (ret != 0) { + if (errno == ENOENT) { + /* entry not present, fall back on 1 */ + numPhysicalCores = 1; + } else { + perror("zstd: can't get number of physical cpus"); + exit(1); + } + } + + return numPhysicalCores; + } +} + +#elif defined(__linux__) + +/* parse /proc/cpuinfo + * siblings / cpu cores should give hyperthreading ratio + * otherwise fall back on sysconf */ +UTIL_STATIC int UTIL_countPhysicalCores(void) +{ + static int numPhysicalCores; + + if (numPhysicalCores != 0) return numPhysicalCores; + + numPhysicalCores = sysconf(_SC_NPROCESSORS_ONLN); + if (numPhysicalCores == -1) { + /* value not queryable, fall back on 1 */ + return numPhysicalCores = 1; + } + + /* try to determine if there's hyperthreading */ + { FILE* const cpuinfo = fopen("/proc/cpuinfo", "r"); + size_t const BUF_SIZE = 80; + char buff[BUF_SIZE]; + + int siblings = 0; + int cpu_cores = 0; + int ratio = 1; + + if (cpuinfo == NULL) { + /* fall back on the sysconf value */ + return numPhysicalCores; + } + + /* assume the cpu cores/siblings values will be constant across all + * present processors */ + while (!feof(cpuinfo)) { + if (fgets(buff, BUF_SIZE, cpuinfo) != NULL) { + if (strncmp(buff, "siblings", 8) == 0) { + const char* const sep = strchr(buff, ':'); + if (*sep == '\0') { + /* formatting was broken? */ + goto failed; + } + + siblings = atoi(sep + 1); + } + if (strncmp(buff, "cpu cores", 9) == 0) { + const char* const sep = strchr(buff, ':'); + if (*sep == '\0') { + /* formatting was broken? */ + goto failed; + } + + cpu_cores = atoi(sep + 1); + } + } else if (ferror(cpuinfo)) { + /* fall back on the sysconf value */ + goto failed; + } + } + if (siblings && cpu_cores) { + ratio = siblings / cpu_cores; + } +failed: + fclose(cpuinfo); + return numPhysicalCores = numPhysicalCores / ratio; + } +} + +#else + +UTIL_STATIC int UTIL_countPhysicalCores(void) +{ + /* assume 1 */ + return 1; +} + +#endif #if defined (__cplusplus) } diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 18e259c74..73b4eab9d 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -602,6 +602,11 @@ int main(int argCount, const char* argv[]) DISPLAYLEVEL(4, "PLATFORM_POSIX_VERSION defined: %ldL\n", (long) PLATFORM_POSIX_VERSION); #endif + if (nbThreads == 0) { + /* try to guess */ + nbThreads = UTIL_countPhysicalCores(); + DISPLAYLEVEL(3, "Note: %d physical core(s) detected\n", nbThreads); + } g_utilDisplayLevel = g_displayLevel; if (!followLinks) { From f876f1200cab5d616bdecfb30144a23ea76bf9fd Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 13 Apr 2017 12:33:45 -0700 Subject: [PATCH 181/305] Fix compilation on macOS --- programs/bench.c | 9 +++++++-- programs/dibio.c | 4 +++- programs/fileio.c | 4 +++- programs/util.h | 3 ++- 4 files changed, 15 insertions(+), 5 deletions(-) diff --git a/programs/bench.c b/programs/bench.c index d25ff8f04..8d8ed4ffa 100644 --- a/programs/bench.c +++ b/programs/bench.c @@ -146,8 +146,13 @@ typedef struct { } blockParam_t; -#define MIN(a,b) ((a)<(b) ? (a) : (b)) -#define MAX(a,b) ((a)>(b) ? (a) : (b)) + +#ifndef MIN +# define MIN(a,b) ((a) < (b) ? (a) : (b)) +#endif +#ifndef MAX +# define MAX(a,b) ((a) > (b) ? (a) : (b)) +#endif static int BMK_benchMem(const void* srcBuffer, size_t srcSize, const char* displayName, int cLevel, diff --git a/programs/dibio.c b/programs/dibio.c index b7ed280ea..5e685a325 100644 --- a/programs/dibio.c +++ b/programs/dibio.c @@ -89,7 +89,9 @@ unsigned DiB_isError(size_t errorCode) { return ERR_isError(errorCode); } const char* DiB_getErrorName(size_t errorCode) { return ERR_getErrorName(errorCode); } -#define MIN(a,b) ( (a) < (b) ? (a) : (b) ) +#ifndef MIN +# define MIN(a,b) ((a) < (b) ? (a) : (b)) +#endif /* ******************************************************** diff --git a/programs/fileio.c b/programs/fileio.c index f1f5a5577..1dc222b65 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -96,7 +96,9 @@ void FIO_setNotificationLevel(unsigned level) { g_displayLevel=level; } static const clock_t refreshRate = CLOCKS_PER_SEC * 15 / 100; static clock_t g_time = 0; -#define MIN(a,b) ((a) < (b) ? (a) : (b)) +#ifndef MIN +# define MIN(a,b) ((a) < (b) ? (a) : (b)) +#endif /* ************************************************************ * Avoid fseek()'s 2GiB barrier with MSVC, MacOS, *BSD, MinGW diff --git a/programs/util.h b/programs/util.h index 0b90cda19..104f8dad2 100644 --- a/programs/util.h +++ b/programs/util.h @@ -569,7 +569,8 @@ UTIL_STATIC int UTIL_countPhysicalCores(void) static S32 numPhysicalCores; /* apple specifies int32_t */ if (numPhysicalCores != 0) return numPhysicalCores; - { int const ret = sysctlbyname("hw.physicalcpu", &numPhysicalCores, sizeof(int), NULL, 0); + { size_t size = sizeof(S32); + int const ret = sysctlbyname("hw.physicalcpu", &numPhysicalCores, &size, NULL, 0); if (ret != 0) { if (errno == ENOENT) { /* entry not present, fall back on 1 */ From 3b6207d4bd9143204745f90a38278891dbaa4474 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 13 Apr 2017 14:03:56 -0700 Subject: [PATCH 182/305] Fix compilation on windows --- programs/util.h | 19 ++++++++++++------- 1 file changed, 12 insertions(+), 7 deletions(-) diff --git a/programs/util.h b/programs/util.h index 104f8dad2..63cfa9647 100644 --- a/programs/util.h +++ b/programs/util.h @@ -536,12 +536,16 @@ UTIL_STATIC int UTIL_countPhysicalCores(void) } } - while (byteOffset + sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION) <= returnLength) { - ptr = buffer; + ptr = buffer; - if (ptr->RelationShip == RelationProcessorCore) { + while (byteOffset + sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION) <= returnLength) { + + if (ptr->Relationship == RelationProcessorCore) { numPhysicalCores++; } + + ptr++; + byteOffset += sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION); } free(buffer); @@ -551,10 +555,11 @@ UTIL_STATIC int UTIL_countPhysicalCores(void) failed: /* try to fall back on GetSystemInfo */ - SYSTEM_INFO sysinfo; - GetSystemInfo(&sysinfo); - numPhysicalCores = sysinfo.dwNumberOfProcessors; - if (numPhysicalCores == 0) numPhysicalCores = 1; /* just in case */ + { SYSTEM_INFO sysinfo; + GetSystemInfo(&sysinfo); + numPhysicalCores = sysinfo.dwNumberOfProcessors; + if (numPhysicalCores == 0) numPhysicalCores = 1; /* just in case */ + } return numPhysicalCores; } From 9227aae0013b44fc7628911d735731e01d1f9480 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 13 Apr 2017 14:06:40 -0700 Subject: [PATCH 183/305] Fix clang linux compilation --- programs/util.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/programs/util.h b/programs/util.h index 63cfa9647..a50b5e55b 100644 --- a/programs/util.h +++ b/programs/util.h @@ -601,7 +601,7 @@ UTIL_STATIC int UTIL_countPhysicalCores(void) if (numPhysicalCores != 0) return numPhysicalCores; - numPhysicalCores = sysconf(_SC_NPROCESSORS_ONLN); + numPhysicalCores = (int)sysconf(_SC_NPROCESSORS_ONLN); if (numPhysicalCores == -1) { /* value not queryable, fall back on 1 */ return numPhysicalCores = 1; From ad8da8855bd3a14c84789eda116d7f33920f454b Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 13 Apr 2017 14:40:06 -0700 Subject: [PATCH 184/305] Make appveyor small tests use new mingw as well --- appveyor.yml | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index 27994853a..5887d52e6 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -218,10 +218,10 @@ - ECHO Installing %COMPILER% %PLATFORM% %CONFIGURATION% - SET PATH_ORIGINAL=%PATH% - if [%HOST%]==[mingw] ( - SET "PATH_MINGW32=C:\MinGW\bin;C:\MinGW\usr\bin" && - SET "PATH_MINGW64=C:\msys64\mingw64\bin;C:\msys64\usr\bin" && - COPY C:\msys64\usr\bin\make.exe C:\MinGW\bin\make.exe && - COPY C:\MinGW\bin\gcc.exe C:\MinGW\bin\cc.exe + SET "PATH_MINGW32=C:\mingw-w64\i686-6.3.0-posix-dwarf-rt_v5-rev1\mingw32\bin" && + SET "PATH_MINGW64=C:\mingw-w64\x86_64-6.3.0-posix-seh-rt_v5-rev1\mingw64\bin" && + COPY C:\msys64\usr\bin\make.exe C:\mingw-w64\i686-6.3.0-posix-dwarf-rt_v5-rev1\mingw32\bin\make.exe && + COPY C:\msys64\usr\bin\make.exe C:\mingw-w64\x86_64-6.3.0-posix-seh-rt_v5-rev1\mingw64\bin\make.exe ) - IF [%HOST%]==[visual] IF [%PLATFORM%]==[x64] ( SET ADDITIONALPARAM=/p:LibraryPath="C:\Program Files\Microsoft SDKs\Windows\v7.1\lib\x64;c:\Program Files (x86)\Microsoft Visual Studio 10.0\VC\lib\amd64;C:\Program Files (x86)\Microsoft Visual Studio 10.0\;C:\Program Files (x86)\Microsoft Visual Studio 10.0\lib\amd64;" From 42bac7fa84e5ce26c582ea517e4c51befd19cd88 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 13 Apr 2017 15:35:05 -0700 Subject: [PATCH 185/305] Change ifndef's to undef's --- programs/bench.c | 10 ++++------ programs/dibio.c | 5 ++--- programs/fileio.c | 5 ++--- 3 files changed, 8 insertions(+), 12 deletions(-) diff --git a/programs/bench.c b/programs/bench.c index 8d8ed4ffa..c3681eb05 100644 --- a/programs/bench.c +++ b/programs/bench.c @@ -147,12 +147,10 @@ typedef struct { -#ifndef MIN -# define MIN(a,b) ((a) < (b) ? (a) : (b)) -#endif -#ifndef MAX -# define MAX(a,b) ((a) > (b) ? (a) : (b)) -#endif +#undef MIN +#undef MAX +#define MIN(a,b) ((a) < (b) ? (a) : (b)) +#define MAX(a,b) ((a) > (b) ? (a) : (b)) static int BMK_benchMem(const void* srcBuffer, size_t srcSize, const char* displayName, int cLevel, diff --git a/programs/dibio.c b/programs/dibio.c index 5e685a325..aac36425c 100644 --- a/programs/dibio.c +++ b/programs/dibio.c @@ -89,9 +89,8 @@ unsigned DiB_isError(size_t errorCode) { return ERR_isError(errorCode); } const char* DiB_getErrorName(size_t errorCode) { return ERR_getErrorName(errorCode); } -#ifndef MIN -# define MIN(a,b) ((a) < (b) ? (a) : (b)) -#endif +#undef MIN +#define MIN(a,b) ((a) < (b) ? (a) : (b)) /* ******************************************************** diff --git a/programs/fileio.c b/programs/fileio.c index 1dc222b65..9bca20664 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -96,9 +96,8 @@ void FIO_setNotificationLevel(unsigned level) { g_displayLevel=level; } static const clock_t refreshRate = CLOCKS_PER_SEC * 15 / 100; static clock_t g_time = 0; -#ifndef MIN -# define MIN(a,b) ((a) < (b) ? (a) : (b)) -#endif +#undef MIN +#define MIN(a,b) ((a) < (b) ? (a) : (b)) /* ************************************************************ * Avoid fseek()'s 2GiB barrier with MSVC, MacOS, *BSD, MinGW From e4f3235c85303cee20a2f93732b93a346f273c48 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 13 Apr 2017 16:34:28 -0700 Subject: [PATCH 186/305] Add 0 initializers to static variables --- programs/util.h | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/programs/util.h b/programs/util.h index a50b5e55b..b989e8232 100644 --- a/programs/util.h +++ b/programs/util.h @@ -498,7 +498,7 @@ typedef BOOL(WINAPI* LPFN_GLPI)(PSYSTEM_LOGICAL_PROCESSOR_INFORMATION, PDWORD); UTIL_STATIC int UTIL_countPhysicalCores(void) { - static int numPhysicalCores; + static int numPhysicalCores = 0; if (numPhysicalCores != 0) return numPhysicalCores; { LPFN_GLPI glpi; @@ -571,7 +571,7 @@ failed: * see: man 3 sysctl */ UTIL_STATIC int UTIL_countPhysicalCores(void) { - static S32 numPhysicalCores; /* apple specifies int32_t */ + static S32 numPhysicalCores = 0; /* apple specifies int32_t */ if (numPhysicalCores != 0) return numPhysicalCores; { size_t size = sizeof(S32); @@ -597,7 +597,7 @@ UTIL_STATIC int UTIL_countPhysicalCores(void) * otherwise fall back on sysconf */ UTIL_STATIC int UTIL_countPhysicalCores(void) { - static int numPhysicalCores; + static int numPhysicalCores = 0; if (numPhysicalCores != 0) return numPhysicalCores; From 9626cf1ac636edd84341ada6419302335185bf7d Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 13 Apr 2017 17:47:55 -0700 Subject: [PATCH 187/305] Address @terrelln's comments --- contrib/seekable_format/examples/Makefile | 2 +- contrib/seekable_format/zstd_seekable.h | 9 +- contrib/seekable_format/zstdseek_compress.c | 129 +++++++++++------- contrib/seekable_format/zstdseek_decompress.c | 49 ++++--- 4 files changed, 109 insertions(+), 80 deletions(-) diff --git a/contrib/seekable_format/examples/Makefile b/contrib/seekable_format/examples/Makefile index b57774ef3..fcd1d9146 100644 --- a/contrib/seekable_format/examples/Makefile +++ b/contrib/seekable_format/examples/Makefile @@ -12,7 +12,7 @@ LDFLAGS += -L../../../lib/ -lzstd CPPFLAGS += -I../ -I../../../lib -I../../../lib/common -CFLAGS = -O3 +CFLAGS ?= -O3 CFLAGS += -g SEEKABLE_OBJS = ../zstdseek_compress.c ../zstdseek_decompress.c diff --git a/contrib/seekable_format/zstd_seekable.h b/contrib/seekable_format/zstd_seekable.h index 2e148ebf3..3ab4f185e 100644 --- a/contrib/seekable_format/zstd_seekable.h +++ b/contrib/seekable_format/zstd_seekable.h @@ -38,8 +38,8 @@ typedef struct ZSTD_seekable_DStream_s ZSTD_seekable_DStream; * * Data streamed to the seekable compressor will automatically be split into * frames of size `maxFrameSize` (provided in ZSTD_seekable_initCStream()), -* or if none is provided, will be cut off whenver ZSTD_endFrame() is called -* or when the default maximum frame size is reached (approximately 4GB). +* or if none is provided, will be cut off whenever ZSTD_seekable_endFrame() is +* called or when the default maximum frame size (2GB) is reached. * * Use ZSTD_seekable_initCStream() to initialize a ZSTD_seekable_CStream object * for a new compression operation. @@ -59,7 +59,6 @@ typedef struct ZSTD_seekable_DStream_s ZSTD_seekable_DStream; * ZSTD_isError(). * Note 1 : it's just a hint, to help latency a little, any other * value will work fine. -* Note 2 : size hint is guaranteed to be <= ZSTD_CStreamInSize() * * At any time, call ZSTD_seekable_endFrame() to end the current frame and * start a new one. @@ -98,8 +97,8 @@ ZSTDLIB_API size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outB * small, a size hint for how much data to provide. * An error code may also be returned, checkable with ZSTD_isError() * -* Use ZSTD_initDStream to prepare for a new decompression operation using the -* seektable loaded with ZSTD_seekable_loadSeekTable(). +* Use ZSTD_seekable_initDStream to prepare for a new decompression operation +* using the seektable loaded with ZSTD_seekable_loadSeekTable(). * Data in the range [rangeStart, rangeEnd) will be decompressed. * * Call ZSTD_seekable_decompressStream() repetitively to consume input stream. diff --git a/contrib/seekable_format/zstdseek_compress.c b/contrib/seekable_format/zstdseek_compress.c index df4feff28..2d1698bee 100644 --- a/contrib/seekable_format/zstdseek_compress.c +++ b/contrib/seekable_format/zstdseek_compress.c @@ -13,9 +13,22 @@ #define XXH_NAMESPACE ZSTD_ #include "xxhash.h" -#include "zstd_internal.h" /* includes zstd.h */ +#define ZSTD_STATIC_LINKING_ONLY +#include "zstd.h" +#include "zstd_errors.h" +#include "mem.h" #include "zstd_seekable.h" +#define CHECK_Z(f) { size_t const ret = (f); if (ret != 0) return ret; } + +#undef ERROR +#define ERROR(name) ((size_t)-ZSTD_error_##name) + +#undef MIN +#undef MAX +#define MIN(a, b) ((a) < (b) ? (a) : (b)) +#define MAX(a, b) ((a) > (b) ? (a) : (b)) + typedef struct { U32 cSize; U32 dSize; @@ -42,6 +55,8 @@ struct ZSTD_seekable_CStream_s { int checksumFlag; int writingSeekTable; + U32 seekTablePos; + U32 seekTableIndex; }; ZSTD_seekable_CStream* ZSTD_seekable_createCStream() @@ -57,8 +72,8 @@ ZSTD_seekable_CStream* ZSTD_seekable_createCStream() /* allocate some initial space */ { size_t const FRAMELOG_STARTING_CAPACITY = 16; - zcs->framelog.entries = - malloc(sizeof(framelogEntry_t) * FRAMELOG_STARTING_CAPACITY); + zcs->framelog.entries = (framelogEntry_t*)malloc( + sizeof(framelogEntry_t) * FRAMELOG_STARTING_CAPACITY); if (zcs->framelog.entries == NULL) goto failed2; zcs->framelog.capacity = FRAMELOG_STARTING_CAPACITY; } @@ -105,6 +120,8 @@ size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, XXH64_reset(&zcs->xxhState, 0); } + zcs->seekTablePos = 0; + zcs->seekTableIndex = 0; zcs->writingSeekTable = 0; return ZSTD_initCStream(zcs->cstream, compressionLevel); @@ -115,17 +132,6 @@ static size_t ZSTD_seekable_logFrame(ZSTD_seekable_CStream* zcs) if (zcs->framelog.size == ZSTD_SEEKABLE_MAXFRAMES) return ERROR(frameIndex_tooLarge); - zcs->framelog.entries[zcs->framelog.size] = (framelogEntry_t) - { - .cSize = zcs->frameCSize, - .dSize = zcs->frameDSize, - }; - if (zcs->checksumFlag) - zcs->framelog.entries[zcs->framelog.size].checksum = - /* take lower 32 bits of digest */ - XXH64_digest(&zcs->xxhState) & 0xFFFFFFFFU; - - zcs->framelog.size++; /* grow the buffer if required */ if (zcs->framelog.size == zcs->framelog.capacity) { /* exponential size increase for constant amortized runtime */ @@ -139,6 +145,15 @@ static size_t ZSTD_seekable_logFrame(ZSTD_seekable_CStream* zcs) zcs->framelog.capacity = newCapacity; } + zcs->framelog.entries[zcs->framelog.size] = (framelogEntry_t){ + zcs->frameCSize, zcs->frameDSize, + }; + if (zcs->checksumFlag) + zcs->framelog.entries[zcs->framelog.size].checksum = + XXH64_digest(&zcs->xxhState) & 0xFFFFFFFFU; /* take lower 32 bits of digest */ + + zcs->framelog.size++; + return 0; } @@ -208,7 +223,7 @@ size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* return (size_t)(zcs->maxFrameSize - zcs->frameDSize); } -static size_t ZSTD_seekable_seekTableSize(ZSTD_seekable_CStream* zcs) +static inline size_t ZSTD_seekable_seekTableSize(ZSTD_seekable_CStream const* zcs) { size_t const sizePerFrame = 8 + (zcs->checksumFlag?4:0); size_t const seekTableLen = ZSTD_skippableHeaderSize + @@ -218,14 +233,29 @@ static size_t ZSTD_seekable_seekTableSize(ZSTD_seekable_CStream* zcs) return seekTableLen; } +static inline size_t ZSTD_stwrite32(ZSTD_seekable_CStream* zcs, + ZSTD_outBuffer* output, U32 const value, + U32 const offset) +{ + if (zcs->seekTablePos < offset + 4) { + BYTE tmp[4]; /* so that we can work with buffers too small to write a whole word to */ + size_t const lenWrite = + MIN(output->size - output->pos, offset + 4 - zcs->seekTablePos); + MEM_writeLE32(tmp, value); + memcpy((BYTE*)output->dst + output->pos, + tmp + (zcs->seekTablePos - offset), lenWrite); + output->pos += lenWrite; + zcs->seekTablePos += lenWrite; + + if (lenWrite < 4) return ZSTD_seekable_seekTableSize(zcs) - zcs->seekTablePos; + } + return 0; +} + static size_t ZSTD_seekable_writeSeekTable(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output) { - BYTE* op = (BYTE*) output->dst; - BYTE tmp[4]; /* so that we can work with buffers too small to write a whole word to */ - - /* repurpose - * zcs->frameDSize: the current index in the table and - * zcs->frameCSize: the amount of the table written so far + /* seekTableIndex: the current index in the table and + * seekTableSize: the amount of the table written so far * * This function is written this way so that if it has to return early * because of a small buffer, it can keep going where it left off. @@ -234,53 +264,48 @@ static size_t ZSTD_seekable_writeSeekTable(ZSTD_seekable_CStream* zcs, ZSTD_outB size_t const sizePerFrame = 8 + (zcs->checksumFlag?4:0); size_t const seekTableLen = ZSTD_seekable_seekTableSize(zcs); -#define st_write32(x, o) \ - do { \ - if (zcs->frameCSize < (o) + 4) { \ - size_t const lenWrite = MIN(output->size - output->pos, \ - (o) + 4 - zcs->frameCSize); \ - MEM_writeLE32(tmp, (x)); \ - memcpy(op + output->pos, tmp + (zcs->frameCSize - (o)), lenWrite); \ - zcs->frameCSize += lenWrite; \ - output->pos += lenWrite; \ - if (lenWrite < 4) return seekTableLen - zcs->frameCSize; \ - } \ - } while (0) + CHECK_Z(ZSTD_stwrite32(zcs, output, ZSTD_MAGIC_SKIPPABLE_START | 0xE, 0)); + CHECK_Z(ZSTD_stwrite32(zcs, output, seekTableLen - ZSTD_skippableHeaderSize, + 4)); - st_write32(ZSTD_MAGIC_SKIPPABLE_START | 0xE, 0); - st_write32(seekTableLen - ZSTD_skippableHeaderSize, 4); + while (zcs->seekTableIndex < zcs->framelog.size) { + CHECK_Z(ZSTD_stwrite32( + zcs, output, zcs->framelog.entries[zcs->seekTableIndex].cSize, + ZSTD_skippableHeaderSize + sizePerFrame * zcs->seekTableIndex)); + + CHECK_Z(ZSTD_stwrite32( + zcs, output, zcs->framelog.entries[zcs->seekTableIndex].dSize, + ZSTD_skippableHeaderSize + sizePerFrame * zcs->seekTableIndex + 4)); - while (zcs->frameDSize < zcs->framelog.size) { - st_write32(zcs->framelog.entries[zcs->frameDSize].cSize, - ZSTD_skippableHeaderSize + sizePerFrame * zcs->frameDSize); - st_write32(zcs->framelog.entries[zcs->frameDSize].dSize, - ZSTD_skippableHeaderSize + sizePerFrame * zcs->frameDSize + 4); if (zcs->checksumFlag) { - st_write32(zcs->framelog.entries[zcs->frameDSize].checksum, - ZSTD_skippableHeaderSize + sizePerFrame * zcs->frameDSize + 8); + CHECK_Z(ZSTD_stwrite32( + zcs, output, + zcs->framelog.entries[zcs->seekTableIndex].checksum, + ZSTD_skippableHeaderSize + sizePerFrame * zcs->seekTableIndex + + 8)); } - zcs->frameDSize++; + zcs->seekTableIndex++; } - st_write32(zcs->framelog.size, seekTableLen - ZSTD_seekTableFooterSize); + CHECK_Z(ZSTD_stwrite32(zcs, output, zcs->framelog.size, + seekTableLen - ZSTD_seekTableFooterSize)); - if (output->size - output->pos < 1) return seekTableLen - zcs->frameCSize; - if (zcs->frameCSize < seekTableLen - 4) { + if (output->size - output->pos < 1) return seekTableLen - zcs->seekTablePos; + if (zcs->seekTablePos < seekTableLen - 4) { BYTE sfd = 0; sfd |= (zcs->checksumFlag) << 7; - op[output->pos] = sfd; + ((BYTE*)output->dst)[output->pos] = sfd; output->pos++; - zcs->frameCSize++; + zcs->seekTablePos++; } - st_write32(ZSTD_SEEKABLE_MAGICNUMBER, seekTableLen - 4); + CHECK_Z(ZSTD_stwrite32(zcs, output, ZSTD_SEEKABLE_MAGICNUMBER, + seekTableLen - 4)); - if (zcs->frameCSize != seekTableLen) return ERROR(GENERIC); + if (zcs->seekTablePos != seekTableLen) return ERROR(GENERIC); return 0; - -#undef st_write32 } size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output) diff --git a/contrib/seekable_format/zstdseek_decompress.c b/contrib/seekable_format/zstdseek_decompress.c index add62c762..87a140c02 100644 --- a/contrib/seekable_format/zstdseek_decompress.c +++ b/contrib/seekable_format/zstdseek_decompress.c @@ -13,9 +13,20 @@ #define XXH_NAMESPACE ZSTD_ #include "xxhash.h" -#include "zstd_internal.h" /* includes zstd.h */ +#define ZSTD_STATIC_LINKING_ONLY +#include "zstd.h" +#include "zstd_errors.h" +#include "mem.h" /* includes zstd.h */ #include "zstd_seekable.h" +#undef ERROR +#define ERROR(name) ((size_t)-ZSTD_error_##name) + +#undef MIN +#undef MAX +#define MIN(a, b) ((a) < (b) ? (a) : (b)) +#define MAX(a, b) ((a) > (b) ? (a) : (b)) + typedef struct { U64 cOffset; U64 dOffset; @@ -107,11 +118,8 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, { const BYTE* ip = (const BYTE*)src + srcSize; - U32 numFrames; int checksumFlag; - U32 sizePerEntry; - /* footer is fixed size */ if (srcSize < ZSTD_seekTableFooterSize) return ZSTD_seekTableFooterSize; @@ -129,10 +137,9 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, } } - numFrames = MEM_readLE32(ip-9); - sizePerEntry = 8 + (checksumFlag?4:0); - - { U32 const tableSize = sizePerEntry * numFrames; + { U32 const numFrames = MEM_readLE32(ip-9); + U32 const sizePerEntry = 8 + (checksumFlag?4:0); + U32 const tableSize = sizePerEntry * numFrames; U32 const frameSize = tableSize + ZSTD_seekTableFooterSize + ZSTD_skippableHeaderSize; const BYTE* base = ip - frameSize; @@ -148,7 +155,8 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, { /* Allocate an extra entry at the end so that we can do size * computations on the last element without special case */ - seekEntry_t* entries = malloc(sizeof(seekEntry_t) * (numFrames + 1)); + seekEntry_t* entries = + (seekEntry_t*)malloc(sizeof(seekEntry_t) * (numFrames + 1)); const BYTE* tableBase = base + ZSTD_skippableHeaderSize; U32 idx; @@ -167,8 +175,10 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, entries[idx].cOffset = cOffset; entries[idx].dOffset = dOffset; - cOffset += MEM_readLE32(tableBase + pos); pos += 4; - dOffset += MEM_readLE32(tableBase + pos); pos += 4; + cOffset += MEM_readLE32(tableBase + pos); + pos += 4; + dOffset += MEM_readLE32(tableBase + pos); + pos += 4; if (checksumFlag) { entries[idx].checksum = MEM_readLE32(tableBase + pos); pos += 4; @@ -188,18 +198,17 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, U64 rangeStart, U64 rangeEnd) { /* restrict range to the end of the file, of non-negative size */ - rangeStart = MIN(rangeStart, zds->seekTable.entries[zds->seekTable.tableLen].dOffset); rangeEnd = MIN(rangeEnd, zds->seekTable.entries[zds->seekTable.tableLen].dOffset); - rangeEnd = MAX(rangeEnd, rangeStart); + rangeStart = MIN(rangeStart, rangeEnd); zds->targetStart = rangeStart; zds->targetEnd = rangeEnd; zds->stage = zsds_seek; /* force a seek first */ - zds->curFrame = (U32) -1; - zds->compressedOffset = (U64) -1; - zds->decompressedOffset = (U64) -1; + zds->curFrame = (U32)-1; + zds->compressedOffset = (U64)-1; + zds->decompressedOffset = (U64)-1; if (zds->seekTable.checksumFlag) { XXH64_reset(&zds->xxhState, 0); @@ -247,9 +256,7 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer size_t const prevInputPos = input->pos; ZSTD_outBuffer outTmp = { - .dst = outBase, - .size = (size_t)MIN((U64)outLen, toDecompress), - .pos = 0}; + outBase, (size_t)MIN((U64)outLen, toDecompress), 0}; size_t const ret = ZSTD_decompressStream(zds->dstream, &outTmp, input); @@ -286,9 +293,7 @@ size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer size_t const prevInputPos = input->pos; ZSTD_outBuffer outTmp = { - .dst = outBase, - .size = (size_t)MIN((U64)outLen, toDecompress), - .pos = 0}; + outBase, (size_t)MIN((U64)outLen, toDecompress), 0}; size_t const ret = ZSTD_decompressStream(zds->dstream, &outTmp, input); From f913cbed33e4707fab0fdf61bacf1b5076454a14 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 13 Apr 2017 22:46:41 -0700 Subject: [PATCH 188/305] fixed : memory leak in fuzzer test --- tests/fuzzer.c | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/tests/fuzzer.c b/tests/fuzzer.c index cdbbe8a22..ee051be3e 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -549,7 +549,9 @@ static int basicUnitTests(U32 seed, double compressibility) if (!ZSTD_isError(result)) goto _output_error; if (ZSTD_getErrorCode(result) != ZSTD_error_srcSize_wrong) goto _output_error; DISPLAYLEVEL(4, "OK : %s \n", ZSTD_getErrorName(result)); - } } + } + ZSTD_freeCCtx(cctx); + } /* block API tests */ { ZSTD_CCtx* const cctx = ZSTD_createCCtx(); From 7dd14d03b069965c9577869841ce5646da7b7081 Mon Sep 17 00:00:00 2001 From: Baptiste Daroussin Date: Sat, 15 Apr 2017 16:25:08 +0200 Subject: [PATCH 189/305] Enable multithreading on BSD --- programs/util.h | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/programs/util.h b/programs/util.h index b989e8232..5f437b2b2 100644 --- a/programs/util.h +++ b/programs/util.h @@ -657,6 +657,24 @@ failed: } } +#elif defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__DragonFly__) + +/* Use apple-provided syscall + * see: man 3 sysctl */ +UTIL_STATIC int UTIL_countPhysicalCores(void) +{ + static int numPhysicalCores = 0; + + if (numPhysicalCores != 0) return numPhysicalCores; + + numPhysicalCores = (int)sysconf(_SC_NPROCESSORS_ONLN); + if (numPhysicalCores == -1) { + /* value not queryable, fall back on 1 */ + return numPhysicalCores = 1; + } + return numPhysicalCores; +} + #else UTIL_STATIC int UTIL_countPhysicalCores(void) From c424ec2eae3cbb5451ab82f7390ceb3a28b98878 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 17 Apr 2017 11:38:53 -0700 Subject: [PATCH 190/305] Add multithreading tests to playTests.sh --- tests/playTests.sh | 33 +++++++++++++++++++++++++++++---- 1 file changed, 29 insertions(+), 4 deletions(-) diff --git a/tests/playTests.sh b/tests/playTests.sh index 3675bb168..ce1e2582e 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -31,7 +31,7 @@ fileRoundTripTest() { fi rm -f tmp.zstd tmp.md5.1 tmp.md5.2 - $ECHO "fileRoundTripTest: ./datagen $1 $local_p > tmp && $ZSTD -v$local_c -c | $ZSTD -d" + $ECHO "fileRoundTripTest: ./datagen $1 $local_p > tmp && $ZSTD -v$local_c -c tmp | $ZSTD -d" ./datagen $1 $local_p > tmp cat tmp | $MD5SUM > tmp.md5.1 $ZSTD --ultra -v$local_c -c tmp | $ZSTD -d | $MD5SUM > tmp.md5.2 @@ -45,12 +45,11 @@ then fi isWindows=false -ECHO="echo" +ECHO="echo -e" INTOVOID="/dev/null" case "$OS" in Windows*) isWindows=true - ECHO="echo -e" INTOVOID="NUL" ;; esac @@ -67,11 +66,17 @@ case "$UNAME" in SunOS) DIFF="gdiff" ;; esac - $ECHO "\nStarting playTests.sh isWindows=$isWindows ZSTD='$ZSTD'" [ -n "$ZSTD" ] || die "ZSTD variable must be defined!" +if [ -n "$(echo hello | $ZSTD -v -T2 2>&1 > $INTOVOID | grep 'multi-threading is disabled')" ] +then + hasMT="" +else + hasMT="true" +fi + $ECHO "\n**** simple tests **** " ./datagen > tmp @@ -461,6 +466,16 @@ roundTripTest -g516K 19 # btopt fileRoundTripTest -g500K +if [ -n "$hasMT" ] +then + $ECHO "\n**** zstdmt round-trip tests **** " + roundTripTest -g516K "16 -T0" + roundTripTest -g516K "19 -T2" + fileRoundTripTest -g500K " -T2" +else + $ECHO "\n**** no multithreading, skipping zstdmt tests **** " +fi + rm tmp* if [ "$1" != "--test-large-data" ]; then @@ -498,4 +513,14 @@ roundTripTest -g6000000000 -P99 1 fileRoundTripTest -g4193M -P99 1 +if [ -n "$hasMT" ] +then + $ECHO "\n**** zstdmt long round-trip tests **** " + roundTripTest -g99000000 -P99 "20 -T2" + roundTripTest -g6000000000 -P99 "1 -T2" + fileRoundTripTest -g4193M -P98 " -T0" +else + $ECHO "\n**** no multithreading, skipping zstdmt tests **** " +fi + rm tmp* From 5a61f36474b7384dc16bdc77f884ef9b0ce41c9f Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Fri, 14 Apr 2017 11:33:04 -0700 Subject: [PATCH 191/305] Make zstd compile with mt by default --- appveyor.yml | 7 +++-- build/VS2005/zstd/zstd.vcproj | 8 ++--- build/VS2005/zstdlib/zstdlib.vcproj | 8 ++--- build/VS2008/zstd/zstd.vcproj | 8 ++--- build/VS2008/zstdlib/zstdlib.vcproj | 8 ++--- build/VS2010/libzstd-dll/libzstd-dll.vcxproj | 8 ++--- build/VS2010/libzstd/libzstd.vcxproj | 8 ++--- build/VS2010/zstd/zstd.vcxproj | 8 ++--- build/cmake/programs/CMakeLists.txt | 9 +++--- programs/Makefile | 32 +++++++++++++++----- programs/zstdcli.c | 2 ++ 11 files changed, 63 insertions(+), 43 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index 5887d52e6..4aad71469 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -15,7 +15,7 @@ - COMPILER: "gcc" HOST: "mingw" PLATFORM: "x86" - SCRIPT: "make lib && make zstd && make -C tests allnothread" + SCRIPT: "make allarch" ARTIFACT: "true" BUILD: "true" - COMPILER: "clang" @@ -191,7 +191,7 @@ - COMPILER: "gcc" HOST: "mingw" PLATFORM: "x86" - SCRIPT: "make lib && make zstd && make -C tests allnothread" + SCRIPT: "make allarch" - COMPILER: "clang" HOST: "mingw" PLATFORM: "x64" @@ -237,7 +237,8 @@ ) ) && make -v && sh -c "%COMPILER% -v" && - sh -c "CC=%COMPILER% %SCRIPT%" + set "CC=%COMPILER%" && + sh -c "%SCRIPT%" ) - if [%HOST%]==[visual] ( ECHO *** && diff --git a/build/VS2005/zstd/zstd.vcproj b/build/VS2005/zstd/zstd.vcproj index 1f4febead..46cabbf6e 100644 --- a/build/VS2005/zstd/zstd.vcproj +++ b/build/VS2005/zstd/zstd.vcproj @@ -44,7 +44,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -121,7 +121,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" @@ -196,7 +196,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -274,7 +274,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" diff --git a/build/VS2005/zstdlib/zstdlib.vcproj b/build/VS2005/zstdlib/zstdlib.vcproj index 8da313673..f77df786f 100644 --- a/build/VS2005/zstdlib/zstdlib.vcproj +++ b/build/VS2005/zstdlib/zstdlib.vcproj @@ -44,7 +44,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -120,7 +120,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" @@ -194,7 +194,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -271,7 +271,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" diff --git a/build/VS2008/zstd/zstd.vcproj b/build/VS2008/zstd/zstd.vcproj index 468d25672..2e2923d55 100644 --- a/build/VS2008/zstd/zstd.vcproj +++ b/build/VS2008/zstd/zstd.vcproj @@ -45,7 +45,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -122,7 +122,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" @@ -197,7 +197,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -275,7 +275,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\lib\dictBuilder;$(SolutionDir)..\..\lib\compress" - PreprocessorDefinitions="ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" diff --git a/build/VS2008/zstdlib/zstdlib.vcproj b/build/VS2008/zstdlib/zstdlib.vcproj index 857e1463e..ac85f7aeb 100644 --- a/build/VS2008/zstdlib/zstdlib.vcproj +++ b/build/VS2008/zstdlib/zstdlib.vcproj @@ -45,7 +45,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -121,7 +121,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" @@ -195,7 +195,7 @@ Name="VCCLCompilerTool" Optimization="0" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE" MinimalRebuild="true" BasicRuntimeChecks="3" RuntimeLibrary="3" @@ -272,7 +272,7 @@ EnableIntrinsicFunctions="true" OmitFramePointers="true" AdditionalIncludeDirectories="$(SolutionDir)..\..\lib;$(SolutionDir)..\..\lib\common;$(SolutionDir)..\..\lib\legacy;$(SolutionDir)..\..\programs\legacy;$(SolutionDir)..\..\lib\dictBuilder" - PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" + PreprocessorDefinitions="ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE" RuntimeLibrary="0" EnableFunctionLevelLinking="true" UsePrecompiledHeader="0" diff --git a/build/VS2010/libzstd-dll/libzstd-dll.vcxproj b/build/VS2010/libzstd-dll/libzstd-dll.vcxproj index 866d04a04..364b3bea5 100644 --- a/build/VS2010/libzstd-dll/libzstd-dll.vcxproj +++ b/build/VS2010/libzstd-dll/libzstd-dll.vcxproj @@ -149,7 +149,7 @@ Level4 Disabled - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) true EnableFastChecks MultiThreadedDebugDLL @@ -169,7 +169,7 @@ Level4 Disabled - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) true EnableFastChecks MultiThreadedDebugDLL @@ -189,7 +189,7 @@ MaxSpeed true true - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) false MultiThreaded ProgramDatabase @@ -211,7 +211,7 @@ MaxSpeed true true - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) false false MultiThreaded diff --git a/build/VS2010/libzstd/libzstd.vcxproj b/build/VS2010/libzstd/libzstd.vcxproj index 186b4c4da..6087d737c 100644 --- a/build/VS2010/libzstd/libzstd.vcxproj +++ b/build/VS2010/libzstd/libzstd.vcxproj @@ -146,7 +146,7 @@ Level4 Disabled - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) true EnableFastChecks MultiThreadedDebugDLL @@ -166,7 +166,7 @@ Level4 Disabled - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) true EnableFastChecks MultiThreadedDebugDLL @@ -186,7 +186,7 @@ MaxSpeed true true - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) false MultiThreaded ProgramDatabase @@ -208,7 +208,7 @@ MaxSpeed true true - ZSTD_DLL_EXPORT=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) + ZSTD_DLL_EXPORT=1;ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;_CRT_SECURE_NO_WARNINGS;%(PreprocessorDefinitions) false false MultiThreaded diff --git a/build/VS2010/zstd/zstd.vcxproj b/build/VS2010/zstd/zstd.vcxproj index 7568e4902..438dc6173 100644 --- a/build/VS2010/zstd/zstd.vcxproj +++ b/build/VS2010/zstd/zstd.vcxproj @@ -155,7 +155,7 @@ Level4 Disabled - ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions) + ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions) true false @@ -171,7 +171,7 @@ Level4 Disabled - ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions) + ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions) true false @@ -189,7 +189,7 @@ MaxSpeed true true - ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions) + ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions) false false MultiThreaded @@ -210,7 +210,7 @@ MaxSpeed true true - ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions) + ZSTD_MULTITHREAD=1;ZSTD_LEGACY_SUPPORT=4;WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions) false false MultiThreaded diff --git a/build/cmake/programs/CMakeLists.txt b/build/cmake/programs/CMakeLists.txt index a481a8e1d..588fea41e 100644 --- a/build/cmake/programs/CMakeLists.txt +++ b/build/cmake/programs/CMakeLists.txt @@ -51,17 +51,16 @@ IF (UNIX) ENDIF (UNIX) IF (ZSTD_MULTITHREAD_SUPPORT) - ADD_EXECUTABLE(zstdmt ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c ${PROGRAMS_DIR}/bench.c ${PROGRAMS_DIR}/datagen.c ${PROGRAMS_DIR}/dibio.c ${PlatformDependResources}) - SET_TARGET_PROPERTIES(zstdmt PROPERTIES COMPILE_DEFINITIONS "ZSTD_MULTITHREAD") - TARGET_LINK_LIBRARIES(zstdmt libzstd_shared) + SET_TARGET_PROPERTIES(zstd PROPERTIES COMPILE_DEFINITIONS "ZSTD_MULTITHREAD") SET(THREADS_PREFER_PTHREAD_FLAG ON) FIND_PACKAGE(Threads REQUIRED) IF (CMAKE_USE_PTHREADS_INIT) - TARGET_LINK_LIBRARIES(zstdmt ${CMAKE_THREAD_LIBS_INIT}) + TARGET_LINK_LIBRARIES(zstd ${CMAKE_THREAD_LIBS_INIT}) ELSE() MESSAGE(SEND_ERROR "ZSTD currently does not support thread libraries other than pthreads") ENDIF() - INSTALL(TARGETS zstdmt RUNTIME DESTINATION "bin") + ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd zstdmt COMMENT "Creating zstdmt symlink") + INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/zstdmt DESTINATION "bin") ENDIF (ZSTD_MULTITHREAD_SUPPORT) diff --git a/programs/Makefile b/programs/Makefile index b8e976b21..3f71a5334 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -82,9 +82,22 @@ else EXT = endif +VOID = /dev/null + +# thread detection +NO_THREAD_MSG := ==> no threads, building without multithreading support +HAVE_PTHREAD := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_pthread$(EXT) -x c - -pthread 2> $(VOID) && rm have_pthread$(EXT) && echo 1 || echo 0) +HAVE_THREAD := $(shell [ "$(HAVE_PTHREAD)" -eq "1" -o -n "$(filter Windows%,$(OS))" ] && echo 1 || echo 0) +ifeq ($(HAVE_THREAD), 1) +THREAD_MSG := ==> building with threading support +THREAD_CPP := -DZSTD_MULTITHREAD +THREAD_LD := -pthread +else +THREAD_MSG := NO_THREAD_MSG +endif + # zlib detection NO_ZLIB_MSG := ==> no zlib, building zstd without .gz support -VOID = /dev/null HAVE_ZLIB := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_zlib$(EXT) -x c - -lz 2> $(VOID) && rm have_zlib$(EXT) && echo 1 || echo 0) ifeq ($(HAVE_ZLIB), 1) ZLIB_MSG := ==> building zstd with .gz compression support @@ -115,8 +128,8 @@ all: zstd $(ZSTDDECOMP_O): CFLAGS += $(ALIGN_LOOP) -zstd : CPPFLAGS += $(ZLIBCPP) -zstd : LDFLAGS += $(ZLIBLD) +zstd : CPPFLAGS += $(ZLIBCPP) $(THREAD_CPP) +zstd : LDFLAGS += $(ZLIBLD) $(THREAD_LD) zstd : LZMA_MSG := $(NO_LZMA_MSG) zstd-nogz : ZLIB_MSG := $(NO_ZLIB_MSG) zstd-nogz : LZMA_MSG := $(NO_LZMA_MSG) @@ -124,6 +137,7 @@ xzstd : CPPFLAGS += $(ZLIBCPP) $(LZMACPP) xzstd : LDFLAGS += $(ZLIBLD) $(LZMALD) zstd zstd-nogz xzstd : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) zstd zstd-nogz xzstd : $(ZSTDLIB_OBJ) zstdcli.o fileio.o bench.o datagen.o dibio.o + @echo "$(THREAD_MSG)" @echo "$(ZLIB_MSG)" @echo "$(LZMA_MSG)" ifneq (,$(filter Windows%,$(OS))) @@ -145,6 +159,11 @@ endif zstd-nolegacy : clean_decomp_o $(MAKE) zstd ZSTD_LEGACY_SUPPORT=0 +zstd-nomt : THREAD_CPP := +zstd-nomt : THREAD_LD := +zstd-nomt : THREAD_MSG := NO_THREAD_MSG +zstd-nomt : zstd + zstd-pgo : MOREFLAGS = -fprofile-generate zstd-pgo : clean zstd ./zstd -b19i1 $(PROFILE_WITH) @@ -169,10 +188,6 @@ zstd-decompress: $(ZSTDCOMMON_FILES) $(ZSTDDECOMP_FILES) zstdcli.c fileio.c zstd-compress: $(ZSTDCOMMON_FILES) $(ZSTDCOMP_FILES) zstdcli.c fileio.c $(CC) $(FLAGS) -DZSTD_NOBENCH -DZSTD_NODICT -DZSTD_NODECOMPRESS $^ -o $@$(EXT) -zstdmt: CPPFLAGS += -DZSTD_MULTITHREAD -ifeq (,$(filter Windows%,$(OS))) -zstdmt: LDFLAGS += -lpthread -endif zstdmt: zstd generate_res: @@ -231,6 +246,9 @@ install: zstd @$(INSTALL_PROGRAM) zstd $(DESTDIR)$(BINDIR)/zstd @ln -sf zstd $(DESTDIR)$(BINDIR)/zstdcat @ln -sf zstd $(DESTDIR)$(BINDIR)/unzstd +ifeq ($(HAVE_THREAD),1) + @ln -sf zstd $(DESTDIR)$(BINDIR)/zstdmt +endif @$(INSTALL_SCRIPT) zstdless $(DESTDIR)$(BINDIR)/zstdless @$(INSTALL_SCRIPT) zstdgrep $(DESTDIR)$(BINDIR)/zstdgrep @echo Installing man pages diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 73b4eab9d..6cbe7383f 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -49,6 +49,7 @@ #define AUTHOR "Yann Collet" #define WELCOME_MESSAGE "*** %s %i-bits %s, by %s ***\n", COMPRESSOR_NAME, (int)(sizeof(size_t)*8), ZSTD_VERSION, AUTHOR +#define ZSTD_ZSTDMT "zstdmt" #define ZSTD_UNZSTD "unzstd" #define ZSTD_CAT "zstdcat" #define ZSTD_GZ "gzip" @@ -343,6 +344,7 @@ int main(int argCount, const char* argv[]) programName = lastNameFromPath(programName); /* preset behaviors */ + if (exeNameMatch(programName, ZSTD_ZSTDMT)) nbThreads=0; if (exeNameMatch(programName, ZSTD_UNZSTD)) operation=zom_decompress; if (exeNameMatch(programName, ZSTD_CAT)) { operation=zom_decompress; forceStdout=1; FIO_overwriteMode(); outFileName=stdoutmark; g_displayLevel=1; } if (exeNameMatch(programName, ZSTD_GZ)) { suffix = GZ_EXTENSION; FIO_setCompressionType(FIO_gzipCompression); FIO_setRemoveSrcFile(1); } /* behave like gzip */ From d845dab69c15896ea2d857266ef889ef52e9f7aa Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 17 Apr 2017 12:10:58 -0700 Subject: [PATCH 192/305] Fix input size too small to trigger zstdmt --- tests/playTests.sh | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/tests/playTests.sh b/tests/playTests.sh index ce1e2582e..deeebee2e 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -469,9 +469,9 @@ fileRoundTripTest -g500K if [ -n "$hasMT" ] then $ECHO "\n**** zstdmt round-trip tests **** " - roundTripTest -g516K "16 -T0" - roundTripTest -g516K "19 -T2" - fileRoundTripTest -g500K " -T2" + roundTripTest -g4M "1 -T0" + roundTripTest -g8M "3 -T2" + fileRoundTripTest -g4M "19 -T2 -B1M" else $ECHO "\n**** no multithreading, skipping zstdmt tests **** " fi From f6ef4db20e3025146a338fc399b48b4f1705fa4d Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 17 Apr 2017 12:21:11 -0700 Subject: [PATCH 193/305] Install zstdmt even without threading support --- programs/Makefile | 2 -- 1 file changed, 2 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index 3f71a5334..aca442644 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -246,9 +246,7 @@ install: zstd @$(INSTALL_PROGRAM) zstd $(DESTDIR)$(BINDIR)/zstd @ln -sf zstd $(DESTDIR)$(BINDIR)/zstdcat @ln -sf zstd $(DESTDIR)$(BINDIR)/unzstd -ifeq ($(HAVE_THREAD),1) @ln -sf zstd $(DESTDIR)$(BINDIR)/zstdmt -endif @$(INSTALL_SCRIPT) zstdless $(DESTDIR)$(BINDIR)/zstdless @$(INSTALL_SCRIPT) zstdgrep $(DESTDIR)$(BINDIR)/zstdgrep @echo Installing man pages From 5935c990a0653abd71727d2cb5ebc5775a792864 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 17 Apr 2017 16:05:20 -0700 Subject: [PATCH 194/305] Add zstdmt and -T0 to man page --- programs/zstd.1 | 7 +++++-- programs/zstd.1.md | 8 ++++++-- 2 files changed, 11 insertions(+), 4 deletions(-) diff --git a/programs/zstd.1 b/programs/zstd.1 index c6b2c2742..b8389042e 100644 --- a/programs/zstd.1 +++ b/programs/zstd.1 @@ -2,12 +2,15 @@ .TH "ZSTD" "1" "April 2017" "zstd 1.1.5" "User Commands" . .SH "NAME" -\fBzstd\fR \- zstd, unzstd, zstdcat \- Compress or decompress \.zst files +\fBzstd\fR \- zstd, zstdmt, unzstd, zstdcat \- Compress or decompress \.zst files . .SH "SYNOPSIS" \fBzstd\fR [\fIOPTIONS\fR] [\-|] [\-o ] . .P +\fBzstdmt\fR is equivalent to \fBzstd \-T0\fR +. +.P \fBunzstd\fR is equivalent to \fBzstd \-d\fR . .P @@ -98,7 +101,7 @@ unlocks high compression levels 20+ (maximum 22), using a lot more memory\. Note . .TP \fB\-T#\fR -Compress using # threads (default: 1)\. This modifier is only available if \fBzstd\fR was compiled with multithreading support\. +Compress using # threads (default: 1)\. If \fB#\fR is 0, attempt to detect the number of physical CPU cores and compress with that many threads\. This modifier is only available if \fBzstd\fR was compiled with multithreading support\. . .TP \fB\-D file\fR diff --git a/programs/zstd.1.md b/programs/zstd.1.md index 07c36f683..9044eb34e 100644 --- a/programs/zstd.1.md +++ b/programs/zstd.1.md @@ -1,11 +1,13 @@ -zstd(1) -- zstd, unzstd, zstdcat - Compress or decompress .zst files -==================================================================== +zstd(1) -- zstd, zstdmt, unzstd, zstdcat - Compress or decompress .zst files +============================================================================ SYNOPSIS -------- `zstd` [*OPTIONS*] [-|<INPUT-FILE>] [-o <OUTPUT-FILE>] +`zstdmt` is equivalent to `zstd -T0` + `unzstd` is equivalent to `zstd -d` `zstdcat` is equivalent to `zstd -dcf` @@ -101,6 +103,8 @@ the last one takes effect. Note that decompression will also require more memory when using these levels. * `-T#`: Compress using # threads (default: 1). + If `#` is 0, attempt to detect the number of physical CPU cores and compress with + that many threads. This modifier is only available if `zstd` was compiled with multithreading support. * `-D file`: use `file` as Dictionary to compress or decompress FILE(s) From c47c68f6ca6f782cd9d24c4b8a11c45553cd1700 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Mon, 17 Apr 2017 16:14:21 -0700 Subject: [PATCH 195/305] proper evaluation of Huffman CTable size --- lib/common/huf.h | 4 +++- lib/compress/zstd_compress.c | 5 ++--- 2 files changed, 5 insertions(+), 4 deletions(-) diff --git a/lib/common/huf.h b/lib/common/huf.h index e5572760a..32313cf74 100644 --- a/lib/common/huf.h +++ b/lib/common/huf.h @@ -121,8 +121,10 @@ size_t HUF_compress4X_wksp (void* dst, size_t dstSize, const void* src, size_t s #define HUF_COMPRESSBOUND(size) (HUF_CTABLEBOUND + HUF_BLOCKBOUND(size)) /* Macro version, useful for static allocation */ /* static allocation of HUF's Compression Table */ +#define HUF_CTABLE_SIZE_U32(maxSymbolValue) ((maxSymbolValue)+1) /* Use tables of U32, for proper alignment */ +#define HUF_CTABLE_SIZE(maxSymbolValue) (HUF_CTABLE_SIZE_U32(maxSymbolValue) * sizeof(U32)) #define HUF_CREATE_STATIC_CTABLE(name, maxSymbolValue) \ - U32 name##hb[maxSymbolValue+1]; \ + U32 name##hb[HUF_CTABLE_SIZE_U32(maxSymbolValue)]; \ void* name##hv = &(name##hb); \ HUF_CElt* name = (HUF_CElt*)(name##hv) /* no final ; */ diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index d6774372b..04791ba95 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -304,7 +304,7 @@ static size_t ZSTD_resetCCtx_advanced (ZSTD_CCtx* zc, zc->hufTable = (HUF_CElt*)ptr; zc->flagStaticTables = 0; zc->flagStaticHufTable = HUF_repeat_none; - ptr = ((U32*)ptr) + 256; /* note : HUF_CElt* is incomplete type, size is simulated using U32 */ + ptr = ((U32*)ptr) + HUF_CTABLE_SIZE_U32(255); /* note : HUF_CElt* is incomplete type, size is simulated using U32 */ zc->nextToUpdate = 1; zc->nextSrc = NULL; @@ -362,7 +362,6 @@ size_t ZSTD_copyCCtx(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx, unsigned long { if (srcCCtx->stage!=ZSTDcs_init) return ERROR(stage_wrong); - memcpy(&dstCCtx->customMem, &srcCCtx->customMem, sizeof(ZSTD_customMem)); { ZSTD_parameters params = srcCCtx->params; params.fParams.contentSizeFlag = (pledgedSrcSize > 0); @@ -397,7 +396,7 @@ size_t ZSTD_copyCCtx(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx, unsigned long memcpy(dstCCtx->offcodeCTable, srcCCtx->offcodeCTable, sizeof(dstCCtx->offcodeCTable)); } if (srcCCtx->flagStaticHufTable) { - memcpy(dstCCtx->hufTable, srcCCtx->hufTable, 256*4); + memcpy(dstCCtx->hufTable, srcCCtx->hufTable, HUF_CTABLE_SIZE(255)); } return 0; From e6c504dbe604ffab1e6f9b68a62face1c4c23758 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 17 Apr 2017 17:11:33 -0700 Subject: [PATCH 196/305] Update -T0 comment in man page --- programs/zstd.1 | 2 +- programs/zstd.1.md | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/programs/zstd.1 b/programs/zstd.1 index b8389042e..b84ab3324 100644 --- a/programs/zstd.1 +++ b/programs/zstd.1 @@ -101,7 +101,7 @@ unlocks high compression levels 20+ (maximum 22), using a lot more memory\. Note . .TP \fB\-T#\fR -Compress using # threads (default: 1)\. If \fB#\fR is 0, attempt to detect the number of physical CPU cores and compress with that many threads\. This modifier is only available if \fBzstd\fR was compiled with multithreading support\. +Compress using # threads (default: 1)\. If \fB#\fR is 0, attempt to detect the number of physical CPU cores and compress with that many threads\. This modified does nothing if \fBzstd\fR was compiled without multithread support\. . .TP \fB\-D file\fR diff --git a/programs/zstd.1.md b/programs/zstd.1.md index 9044eb34e..4b918d8fd 100644 --- a/programs/zstd.1.md +++ b/programs/zstd.1.md @@ -105,7 +105,7 @@ the last one takes effect. Compress using # threads (default: 1). If `#` is 0, attempt to detect the number of physical CPU cores and compress with that many threads. - This modifier is only available if `zstd` was compiled with multithreading support. + This modified does nothing if `zstd` was compiled without multithread support. * `-D file`: use `file` as Dictionary to compress or decompress FILE(s) * `--nodictID`: From c8b2df7d6220f62f4f930e62dc361098b5a002c7 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 17 Apr 2017 17:13:44 -0700 Subject: [PATCH 197/305] Compile CLI using files instead of objs This avoids conflicts between how the library was configured and how the CLI was configured. --- programs/Makefile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/programs/Makefile b/programs/Makefile index aca442644..f3140a19c 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -136,7 +136,7 @@ zstd-nogz : LZMA_MSG := $(NO_LZMA_MSG) xzstd : CPPFLAGS += $(ZLIBCPP) $(LZMACPP) xzstd : LDFLAGS += $(ZLIBLD) $(LZMALD) zstd zstd-nogz xzstd : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) -zstd zstd-nogz xzstd : $(ZSTDLIB_OBJ) zstdcli.o fileio.o bench.o datagen.o dibio.o +zstd zstd-nogz xzstd : $(ZSTDLIB_FILES) zstdcli.o fileio.o bench.o datagen.o dibio.o @echo "$(THREAD_MSG)" @echo "$(ZLIB_MSG)" @echo "$(LZMA_MSG)" From 4f818182b80946c21b4196755228bb7136923176 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Mon, 17 Apr 2017 17:57:35 -0700 Subject: [PATCH 198/305] clarified frame parameters for ZSTD_compress*_usingCDict() created ZSTD_compressBegin_usingCDict_internal(), which gives direct control to frame Parameters. ZSTD_resetCStream_internal() now points into it. --- lib/common/zstd_internal.h | 2 ++ lib/compress/zstd_compress.c | 41 +++++++++++++++++++------------ lib/zstd.h | 7 +++--- tests/fuzzer.c | 7 ------ tests/paramgrill.c | 9 ++++--- zlibWrapper/examples/zwrapbench.c | 8 +++--- 6 files changed, 41 insertions(+), 33 deletions(-) diff --git a/lib/common/zstd_internal.h b/lib/common/zstd_internal.h index 5c5b28732..1e2cbd4d7 100644 --- a/lib/common/zstd_internal.h +++ b/lib/common/zstd_internal.h @@ -58,6 +58,8 @@ /*-************************************* * shared macros ***************************************/ +#undef MIN +#undef MAX #define MIN(a,b) ((a)<(b) ? (a) : (b)) #define MAX(a,b) ((a)>(b) ? (a) : (b)) #define CHECK_F(f) { size_t const errcod = f; if (ERR_isError(errcod)) return errcod; } /* check and Forward error code */ diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 04791ba95..851158f61 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -2911,36 +2911,45 @@ static ZSTD_parameters ZSTD_getParamsFromCDict(const ZSTD_CDict* cdict) { return ZSTD_getParamsFromCCtx(cdict->refContext); } -size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize) +/* ZSTD_compressBegin_usingCDict_internal() : + * cdict must be != NULL */ +static size_t ZSTD_compressBegin_usingCDict_internal( + ZSTD_CCtx* const cctx, const ZSTD_CDict* const cdict, + ZSTD_frameParameters const fParams, unsigned long long const pledgedSrcSize) { if (cdict==NULL) return ERROR(GENERIC); /* does not support NULL cdict */ - if (cdict->dictContentSize) CHECK_F(ZSTD_copyCCtx(cctx, cdict->refContext, pledgedSrcSize)) + if (cdict->dictContentSize) + CHECK_F(ZSTD_copyCCtx(cctx, cdict->refContext, pledgedSrcSize)) /* to be changed, to consider fParams */ else { ZSTD_parameters params = cdict->refContext->params; - params.fParams.contentSizeFlag = (pledgedSrcSize > 0); + params.fParams = fParams; CHECK_F(ZSTD_compressBegin_internal(cctx, NULL, 0, params, pledgedSrcSize)); } return 0; } +/* ZSTD_compressBegin_usingCDict() : + * pledgedSrcSize=0 means "unknown" + * if pledgedSrcSize>0, it will enable contentSizeFlag */ +size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize) +{ + ZSTD_frameParameters fParams = { 1 /*content*/, 0 /*checksum*/, 0 /*noDictID*/ }; + fParams.contentSizeFlag = (pledgedSrcSize > 0); + return ZSTD_compressBegin_usingCDict_internal(cctx, cdict, fParams, pledgedSrcSize); +} + /*! ZSTD_compress_usingCDict() : -* Compression using a digested Dictionary. -* Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. -* Note that compression level is decided during dictionary creation */ + * Compression using a digested Dictionary. + * Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. + * Note that compression parameters are decided at CDict creation time + * while frame parameters are hardcoded */ size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const ZSTD_CDict* cdict) { - CHECK_F(ZSTD_compressBegin_usingCDict(cctx, cdict, srcSize)); /* will check if cdict != NULL */ - - if (cdict->refContext->params.fParams.contentSizeFlag == 1) { - cctx->params.fParams.contentSizeFlag = 1; - cctx->frameContentSize = srcSize; - } else { - cctx->params.fParams.contentSizeFlag = 0; - } - + ZSTD_frameParameters const fParams = { 1 /*content*/, 0 /*checksum*/, 0 /*noDictID*/ }; + CHECK_F (ZSTD_compressBegin_usingCDict_internal(cctx, cdict, fParams, srcSize)); /* will check if cdict != NULL */ return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize); } @@ -3022,7 +3031,7 @@ static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, unsigned long long p { if (zcs->inBuffSize==0) return ERROR(stage_wrong); /* zcs has not been init at least once => can't reset */ - if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict(zcs->cctx, zcs->cdict, pledgedSrcSize)) + if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict_internal(zcs->cctx, zcs->cdict, zcs->params.fParams, pledgedSrcSize)) else CHECK_F(ZSTD_compressBegin_internal(zcs->cctx, NULL, 0, zcs->params, pledgedSrcSize)); zcs->inToCompress = 0; diff --git a/lib/zstd.h b/lib/zstd.h index 6066db45e..18ec1cd7f 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -194,9 +194,10 @@ ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict(const void* dictBuffer, size_t dictSize ZSTDLIB_API size_t ZSTD_freeCDict(ZSTD_CDict* CDict); /*! ZSTD_compress_usingCDict() : -* Compression using a digested Dictionary. -* Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. -* Note that compression level is decided during dictionary creation. */ + * Compression using a digested Dictionary. + * Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. + * Note that compression level is decided during dictionary creation. + * Frame parameters are hardcoded (dictID=yes, contentSize=yes, checksum=no) */ ZSTDLIB_API size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, diff --git a/tests/fuzzer.c b/tests/fuzzer.c index ee051be3e..3123d1825 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -406,7 +406,6 @@ static int basicUnitTests(U32 seed, double compressibility) DISPLAYLEVEL(4, "test%3i : compress with preprocessed dictionary : ", testNb++); { ZSTD_parameters params = ZSTD_getParams(1, CNBuffSize, dictSize); - params.fParams.contentSizeFlag = 0; { ZSTD_customMem customMem = { NULL, NULL, NULL }; ZSTD_CDict* cdict = ZSTD_createCDict_advanced(dictBuffer, dictSize, 1, params, customMem); cSize = ZSTD_compress_usingCDict(cctx, compressedBuffer, ZSTD_compressBound(CNBuffSize), @@ -423,12 +422,6 @@ static int basicUnitTests(U32 seed, double compressibility) } DISPLAYLEVEL(4, "OK \n"); - DISPLAYLEVEL(4, "test%3i : frame should not have content size : ", testNb++); - { unsigned long long const contentSize = ZSTD_findDecompressedSize(compressedBuffer, cSize); - if (contentSize != ZSTD_CONTENTSIZE_UNKNOWN) goto _output_error; /* cdict contentSizeFlag not used */ - } - DISPLAYLEVEL(4, "OK \n"); - DISPLAYLEVEL(4, "test%3i : frame built with dictionary should be decompressible : ", testNb++); CHECKPLUS(r, ZSTD_decompress_usingDict(dctx, decodedBuffer, CNBuffSize, diff --git a/tests/paramgrill.c b/tests/paramgrill.c index 3c1f0150b..1913b54d0 100644 --- a/tests/paramgrill.c +++ b/tests/paramgrill.c @@ -58,6 +58,11 @@ static const int g_maxNbVariations = 64; **************************************/ #define DISPLAY(...) fprintf(stderr, __VA_ARGS__) +#undef MIN +#undef MAX +#define MIN(a,b) ( (a) < (b) ? (a) : (b) ) +#define MAX(a,b) ( (a) > (b) ? (a) : (b) ) + /*-************************************ * Benchmark Parameters @@ -145,8 +150,6 @@ typedef struct } blockParam_t; -#define MIN(a,b) ( (a) < (b) ? (a) : (b) ) - static size_t BMK_benchParam(BMK_result_t* resultPtr, const void* srcBuffer, size_t srcSize, ZSTD_CCtx* ctx, @@ -513,8 +516,6 @@ static BYTE g_alreadyTested[PARAMTABLESIZE] = {0}; /* init to zero */ g_alreadyTested[(XXH64(sanitizeParams(p), sizeof(p), 0) >> 3) & PARAMTABLEMASK] -#define MAX(a,b) ( (a) > (b) ? (a) : (b) ) - static void playAround(FILE* f, winnerInfo_t* winners, ZSTD_compressionParameters params, const void* srcBuffer, size_t srcSize, diff --git a/zlibWrapper/examples/zwrapbench.c b/zlibWrapper/examples/zwrapbench.c index 1c3391e90..401daa1b5 100644 --- a/zlibWrapper/examples/zwrapbench.c +++ b/zlibWrapper/examples/zwrapbench.c @@ -128,6 +128,11 @@ void BMK_SetBlockSize(size_t blockSize) /* ******************************************************** * Bench functions **********************************************************/ +#undef MIN +#undef MAX +#define MIN(a,b) ((a)<(b) ? (a) : (b)) +#define MAX(a,b) ((a)>(b) ? (a) : (b)) + typedef struct { z_const char* srcPtr; @@ -142,9 +147,6 @@ typedef struct typedef enum { BMK_ZSTD, BMK_ZSTD_STREAM, BMK_ZLIB, BMK_ZWRAP_ZLIB, BMK_ZWRAP_ZSTD, BMK_ZLIB_REUSE, BMK_ZWRAP_ZLIB_REUSE, BMK_ZWRAP_ZSTD_REUSE } BMK_compressor; -#define MIN(a,b) ((a)<(b) ? (a) : (b)) -#define MAX(a,b) ((a)>(b) ? (a) : (b)) - static int BMK_benchMem(z_const void* srcBuffer, size_t srcSize, const char* displayName, int cLevel, const size_t* fileSizes, U32 nbFiles, From af4f45b68243748b7c16c09a7a1cb4529363b054 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 18 Apr 2017 03:17:44 -0700 Subject: [PATCH 199/305] Improved code comments for block functions --- lib/zstd.h | 19 ++++++++++--------- 1 file changed, 10 insertions(+), 9 deletions(-) diff --git a/lib/zstd.h b/lib/zstd.h index 18ec1cd7f..2f195636a 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -754,19 +754,20 @@ ZSTDLIB_API ZSTD_nextInputType_e ZSTD_nextInputType(ZSTD_DCtx* dctx); - Compressing and decompressing require a context structure + Use ZSTD_createCCtx() and ZSTD_createDCtx() - It is necessary to init context before starting - + compression : ZSTD_compressBegin() - + decompression : ZSTD_decompressBegin() - + variants _usingDict() are also allowed - + copyCCtx() and copyDCtx() work too - - Block size is limited, it must be <= ZSTD_getBlockSizeMax() - + If you need to compress more, cut data into multiple blocks - + Consider using the regular ZSTD_compress() instead, as frame metadata costs become negligible when source size is large. + + compression : any ZSTD_compressBegin*() variant, including with dictionary + + decompression : any ZSTD_decompressBegin*() variant, including with dictionary + + copyCCtx() and copyDCtx() can be used too + - Block size is limited, it must be <= ZSTD_getBlockSizeMax() <= ZSTD_BLOCKSIZE_ABSOLUTEMAX + + If input is larger than a block size, it's necessary to split input data into multiple blocks + + For inputs larger than a single block size, consider using the regular ZSTD_compress() instead. + Frame metadata is not that costly, and quickly becomes negligible as source size grows larger. - When a block is considered not compressible enough, ZSTD_compressBlock() result will be zero. In which case, nothing is produced into `dst`. + User must test for such outcome and deal directly with uncompressed data + ZSTD_decompressBlock() doesn't accept uncompressed data as input !!! - + In case of multiple successive blocks, decoder must be informed of uncompressed block existence to follow proper history. - Use ZSTD_insertBlock() in such a case. + + In case of multiple successive blocks, should some of them be uncompressed, + decoder must be informed of their existence in order to follow proper history. + Use ZSTD_insertBlock() for such a case. */ #define ZSTD_BLOCKSIZE_ABSOLUTEMAX (128 * 1024) /* define, for static allocation */ From 9606256a8d4c24ecf93740d37d55e14676122626 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 18 Apr 2017 13:52:00 -0700 Subject: [PATCH 200/305] Fix no thread message --- programs/Makefile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index f3140a19c..4ab4dfd62 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -93,7 +93,7 @@ THREAD_MSG := ==> building with threading support THREAD_CPP := -DZSTD_MULTITHREAD THREAD_LD := -pthread else -THREAD_MSG := NO_THREAD_MSG +THREAD_MSG := $(NO_THREAD_MSG) endif # zlib detection @@ -161,7 +161,7 @@ zstd-nolegacy : clean_decomp_o zstd-nomt : THREAD_CPP := zstd-nomt : THREAD_LD := -zstd-nomt : THREAD_MSG := NO_THREAD_MSG +zstd-nomt : THREAD_MSG := $(NO_THREAD_MSG) zstd-nomt : zstd zstd-pgo : MOREFLAGS = -fprofile-generate From 715b9aa113e4c9d7953b0cbeb1756d0ea3572f79 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 18 Apr 2017 13:55:53 -0700 Subject: [PATCH 201/305] created ZSTD_compressBegin_usingCDict_advanced() --- doc/zstd_manual.html | 31 ++++++++++++++++--------------- lib/compress/zstd_compress.c | 10 +++++----- lib/zstd.h | 4 +++- 3 files changed, 24 insertions(+), 21 deletions(-) diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index a5b66f7d3..d267cfc3c 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -169,9 +169,10 @@ size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx); void* dst, size_t dstCapacity, const void* src, size_t srcSize, const ZSTD_CDict* cdict); -

    Compression using a digested Dictionary. - Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. - Note that compression level is decided during dictionary creation. +

    Compression using a digested Dictionary. + Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times. + Note that compression level is decided during dictionary creation. + Frame parameters are hardcoded (dictID=yes, contentSize=yes, checksum=no)


ZSTD_DDict* ZSTD_createDDict(const void* dictBuffer, size_t dictSize);
@@ -560,10 +561,9 @@ size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds);
 

Buffer-less streaming compression functions

size_t ZSTD_compressBegin(ZSTD_CCtx* cctx, int compressionLevel);
 size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel);
 size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */
-size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**<  note: if pledgedSrcSize can be 0, indicating unknown size.  if it is non-zero, it must be accurate.  for 0 size frames, use compressBegin_advanced */
 size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: fail if cdict==NULL. pledgedSrcSize can be 0, indicating unknown size.  For 0 size frames, use compressBegin_advanced */
-size_t ZSTD_compressContinue(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
-size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
+size_t ZSTD_compressBegin_usingCDict_advanced(ZSTD_CCtx* const cctx, const ZSTD_CDict* const cdict, ZSTD_frameParameters const fParams, unsigned long long const pledgedSrcSize);   /* compression parameters are already set within cdict. pledgedSrcSize=0 means null-size */
+size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**<  note: if pledgedSrcSize can be 0, indicating unknown size.  if it is non-zero, it must be accurate.  for 0 size frames, use compressBegin_advanced */
 

Buffer-less streaming decompression (synchronous mode)

   A ZSTD_DCtx object is required to track streaming operations.
@@ -648,19 +648,20 @@ ZSTD_nextInputType_e ZSTD_nextInputType(ZSTD_DCtx* dctx);
     - Compressing and decompressing require a context structure
       + Use ZSTD_createCCtx() and ZSTD_createDCtx()
     - It is necessary to init context before starting
-      + compression : ZSTD_compressBegin()
-      + decompression : ZSTD_decompressBegin()
-      + variants _usingDict() are also allowed
-      + copyCCtx() and copyDCtx() work too
-    - Block size is limited, it must be <= ZSTD_getBlockSizeMax()
-      + If you need to compress more, cut data into multiple blocks
-      + Consider using the regular ZSTD_compress() instead, as frame metadata costs become negligible when source size is large.
+      + compression : any ZSTD_compressBegin*() variant, including with dictionary
+      + decompression : any ZSTD_decompressBegin*() variant, including with dictionary
+      + copyCCtx() and copyDCtx() can be used too
+    - Block size is limited, it must be <= ZSTD_getBlockSizeMax() <= ZSTD_BLOCKSIZE_ABSOLUTEMAX
+      + If input is larger than a block size, it's necessary to split input data into multiple blocks
+      + For inputs larger than a single block size, consider using the regular ZSTD_compress() instead.
+        Frame metadata is not that costly, and quickly becomes negligible as source size grows larger.
     - When a block is considered not compressible enough, ZSTD_compressBlock() result will be zero.
       In which case, nothing is produced into `dst`.
       + User must test for such outcome and deal directly with uncompressed data
       + ZSTD_decompressBlock() doesn't accept uncompressed data as input !!!
-      + In case of multiple successive blocks, decoder must be informed of uncompressed block existence to follow proper history.
-        Use ZSTD_insertBlock() in such a case.
+      + In case of multiple successive blocks, should some of them be uncompressed,
+        decoder must be informed of their existence in order to follow proper history.
+        Use ZSTD_insertBlock() for such a case.
 

Raw zstd block functions

size_t ZSTD_getBlockSizeMax(ZSTD_CCtx* cctx);
diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 851158f61..33d4ab005 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -2911,9 +2911,9 @@ static ZSTD_parameters ZSTD_getParamsFromCDict(const ZSTD_CDict* cdict) {
     return ZSTD_getParamsFromCCtx(cdict->refContext);
 }
 
-/* ZSTD_compressBegin_usingCDict_internal() :
+/* ZSTD_compressBegin_usingCDict_advanced() :
  * cdict must be != NULL */
-static size_t ZSTD_compressBegin_usingCDict_internal(
+size_t ZSTD_compressBegin_usingCDict_advanced(
     ZSTD_CCtx* const cctx, const ZSTD_CDict* const cdict,
     ZSTD_frameParameters const fParams, unsigned long long const pledgedSrcSize)
 {
@@ -2935,7 +2935,7 @@ size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, u
 {
     ZSTD_frameParameters fParams = { 1 /*content*/, 0 /*checksum*/, 0 /*noDictID*/ };
     fParams.contentSizeFlag = (pledgedSrcSize > 0);
-    return ZSTD_compressBegin_usingCDict_internal(cctx, cdict, fParams, pledgedSrcSize);
+    return ZSTD_compressBegin_usingCDict_advanced(cctx, cdict, fParams, pledgedSrcSize);
 }
 
 /*! ZSTD_compress_usingCDict() :
@@ -2949,7 +2949,7 @@ size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx,
                                 const ZSTD_CDict* cdict)
 {
     ZSTD_frameParameters const fParams = { 1 /*content*/, 0 /*checksum*/, 0 /*noDictID*/ };
-    CHECK_F (ZSTD_compressBegin_usingCDict_internal(cctx, cdict, fParams, srcSize));   /* will check if cdict != NULL */
+    CHECK_F (ZSTD_compressBegin_usingCDict_advanced(cctx, cdict, fParams, srcSize));   /* will check if cdict != NULL */
     return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize);
 }
 
@@ -3031,7 +3031,7 @@ static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, unsigned long long p
 {
     if (zcs->inBuffSize==0) return ERROR(stage_wrong);   /* zcs has not been init at least once => can't reset */
 
-    if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict_internal(zcs->cctx, zcs->cdict, zcs->params.fParams, pledgedSrcSize))
+    if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict_advanced(zcs->cctx, zcs->cdict, zcs->params.fParams, pledgedSrcSize))
     else CHECK_F(ZSTD_compressBegin_internal(zcs->cctx, NULL, 0, zcs->params, pledgedSrcSize));
 
     zcs->inToCompress = 0;
diff --git a/lib/zstd.h b/lib/zstd.h
index 2f195636a..5101fd26c 100644
--- a/lib/zstd.h
+++ b/lib/zstd.h
@@ -659,8 +659,10 @@ ZSTDLIB_API size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds);
 ZSTDLIB_API size_t ZSTD_compressBegin(ZSTD_CCtx* cctx, int compressionLevel);
 ZSTDLIB_API size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel);
 ZSTDLIB_API size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */
-ZSTDLIB_API size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**<  note: if pledgedSrcSize can be 0, indicating unknown size.  if it is non-zero, it must be accurate.  for 0 size frames, use compressBegin_advanced */
 ZSTDLIB_API size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: fail if cdict==NULL. pledgedSrcSize can be 0, indicating unknown size.  For 0 size frames, use compressBegin_advanced */
+ZSTDLIB_API size_t ZSTD_compressBegin_usingCDict_advanced(ZSTD_CCtx* const cctx, const ZSTD_CDict* const cdict, ZSTD_frameParameters const fParams, unsigned long long const pledgedSrcSize);   /* compression parameters are already set within cdict. pledgedSrcSize=0 means null-size */
+ZSTDLIB_API size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**<  note: if pledgedSrcSize can be 0, indicating unknown size.  if it is non-zero, it must be accurate.  for 0 size frames, use compressBegin_advanced */
+
 ZSTDLIB_API size_t ZSTD_compressContinue(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
 ZSTDLIB_API size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
 

From 30fb499208e650ba3e5f28bafde788fda474dfff Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Tue, 18 Apr 2017 14:08:50 -0700
Subject: [PATCH 202/305] Changed ZSTD_resetCCtx_advanced() into
 ZSTD_resetCCtx_internal()

for naming consistency :
_advanced() can be invoked
while _internal() are strictly static
---
 lib/compress/zstd_compress.c | 8 ++++----
 1 file changed, 4 insertions(+), 4 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 33d4ab005..9328b7c40 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -257,9 +257,9 @@ static size_t ZSTD_continueCCtx(ZSTD_CCtx* cctx, ZSTD_parameters params, U64 fra
 
 typedef enum { ZSTDcrp_continue, ZSTDcrp_noMemset, ZSTDcrp_fullReset } ZSTD_compResetPolicy_e;
 
-/*! ZSTD_resetCCtx_advanced() :
+/*! ZSTD_resetCCtx_internal() :
     note : `params` must be validated */
-static size_t ZSTD_resetCCtx_advanced (ZSTD_CCtx* zc,
+static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
                                        ZSTD_parameters params, U64 frameContentSize,
                                        ZSTD_compResetPolicy_e const crp)
 {
@@ -365,7 +365,7 @@ size_t ZSTD_copyCCtx(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx, unsigned long
     memcpy(&dstCCtx->customMem, &srcCCtx->customMem, sizeof(ZSTD_customMem));
     {   ZSTD_parameters params = srcCCtx->params;
         params.fParams.contentSizeFlag = (pledgedSrcSize > 0);
-        ZSTD_resetCCtx_advanced(dstCCtx, params, pledgedSrcSize, ZSTDcrp_noMemset);
+        ZSTD_resetCCtx_internal(dstCCtx, params, pledgedSrcSize, ZSTDcrp_noMemset);
     }
 
     /* copy tables */
@@ -2694,7 +2694,7 @@ static size_t ZSTD_compressBegin_internal(ZSTD_CCtx* cctx,
 {
     ZSTD_compResetPolicy_e const crp = dictSize ? ZSTDcrp_fullReset : ZSTDcrp_continue;
     assert(!ZSTD_isError(ZSTD_checkCParams(params.cParams)));
-    CHECK_F(ZSTD_resetCCtx_advanced(cctx, params, pledgedSrcSize, crp));
+    CHECK_F(ZSTD_resetCCtx_internal(cctx, params, pledgedSrcSize, crp));
     return ZSTD_compress_insertDictionary(cctx, dict, dictSize);
 }
 

From ca6fae78080d0a532ce90be830584212c7350fc2 Mon Sep 17 00:00:00 2001
From: Sean Purcell 
Date: Tue, 18 Apr 2017 14:13:01 -0700
Subject: [PATCH 203/305] Add MT enabled targets for libzstd

---
 lib/Makefile | 12 +++++++++++-
 1 file changed, 11 insertions(+), 1 deletion(-)

diff --git a/lib/Makefile b/lib/Makefile
index 197fdeeea..d8d8e179d 100644
--- a/lib/Makefile
+++ b/lib/Makefile
@@ -71,6 +71,9 @@ libzstd.a: $(ZSTD_OBJ)
 	@echo compiling static library
 	@$(AR) $(ARFLAGS) $@ $^
 
+libzstd.a-mt: CPPFLAGS += -DZSTD_MULTHREAD
+libzstd.a-mt: libzstd.a
+
 $(LIBZSTD): LDFLAGS += -shared -fPIC -fvisibility=hidden
 $(LIBZSTD): $(ZSTD_FILES)
 	@echo compiling dynamic library $(LIBVER)
@@ -86,10 +89,17 @@ endif
 
 libzstd : $(LIBZSTD)
 
+libzstd-mt : CPPFLAGS += -DZSTD_MULTITHREAD
+libzstd-mt : libzstd
+
 lib: libzstd.a libzstd
 
-lib-release: DEBUGFLAGS :=
+lib-mt: CPPFLAGS += -DZSTD_MULTITHREAD
+lib-mt: lib
+
+lib-release lib-release-mt: DEBUGFLAGS :=
 lib-release: lib
+lib-release-mt: lib-mt
 
 clean:
 	@$(RM) -r *.dSYM   # Mac OS-X specific

From b402f1ef37b76a44803da8ed6104d8e01ce8f223 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Tue, 18 Apr 2017 14:34:24 -0700
Subject: [PATCH 204/305] added make list

---
 Makefile | 7 ++++++-
 1 file changed, 6 insertions(+), 1 deletion(-)

diff --git a/Makefile b/Makefile
index 7e57e1680..dc2ac4ce0 100644
--- a/Makefile
+++ b/Makefile
@@ -109,10 +109,15 @@ clean:
 # make install is validated only for Linux, OSX, Hurd and some BSD targets
 #------------------------------------------------------------------------------
 ifneq (,$(filter $(shell uname),Linux Darwin GNU/kFreeBSD GNU FreeBSD DragonFly NetBSD))
+
 HOST_OS = POSIX
 CMAKE_PARAMS = -DZSTD_BUILD_CONTRIB:BOOL=ON -DZSTD_BUILD_STATIC:BOOL=ON -DZSTD_BUILD_TESTS:BOOL=ON
-.PHONY: install uninstall travis-install clangtest gpptest armtest usan asan uasan
 
+.PHONY: list
+list:
+	@$(MAKE) -pRrq -f $(lastword $(MAKEFILE_LIST)) : 2>/dev/null | awk -v RS= -F: '/^# File/,/^# Finished Make data base/ {if ($$1 !~ "^[#.]") {print $$1}}' | sort | egrep -v -e '^[^[:alnum:]]' -e '^$@$$' | xargs
+
+.PHONY: install uninstall travis-install clangtest gpptest armtest usan asan uasan
 install:
 	@$(MAKE) -C $(ZSTDDIR) $@
 	@$(MAKE) -C $(PRGDIR) $@

From a4cab801835561ef63a8005482b8090f81c14aca Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Tue, 18 Apr 2017 14:54:54 -0700
Subject: [PATCH 205/305] added ZSTD_copyCCtx_internal()

which respects provided fParams.
---
 lib/compress/zstd_compress.c | 32 ++++++++++++++++++++++++--------
 1 file changed, 24 insertions(+), 8 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 9328b7c40..9d9cb394f 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -354,17 +354,20 @@ void ZSTD_invalidateRepCodes(ZSTD_CCtx* cctx) {
     for (i=0; irep[i] = 0;
 }
 
-/*! ZSTD_copyCCtx() :
-*   Duplicate an existing context `srcCCtx` into another one `dstCCtx`.
-*   Only works during stage ZSTDcs_init (i.e. after creation, but before first call to ZSTD_compressContinue()).
-*   @return : 0, or an error code */
-size_t ZSTD_copyCCtx(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx, unsigned long long pledgedSrcSize)
+
+/*! ZSTD_copyCCtx_internal() :
+ *  Duplicate an existing context `srcCCtx` into another one `dstCCtx`.
+ *  Only works during stage ZSTDcs_init (i.e. after creation, but before first call to ZSTD_compressContinue()).
+ *  pledgedSrcSize=0 means "empty" if fParams.contentSizeFlag=1
+ *  @return : 0, or an error code */
+size_t ZSTD_copyCCtx_internal(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx,
+                              ZSTD_frameParameters fParams, unsigned long long pledgedSrcSize)
 {
     if (srcCCtx->stage!=ZSTDcs_init) return ERROR(stage_wrong);
 
     memcpy(&dstCCtx->customMem, &srcCCtx->customMem, sizeof(ZSTD_customMem));
     {   ZSTD_parameters params = srcCCtx->params;
-        params.fParams.contentSizeFlag = (pledgedSrcSize > 0);
+        params.fParams = fParams;
         ZSTD_resetCCtx_internal(dstCCtx, params, pledgedSrcSize, ZSTDcrp_noMemset);
     }
 
@@ -402,9 +405,22 @@ size_t ZSTD_copyCCtx(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx, unsigned long
     return 0;
 }
 
+/*! ZSTD_copyCCtx() :
+ *  Duplicate an existing context `srcCCtx` into another one `dstCCtx`.
+ *  Only works during stage ZSTDcs_init (i.e. after creation, but before first call to ZSTD_compressContinue()).
+ *  pledgedSrcSize==0 means "unknown".
+*   @return : 0, or an error code */
+size_t ZSTD_copyCCtx(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx, unsigned long long pledgedSrcSize)
+{
+    ZSTD_frameParameters fParams = { 1 /*content*/, 0 /*checksum*/, 0 /*noDictID*/ };
+    fParams.contentSizeFlag = pledgedSrcSize>0;
+
+    return ZSTD_copyCCtx_internal(dstCCtx, srcCCtx, fParams, pledgedSrcSize);
+}
+
 
 /*! ZSTD_reduceTable() :
-*   reduce table indexes by `reducerValue` */
+ *  reduce table indexes by `reducerValue` */
 static void ZSTD_reduceTable (U32* const table, U32 const size, U32 const reducerValue)
 {
     U32 u;
@@ -2919,7 +2935,7 @@ size_t ZSTD_compressBegin_usingCDict_advanced(
 {
     if (cdict==NULL) return ERROR(GENERIC);  /* does not support NULL cdict */
     if (cdict->dictContentSize)
-        CHECK_F(ZSTD_copyCCtx(cctx, cdict->refContext, pledgedSrcSize))  /* to be changed, to consider fParams */
+        CHECK_F( ZSTD_copyCCtx_internal(cctx, cdict->refContext, fParams, pledgedSrcSize) )
     else {
         ZSTD_parameters params = cdict->refContext->params;
         params.fParams = fParams;

From 0bb381dad88105946728d04c3ea45c19204c2636 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Tue, 18 Apr 2017 15:08:52 -0700
Subject: [PATCH 206/305] added test for ZSTD_initCStream_advanced()

when params.fParams.contentSizeFlags = 1
and pledgedSrcSize = 0
then srcSize should be considered 0 (empty), and not "unknown"
---
 tests/zstreamtest.c | 19 ++++++++++++++++++-
 1 file changed, 18 insertions(+), 1 deletion(-)

diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c
index ac200a7a6..6673208b0 100644
--- a/tests/zstreamtest.c
+++ b/tests/zstreamtest.c
@@ -479,7 +479,24 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
         DISPLAYLEVEL(3, "OK (%s)\n", ZSTD_getErrorName(r));
     }
 
-    /* Unknown srcSize */
+    /* Empty srcSize */
+    DISPLAYLEVEL(3, "test%3i : ZSTD_initCStream_advanced with pledgedSrcSize=0 and dict : ", testNb++);
+    {   ZSTD_parameters params = ZSTD_getParams(5, 0, 0);
+        params.fParams.contentSizeFlag = 1;
+        ZSTD_initCStream_advanced(zc, dictionary.start, dictionary.filled, params, 0);
+    } /* cstream advanced shall write content size = 0 */
+    inBuff.src = CNBuffer;
+    inBuff.size = 0;
+    inBuff.pos = 0;
+    outBuff.dst = compressedBuffer;
+    outBuff.size = compressedBufferSize;
+    outBuff.pos = 0;
+    if (ZSTD_isError(ZSTD_compressStream(zc, &outBuff, &inBuff))) goto _output_error;
+    if (ZSTD_endStream(zc, &outBuff) != 0) goto _output_error;
+    cSize = outBuff.pos;
+    if (ZSTD_findDecompressedSize(compressedBuffer, cSize) != 0) goto _output_error;
+    DISPLAYLEVEL(3, "OK \n");
+
     DISPLAYLEVEL(3, "test%3i : pledgedSrcSize == 0 behaves properly : ", testNb++);
     {   ZSTD_parameters params = ZSTD_getParams(5, 0, 0);
         params.fParams.contentSizeFlag = 1;

From 0f7bd772e62866077cb3d72665b600de3a7bae57 Mon Sep 17 00:00:00 2001
From: Sean Purcell 
Date: Tue, 18 Apr 2017 16:47:28 -0700
Subject: [PATCH 207/305] Update seekable API to simplify IO

---
 .../examples/seekable_decompression.c         |  72 +--
 contrib/seekable_format/zstd_seekable.h       | 109 ++--
 contrib/seekable_format/zstdseek_decompress.c | 538 ++++++++++--------
 lib/common/error_private.c                    |   2 +-
 lib/common/zstd_errors.h                      |   2 +-
 5 files changed, 397 insertions(+), 326 deletions(-)

diff --git a/contrib/seekable_format/examples/seekable_decompression.c b/contrib/seekable_format/examples/seekable_decompression.c
index b134b87b6..d18def7cd 100644
--- a/contrib/seekable_format/examples/seekable_decompression.c
+++ b/contrib/seekable_format/examples/seekable_decompression.c
@@ -17,6 +17,7 @@
 
 #include "zstd_seekable.h"
 
+#define MIN(a, b) ((a) < (b) ? (a) : (b))
 
 static void* malloc_orDie(size_t size)
 {
@@ -85,74 +86,31 @@ static void fseek_orDie(FILE* file, long int offset, int origin) {
 static void decompressFile_orDie(const char* fname, unsigned startOffset, unsigned endOffset)
 {
     FILE* const fin  = fopen_orDie(fname, "rb");
-    size_t const buffInSize = ZSTD_DStreamInSize();
-    void*  const buffIn  = malloc_orDie(buffInSize);
     FILE* const fout = stdout;
     size_t const buffOutSize = ZSTD_DStreamOutSize();  /* Guarantee to successfully flush at least one complete compressed block in all circumstances. */
     void*  const buffOut = malloc_orDie(buffOutSize);
 
-    ZSTD_seekable_DStream* const dstream = ZSTD_seekable_createDStream();
-    if (dstream==NULL) { fprintf(stderr, "ZSTD_seekable_createDStream() error \n"); exit(10); }
+    ZSTD_seekable* const seekable = ZSTD_seekable_create();
+    if (seekable==NULL) { fprintf(stderr, "ZSTD_seekable_create() error \n"); exit(10); }
 
-    {   size_t sizeNeeded = 0;
-        void* buffSeekTable = NULL;
+    size_t const initResult = ZSTD_seekable_initFile(seekable, fin);
+    if (ZSTD_isError(initResult)) { fprintf(stderr, "ZSTD_seekable_init() error : %s \n", ZSTD_getErrorName(initResult)); exit(11); }
 
-        do {
-            sizeNeeded = ZSTD_seekable_loadSeekTable(dstream, buffSeekTable, sizeNeeded);
-            if (!sizeNeeded) break;
+    while (startOffset < endOffset) {
+        size_t const result = ZSTD_seekable_decompress(seekable, buffOut, MIN(endOffset - startOffset, buffOutSize), startOffset);
 
-            if (ZSTD_isError(sizeNeeded)) {
-                fprintf(stderr, "ZSTD_seekable_loadSeekTable() error : %s \n",
-                        ZSTD_getErrorName(sizeNeeded));
-                exit(11);
-            }
-
-            fseek_orDie(fin, -(long) sizeNeeded, SEEK_END);
-            buffSeekTable = realloc_orDie(buffSeekTable, sizeNeeded);
-            fread_orDie(buffSeekTable, sizeNeeded, fin);
-        } while (sizeNeeded > 0);
-
-        free(buffSeekTable);
+        if (ZSTD_isError(result)) {
+            fprintf(stderr, "ZSTD_seekable_decompress() error : %s \n",
+                    ZSTD_getErrorName(result));
+            exit(12);
+        }
+        fwrite_orDie(buffOut, result, fout);
+        startOffset += result;
     }
 
-    /* In more complex scenarios, a file may consist of multiple appended frames (ex : pzstd).
-    *  The following example decompresses only the first frame.
-    *  It is compatible with other provided streaming examples */
-    size_t const initResult = ZSTD_seekable_initDStream(dstream, startOffset, endOffset);
-    if (ZSTD_isError(initResult)) { fprintf(stderr, "ZSTD_seekable_initDStream() error : %s \n", ZSTD_getErrorName(initResult)); exit(11); }
-
-    size_t result, read, toRead = 0;
-
-    do {
-        read = fread_orDie(buffIn, toRead, fin);
-        {   ZSTD_inBuffer input = { buffIn, read, 0 };
-            ZSTD_outBuffer output = { buffOut, buffOutSize, 0 };
-            result = ZSTD_seekable_decompressStream(dstream, &output, &input);
-
-            if (ZSTD_isError(result)) {
-                if (ZSTD_getErrorCode(result) == ZSTD_error_needSeek) {
-                    unsigned long long const offset = ZSTD_seekable_getSeekOffset(dstream);
-                    fseek_orDie(fin, offset, SEEK_SET);
-                    ZSTD_seekable_updateOffset(dstream, offset);
-                    toRead = 0;
-                } else {
-                    fprintf(stderr,
-                            "ZSTD_seekable_decompressStream() error : %s \n",
-                            ZSTD_getErrorName(result));
-                    exit(12);
-                }
-            } else {
-                toRead = result;
-            }
-            fwrite_orDie(buffOut, output.pos, fout);
-            if (toRead > buffInSize) toRead = buffInSize;
-        }
-    } while (result > 0);
-
-    ZSTD_seekable_freeDStream(dstream);
+    ZSTD_seekable_free(seekable);
     fclose_orDie(fin);
     fclose_orDie(fout);
-    free(buffIn);
     free(buffOut);
 }
 
diff --git a/contrib/seekable_format/zstd_seekable.h b/contrib/seekable_format/zstd_seekable.h
index 3ab4f185e..54b50fa50 100644
--- a/contrib/seekable_format/zstd_seekable.h
+++ b/contrib/seekable_format/zstd_seekable.h
@@ -5,6 +5,8 @@
 extern "C" {
 #endif
 
+#include 
+
 static const unsigned ZSTD_seekTableFooterSize = 9;
 
 #define ZSTD_SEEKABLE_MAGICNUMBER 0x8F92EAB1
@@ -14,6 +16,8 @@ static const unsigned ZSTD_seekTableFooterSize = 9;
 /* Limit the maximum size to avoid any potential issues storing the compressed size */
 #define ZSTD_SEEKABLE_MAX_FRAME_DECOMPRESSED_SIZE 0x80000000U
 
+#define ZSTD_SEEKABLE_FRAMEINDEX_TOOLARGE (0ULL-2)
+
 /*-****************************************************************************
 *  Seekable Format
 *
@@ -24,7 +28,7 @@ static const unsigned ZSTD_seekTableFooterSize = 9;
 ******************************************************************************/
 
 typedef struct ZSTD_seekable_CStream_s ZSTD_seekable_CStream;
-typedef struct ZSTD_seekable_DStream_s ZSTD_seekable_DStream;
+typedef struct ZSTD_seekable_s ZSTD_seekable;
 
 /*-****************************************************************************
 *  Seekable compression - HowTo
@@ -82,55 +86,76 @@ ZSTDLIB_API size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outB
 
 /*-****************************************************************************
 *  Seekable decompression - HowTo
-*  A ZSTD_seekable_DStream object is required to tracking streaming operation.
-*  Use ZSTD_seekable_createDStream() and ZSTD_seekable_freeDStream() to create/
-*  release resources.
+*  A ZSTD_seekable object is required to tracking the seekTable.
 *
-*  Streaming objects are reusable to avoid allocation and deallocation,
-*  to start a new compression operation call ZSTD_seekable_initDStream() on the
-*  compressor.
+*  Call ZSTD_seekable_init* to initialize a ZSTD_seekable object with the
+*  the seek table provided in the input.
+*  There are three modes for ZSTD_seekable_init:
+*    - ZSTD_seekable_initBuff() : An in-memory API.  The data contained in
+*      `src` should be the entire seekable file, including the seek table.
+*      `src` should be kept alive and unmodified until the ZSTD_seekable object
+*      is freed or reset.
+*    - ZSTD_seekable_initFile() : A simplified file API using stdio.  fread and
+*      fseek will be used to access the required data for building the seek
+*      table and doing decompression operations.  `src` should not be closed
+*      or modified until the ZSTD_seekable object is freed or reset.
+*    - ZSTD_seekable_initAdvanced() : A general API allowing the client to
+*      provide its own read and seek callbacks.
+*        + ZSTD_seekable_read() : read exactly `n` bytes into `buffer`.
+*                                 Premature EOF should be treated as an error.
+*        + ZSTD_seekable_seek() : seek the read head to `offset` from `origin`,
+*                                 where origin is either SEEK_SET (beginning of
+*                                 file), or SEEK_END (end of file).
+*  Both functions should return a non-negative value in case of success, and a
+*  negative value in case of failure.  If implementing using this API and
+*  stdio, be careful with files larger than 4GB and fseek.  All of these
+*  functions return an error code checkable with ZSTD_isError().
 *
-*  Use ZSTD_seekable_loadSeekTable() to load the seek table from a file.
-*  `src` should point to a block of data read from the end of the file,
-*  i.e. `src + srcSize` should always be the end of the file.
-*  @return : 0 if the table was loaded successfully, or if `srcSize` was too
-*            small, a size hint for how much data to provide.
-*            An error code may also be returned, checkable with ZSTD_isError()
+*  Call ZSTD_seekable_decompress to decompress `dstSize` bytes at decompressed
+*  offset `offset`.  ZSTD_seekable_decompress may have to decompress the entire
+*  prefix of the frame before the desired data if it has not already processed
+*  this section. If ZSTD_seekable_decompress is called multiple times for a
+*  consecutive range of data, it will efficiently retain the decompressor object
+*  and avoid redecompressing frame prefixes.  The return value is the number of
+*  bytes decompressed, or an error code checkable with ZSTD_isError().
 *
-*  Use ZSTD_seekable_initDStream to prepare for a new decompression operation
-*  using the seektable loaded with ZSTD_seekable_loadSeekTable().
-*  Data in the range [rangeStart, rangeEnd) will be decompressed.
-*
-*  Call ZSTD_seekable_decompressStream() repetitively to consume input stream.
-*  @return : There are a number of possible return codes for this function
-*           - 0, the decompression operation has completed.
-*           - An error code checkable with ZSTD_isError
-*             + If this error code is ZSTD_error_needSeek, the user should seek
-*               to the file position provided by ZSTD_seekable_getSeekOffset()
-*               and indicate this to the stream with
-*               ZSTD_seekable_updateOffset(), before resuming decompression
-*             + Otherwise, this is a regular decompression error and the input
-*               file is likely corrupted or the API was incorrectly used.
-*           - A size hint, the preferred nb of bytes to provide as input to the
-*             next function call to improve latency.
-*
-*  ZSTD_seekable_getSeekOffset() and ZSTD_seekable_updateOffset() are helper
-*  functions to indicate where the user should seek their file stream to, when
-*  a different position is required to continue decompression.
-*  Note that ZSTD_seekable_updateOffset will error if given an offset other
-*  than the one requested from ZSTD_seekable_getSeekOffset().
+*  The seek table access functions can be used to obtain the data contained
+*  in the seek table.  If frameIndex is larger than the value returned by
+*  ZSTD_seekable_getNumFrames(), they will return error codes checkable with
+*  ZSTD_isError().  Note that since the offset access functions return
+*  unsigned long long instead of size_t, in this case they will instead return
+*  the value ZSTD_SEEKABLE_FRAMEINDEX_TOOLARGE.
 ******************************************************************************/
 
 /*===== Seekable decompressor management =====*/
-ZSTDLIB_API ZSTD_seekable_DStream* ZSTD_seekable_createDStream(void);
-ZSTDLIB_API size_t ZSTD_seekable_freeDStream(ZSTD_seekable_DStream* zds);
+ZSTDLIB_API ZSTD_seekable* ZSTD_seekable_create(void);
+ZSTDLIB_API size_t ZSTD_seekable_free(ZSTD_seekable* zs);
 
 /*===== Seekable decompression functions =====*/
-ZSTDLIB_API size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, size_t srcSize);
-ZSTDLIB_API size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, unsigned long long rangeStart, unsigned long long rangeEnd);
-ZSTDLIB_API size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input);
-ZSTDLIB_API unsigned long long ZSTD_seekable_getSeekOffset(ZSTD_seekable_DStream* zds);
-ZSTDLIB_API size_t ZSTD_seekable_updateOffset(ZSTD_seekable_DStream* zds, unsigned long long offset);
+ZSTDLIB_API size_t ZSTD_seekable_initBuff(ZSTD_seekable* zs, const void* src, size_t srcSize);
+ZSTDLIB_API size_t ZSTD_seekable_initFile(ZSTD_seekable* zs, FILE* src);
+ZSTDLIB_API size_t ZSTD_seekable_decompress(ZSTD_seekable* zs, void* dst, size_t dstSize, unsigned long long offset);
+ZSTDLIB_API size_t ZSTD_seekable_decompressFrame(ZSTD_seekable* zs, void* dst, size_t dstSize, unsigned frameIndex);
+
+/*===== Seek Table access functions =====*/
+ZSTDLIB_API unsigned ZSTD_seekable_getNumFrames(ZSTD_seekable* const zs);
+ZSTDLIB_API unsigned long long ZSTD_seekable_getFrameCompressedOffset(ZSTD_seekable* const zs, unsigned frameIndex);
+ZSTDLIB_API unsigned long long ZSTD_seekable_getFrameDecompressedOffset(ZSTD_seekable* const zs, unsigned frameIndex);
+ZSTDLIB_API size_t ZSTD_seekable_getFrameCompressedSize(ZSTD_seekable* const zs, unsigned frameIndex);
+ZSTDLIB_API size_t ZSTD_seekable_getFrameDecompressedSize(ZSTD_seekable* const zs, unsigned frameIndex);
+
+ZSTDLIB_API unsigned ZSTD_seekable_offsetToFrame(ZSTD_seekable* const zs, unsigned long long offset);
+
+/*===== Seekable advanced I/O API =====*/
+typedef int(ZSTD_seekable_read)(void* opaque, void* buffer, size_t n);
+typedef int(ZSTD_seekable_seek)(void* opaque, long long offset, int origin);
+typedef struct {
+    void* opaque;
+    ZSTD_seekable_read* read;
+    ZSTD_seekable_seek* seek;
+} ZSTD_seekable_customFile;
+
+ZSTDLIB_API size_t ZSTD_seekable_initAdvanced(ZSTD_seekable* zs, ZSTD_seekable_customFile src);
 
 #if defined (__cplusplus)
 }
diff --git a/contrib/seekable_format/zstdseek_decompress.c b/contrib/seekable_format/zstdseek_decompress.c
index 87a140c02..9bcfea91b 100644
--- a/contrib/seekable_format/zstdseek_decompress.c
+++ b/contrib/seekable_format/zstdseek_decompress.c
@@ -7,7 +7,54 @@
  * of patent rights can be found in the PATENTS file in the same directory.
  */
 
+/* *********************************************************
+*  Turn on Large Files support (>4GB) for 32-bit Linux/Unix
+***********************************************************/
+#if !defined(__64BIT__) || defined(__MINGW32__)       /* No point defining Large file for 64 bit but MinGW-w64 requires it */
+#  if !defined(_FILE_OFFSET_BITS)
+#    define _FILE_OFFSET_BITS 64                      /* turn off_t into a 64-bit type for ftello, fseeko */
+#  endif
+#  if !defined(_LARGEFILE_SOURCE)                     /* obsolete macro, replaced with _FILE_OFFSET_BITS */
+#    define _LARGEFILE_SOURCE 1                       /* Large File Support extension (LFS) - fseeko, ftello */
+#  endif
+#  if defined(_AIX) || defined(__hpux)
+#    define _LARGE_FILES                              /* Large file support on 32-bits AIX and HP-UX */
+#  endif
+#endif
+
+/* ************************************************************
+* Avoid fseek()'s 2GiB barrier with MSVC, MacOS, *BSD, MinGW
+***************************************************************/
+#if defined(_MSC_VER) && _MSC_VER >= 1400
+#   define LONG_SEEK _fseeki64
+#elif !defined(__64BIT__) && (PLATFORM_POSIX_VERSION >= 200112L) /* No point defining Large file for 64 bit */
+#  define LONG_SEEK fseeko
+#elif defined(__MINGW32__) && !defined(__STRICT_ANSI__) && !defined(__NO_MINGW_LFS) && defined(__MSVCRT__)
+#   define LONG_SEEK fseeko64
+#elif defined(_WIN32) && !defined(__DJGPP__)
+#   include 
+    static int LONG_SEEK(FILE* file, __int64 offset, int origin) {
+        LARGE_INTEGER off;
+        DWORD method;
+        off.QuadPart = offset;
+        if (origin == SEEK_END)
+            method = FILE_END;
+        else if (origin == SEEK_CUR)
+            method = FILE_CURRENT;
+        else
+            method = FILE_BEGIN;
+
+        if (SetFilePointerEx((HANDLE) _get_osfhandle(_fileno(file)), off, NULL, method))
+            return 0;
+        else
+            return -1;
+    }
+#else
+#   define LONG_SEEK fseek
+#endif
+
 #include  /* malloc, free */
+#include   /* FILE* */
 
 #define XXH_STATIC_LINKING_ONLY
 #define XXH_NAMESPACE ZSTD_
@@ -16,17 +63,74 @@
 #define ZSTD_STATIC_LINKING_ONLY
 #include "zstd.h"
 #include "zstd_errors.h"
-#include "mem.h" /* includes zstd.h */
+#include "mem.h"
 #include "zstd_seekable.h"
 
 #undef ERROR
 #define ERROR(name) ((size_t)-ZSTD_error_##name)
 
+#define CHECK_IO(f) { int const errcod = (f); if (errcod < 0) return ERROR(seekableIO); }
+
 #undef MIN
 #undef MAX
 #define MIN(a, b) ((a) < (b) ? (a) : (b))
 #define MAX(a, b) ((a) > (b) ? (a) : (b))
 
+/* Special-case callbacks for FILE* and in-memory modes, so that we can treat
+ * them the same way as the advanced API */
+static int ZSTD_seekable_read_FILE(void* opaque, void* buffer, size_t n)
+{
+    size_t const result = fread(buffer, 1, n, (FILE*)opaque);
+    if (result != n) {
+        return -1;
+    }
+    return 0;
+}
+
+static int ZSTD_seekable_seek_FILE(void* opaque, S64 offset, int origin)
+{
+    int const ret = LONG_SEEK((FILE*)opaque, offset, origin);
+    if (ret) return ret;
+    return fflush((FILE*)opaque);
+}
+
+typedef struct {
+    const void *ptr;
+    size_t size;
+    size_t pos;
+} buffWrapper_t;
+
+static int ZSTD_seekable_read_buff(void* opaque, void* buffer, size_t n)
+{
+    buffWrapper_t* buff = (buffWrapper_t*) opaque;
+    if (buff->size + n > buff->pos) return -1;
+    memcpy(buffer, (const BYTE*)buff->ptr + buff->pos, n);
+    buff->pos += n;
+    return 0;
+}
+
+static int ZSTD_seekable_seek_buff(void* opaque, S64 offset, int origin)
+{
+    buffWrapper_t* buff = (buffWrapper_t*) opaque;
+    unsigned long long newOffset;
+    switch (origin) {
+    case SEEK_SET:
+        newOffset = offset;
+        break;
+    case SEEK_CUR:
+        newOffset = (unsigned long long)buff->pos + offset;
+        break;
+    case SEEK_END:
+        newOffset = (unsigned long long)buff->size - offset;
+        break;
+    }
+    if (newOffset < 0 || newOffset > buff->size) {
+        return -1;
+    }
+    buff->pos = newOffset;
+    return 0;
+}
+
 typedef struct {
     U64 cOffset;
     U64 dOffset;
@@ -40,18 +144,70 @@ typedef struct {
     int checksumFlag;
 } seekTable_t;
 
+#define SEEKABLE_BUFF_SIZE ZSTD_BLOCKSIZE_ABSOLUTEMAX
+
+struct ZSTD_seekable_s {
+    ZSTD_DStream* dstream;
+    seekTable_t seekTable;
+    ZSTD_seekable_customFile src;
+
+    U64 decompressedOffset;
+    U32 curFrame;
+
+    BYTE inBuff[SEEKABLE_BUFF_SIZE]; /* need to do our own input buffering */
+    BYTE outBuff[SEEKABLE_BUFF_SIZE]; /* so we can efficiently decompress the
+                                         starts of chunks before we get to the
+                                         desired section */
+    ZSTD_inBuffer in; /* maintain continuity across ZSTD_seekable_decompress operations */
+    buffWrapper_t buffWrapper; /* for `src.opaque` in in-memory mode */
+
+    XXH64_state_t xxhState;
+};
+
+ZSTD_seekable* ZSTD_seekable_create(void)
+{
+    ZSTD_seekable* zs = malloc(sizeof(ZSTD_seekable));
+
+    if (zs == NULL) return NULL;
+
+    /* also initializes stage to zsds_init */
+    memset(zs, 0, sizeof(*zs));
+
+    zs->dstream = ZSTD_createDStream();
+    if (zs->dstream == NULL) {
+        free(zs);
+        return NULL;
+    }
+
+    return zs;
+}
+
+size_t ZSTD_seekable_free(ZSTD_seekable* zs)
+{
+    if (zs == NULL) return 0; /* support free on null */
+    ZSTD_freeDStream(zs->dstream);
+    free(zs->seekTable.entries);
+    free(zs);
+
+    return 0;
+}
+
 /** ZSTD_seekable_offsetToFrame() :
  *  Performs a binary search to find the last frame with a decompressed offset
  *  <= pos
  *  @return : the frame's index */
-static U32 ZSTD_seekable_offsetToFrame(const seekTable_t* table, U64 pos)
+U32 ZSTD_seekable_offsetToFrame(ZSTD_seekable* const zs, U64 pos)
 {
     U32 lo = 0;
-    U32 hi = table->tableLen;
+    U32 hi = zs->seekTable.tableLen;
+
+    if (pos >= zs->seekTable.entries[zs->seekTable.tableLen].dOffset) {
+        return zs->seekTable.tableLen;
+    }
 
     while (lo + 1 < hi) {
         U32 const mid = lo + ((hi - lo) >> 1);
-        if (table->entries[mid].dOffset <= pos) {
+        if (zs->seekTable.entries[mid].dOffset <= pos) {
             lo = mid;
         } else {
             hi = mid;
@@ -60,75 +216,50 @@ static U32 ZSTD_seekable_offsetToFrame(const seekTable_t* table, U64 pos)
     return lo;
 }
 
-/* Stream decompressor state machine stages */
-enum ZSTD_seekable_DStream_stage {
-    zsds_init = 0,
-    zsds_seek,
-    zsds_decompress,
-    zsds_done,
-};
-
-struct ZSTD_seekable_DStream_s {
-    ZSTD_DStream* dstream;
-    seekTable_t seekTable;
-
-    U32 curFrame;
-    U64 compressedOffset;
-    U64 decompressedOffset;
-
-    U64 targetStart;
-    U64 targetEnd;
-
-    U64 nextSeek;
-
-    enum ZSTD_seekable_DStream_stage stage;
-
-    XXH64_state_t xxhState;
-};
-
-ZSTD_seekable_DStream* ZSTD_seekable_createDStream(void)
+U32 ZSTD_seekable_getNumFrames(ZSTD_seekable* const zs)
 {
-    ZSTD_seekable_DStream* zds = malloc(sizeof(ZSTD_seekable_DStream));
-
-    if (zds == NULL) return NULL;
-
-    /* also initializes stage to zsds_init */
-    memset(zds, 0, sizeof(*zds));
-
-    zds->dstream = ZSTD_createDStream();
-    if (zds->dstream == NULL) {
-        free(zds);
-        return NULL;
-    }
-
-    return zds;
+    return zs->seekTable.tableLen;
 }
 
-size_t ZSTD_seekable_freeDStream(ZSTD_seekable_DStream* zds)
+U64 ZSTD_seekable_getFrameCompressedOffset(ZSTD_seekable* const zs, U32 frameIndex)
 {
-    if (zds == NULL) return 0; /* support free on null */
-    ZSTD_freeDStream(zds->dstream);
-    free(zds->seekTable.entries);
-    free(zds);
-
-    return 0;
+    if (frameIndex >= zs->seekTable.tableLen) return ZSTD_SEEKABLE_FRAMEINDEX_TOOLARGE;
+    return zs->seekTable.entries[frameIndex].cOffset;
 }
 
-size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src, size_t srcSize)
+U64 ZSTD_seekable_getFrameDecompressedOffset(ZSTD_seekable* const zs, U32 frameIndex)
 {
-    const BYTE* ip = (const BYTE*)src + srcSize;
+    if (frameIndex >= zs->seekTable.tableLen) return ZSTD_SEEKABLE_FRAMEINDEX_TOOLARGE;
+    return zs->seekTable.entries[frameIndex].dOffset;
+}
 
+size_t ZSTD_seekable_getFrameCompressedSize(ZSTD_seekable* const zs, U32 frameIndex)
+{
+    if (frameIndex >= zs->seekTable.tableLen) return ERROR(frameIndex_tooLarge);
+    return zs->seekTable.entries[frameIndex + 1].cOffset -
+           zs->seekTable.entries[frameIndex].cOffset;
+}
+
+size_t ZSTD_seekable_getFrameDecompressedSize(ZSTD_seekable* const zs, U32 frameIndex)
+{
+    if (frameIndex > zs->seekTable.tableLen) return ERROR(frameIndex_tooLarge);
+    return zs->seekTable.entries[frameIndex + 1].dOffset -
+           zs->seekTable.entries[frameIndex].dOffset;
+}
+
+static size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable* zs)
+{
     int checksumFlag;
+    ZSTD_seekable_customFile src = zs->src;
+    /* read the footer, fixed size */
+    CHECK_IO(src.seek(src.opaque, -(int)ZSTD_seekTableFooterSize, SEEK_END));
+    CHECK_IO(src.read(src.opaque, zs->inBuff, ZSTD_seekTableFooterSize));
 
-    /* footer is fixed size */
-    if (srcSize < ZSTD_seekTableFooterSize)
-        return ZSTD_seekTableFooterSize;
-
-    if (MEM_readLE32(ip - 4) != ZSTD_SEEKABLE_MAGICNUMBER) {
+    if (MEM_readLE32(zs->inBuff + 5) != ZSTD_SEEKABLE_MAGICNUMBER) {
         return ERROR(prefix_unknown);
     }
 
-    {   BYTE const sfd = ip[-5];
+    {   BYTE const sfd = zs->inBuff[4];
         checksumFlag = sfd >> 7;
 
         /* check reserved bits */
@@ -137,30 +268,36 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src,
         }
     }
 
-    {   U32 const numFrames = MEM_readLE32(ip-9);
+    {   U32 const numFrames = MEM_readLE32(zs->inBuff);
         U32 const sizePerEntry = 8 + (checksumFlag?4:0);
         U32 const tableSize = sizePerEntry * numFrames;
         U32 const frameSize = tableSize + ZSTD_seekTableFooterSize + ZSTD_skippableHeaderSize;
 
-        const BYTE* base = ip - frameSize;
+        U32 remaining = frameSize - ZSTD_seekTableFooterSize; /* don't need to re-read footer */
+        {
+            U32 const toRead = MIN(remaining, SEEKABLE_BUFF_SIZE);
 
-        if (srcSize < frameSize) return frameSize;
+            CHECK_IO(src.seek(src.opaque, -(S64)frameSize, SEEK_END));
+            CHECK_IO(src.read(src.opaque, zs->inBuff, toRead));
 
-        if (MEM_readLE32(base) != (ZSTD_MAGIC_SKIPPABLE_START | 0xE)) {
+            remaining -= toRead;
+        }
+
+        if (MEM_readLE32(zs->inBuff) != (ZSTD_MAGIC_SKIPPABLE_START | 0xE)) {
             return ERROR(prefix_unknown);
         }
-        if (MEM_readLE32(base+4) + ZSTD_skippableHeaderSize != frameSize) {
+        if (MEM_readLE32(zs->inBuff+4) + ZSTD_skippableHeaderSize != frameSize) {
             return ERROR(prefix_unknown);
         }
 
         {   /* Allocate an extra entry at the end so that we can do size
              * computations on the last element without special case */
-            seekEntry_t* entries =
-                    (seekEntry_t*)malloc(sizeof(seekEntry_t) * (numFrames + 1));
-            const BYTE* tableBase = base + ZSTD_skippableHeaderSize;
+            seekEntry_t* entries = (seekEntry_t*)malloc(sizeof(seekEntry_t) * (numFrames + 1));
+            const BYTE* tableBase = zs->inBuff + ZSTD_skippableHeaderSize;
+
+            U32 idx = 0;
+            U32 pos = 8;
 
-            U32 idx;
-            size_t pos;
 
             U64 cOffset = 0;
             U64 dOffset = 0;
@@ -171,202 +308,153 @@ size_t ZSTD_seekable_loadSeekTable(ZSTD_seekable_DStream* zds, const void* src,
             }
 
             /* compute cumulative positions */
-            for (idx = 0, pos = 0; idx < numFrames; idx++) {
+            for (; idx < numFrames; idx++) {
+                if (pos + sizePerEntry > SEEKABLE_BUFF_SIZE) {
+                    U32 const toRead = MIN(remaining, SEEKABLE_BUFF_SIZE);
+                    U32 const offset = SEEKABLE_BUFF_SIZE - pos;
+                    memmove(zs->inBuff, zs->inBuff + pos, offset); /* move any data we haven't read yet */
+                    CHECK_IO(src.read(src.opaque, zs->inBuff+offset, toRead));
+                    remaining -= toRead;
+                    pos = 0;
+                }
                 entries[idx].cOffset = cOffset;
                 entries[idx].dOffset = dOffset;
 
-                cOffset += MEM_readLE32(tableBase + pos);
+                cOffset += MEM_readLE32(zs->inBuff + pos);
                 pos += 4;
-                dOffset += MEM_readLE32(tableBase + pos);
+                dOffset += MEM_readLE32(zs->inBuff + pos);
                 pos += 4;
                 if (checksumFlag) {
-                    entries[idx].checksum = MEM_readLE32(tableBase + pos);
+                    entries[idx].checksum = MEM_readLE32(zs->inBuff + pos);
                     pos += 4;
                 }
             }
             entries[numFrames].cOffset = cOffset;
             entries[numFrames].dOffset = dOffset;
 
-            zds->seekTable.entries = entries;
-            zds->seekTable.tableLen = numFrames;
-            zds->seekTable.checksumFlag = checksumFlag;
+            zs->seekTable.entries = entries;
+            zs->seekTable.tableLen = numFrames;
+            zs->seekTable.checksumFlag = checksumFlag;
             return 0;
         }
     }
 }
 
-size_t ZSTD_seekable_initDStream(ZSTD_seekable_DStream* zds, U64 rangeStart, U64 rangeEnd)
+size_t ZSTD_seekable_initBuff(ZSTD_seekable* zs, const void* src, size_t srcSize)
 {
-    /* restrict range to the end of the file, of non-negative size */
-    rangeEnd = MIN(rangeEnd, zds->seekTable.entries[zds->seekTable.tableLen].dOffset);
-    rangeStart = MIN(rangeStart, rangeEnd);
+    zs->buffWrapper = (buffWrapper_t){src, srcSize, 0};
+    {   ZSTD_seekable_customFile srcFile = {&zs->buffWrapper,
+                                            &ZSTD_seekable_read_buff,
+                                            &ZSTD_seekable_seek_buff};
+        return ZSTD_seekable_initAdvanced(zs, srcFile); }
+}
 
-    zds->targetStart = rangeStart;
-    zds->targetEnd = rangeEnd;
-    zds->stage = zsds_seek;
+size_t ZSTD_seekable_initFile(ZSTD_seekable* zs, FILE* src)
+{
+    ZSTD_seekable_customFile srcFile = {src, &ZSTD_seekable_read_FILE,
+                                        &ZSTD_seekable_seek_FILE};
+    return ZSTD_seekable_initAdvanced(zs, srcFile);
+}
 
-    /* force a seek first */
-    zds->curFrame = (U32)-1;
-    zds->compressedOffset = (U64)-1;
-    zds->decompressedOffset = (U64)-1;
+size_t ZSTD_seekable_initAdvanced(ZSTD_seekable* zs, ZSTD_seekable_customFile src)
+{
+    zs->src = src;
 
-    if (zds->seekTable.checksumFlag) {
-        XXH64_reset(&zds->xxhState, 0);
-    }
+    {   const size_t seekTableInit = ZSTD_seekable_loadSeekTable(zs);
+        if (ZSTD_isError(seekTableInit)) return seekTableInit; }
 
-    if (rangeStart == rangeEnd) zds->stage = zsds_done;
+    zs->decompressedOffset = (U64)-1;
+    zs->curFrame = (U32)-1;
 
-    {   const size_t ret = ZSTD_initDStream(zds->dstream);
-        if (ZSTD_isError(ret)) return ret; }
+    {   const size_t dstreamInit = ZSTD_initDStream(zs->dstream);
+        if (ZSTD_isError(dstreamInit)) return dstreamInit; }
     return 0;
 }
 
-U64 ZSTD_seekable_getSeekOffset(ZSTD_seekable_DStream* zds)
+size_t ZSTD_seekable_decompress(ZSTD_seekable* zs, void* dst, size_t len, U64 offset)
 {
-    return zds->nextSeek;
-}
+    U32 targetFrame = ZSTD_seekable_offsetToFrame(zs, offset);
+    do {
+        /* check if we can continue from a previous decompress job */
+        if (targetFrame != zs->curFrame || offset != zs->decompressedOffset) {
+            zs->decompressedOffset = zs->seekTable.entries[targetFrame].dOffset;
+            zs->curFrame = targetFrame;
 
-size_t ZSTD_seekable_updateOffset(ZSTD_seekable_DStream* zds, U64 offset)
-{
-    if (zds->stage != zsds_seek) {
-        return ERROR(stage_wrong);
-    }
-    if (offset != zds->nextSeek) {
-        return ERROR(needSeek);
-    }
+            CHECK_IO(zs->src.seek(zs->src.opaque,
+                                  zs->seekTable.entries[targetFrame].cOffset,
+                                  SEEK_SET));
+            zs->in = (ZSTD_inBuffer){zs->inBuff, 0, 0};
+            XXH64_reset(&zs->xxhState, 0);
+            ZSTD_resetDStream(zs->dstream);
+        }
 
-    zds->stage = zsds_decompress;
-    zds->compressedOffset = offset;
-    return 0;
-}
+        while (zs->decompressedOffset < offset + len) {
+            size_t toRead;
+            ZSTD_outBuffer outTmp;
+            size_t prevOutPos;
+            if (zs->decompressedOffset < offset) {
+                /* dummy decompressions until we get to the target offset */
+                outTmp = (ZSTD_outBuffer){zs->outBuff, MIN(SEEKABLE_BUFF_SIZE, offset - zs->decompressedOffset), 0};
+            } else {
+                outTmp = (ZSTD_outBuffer){dst, len, zs->decompressedOffset - offset};
+            }
 
-size_t ZSTD_seekable_decompressStream(ZSTD_seekable_DStream* zds, ZSTD_outBuffer* output, ZSTD_inBuffer* input)
-{
-    const seekTable_t* const jt = &zds->seekTable;
-    while (1) {
-        switch (zds->stage) {
-        case zsds_init:
-            return ERROR(init_missing); /* ZSTD_seekable_initDStream should be called first */
-        case zsds_decompress: {
-            BYTE* const outBase = (BYTE*)output->dst + output->pos;
-            size_t const outLen = output->size - output->pos;
-            while (zds->decompressedOffset < zds->targetStart) {
-                U64 const toDecompress =
-                        zds->targetStart - zds->decompressedOffset;
-                size_t const prevInputPos = input->pos;
+            prevOutPos = outTmp.pos;
+            toRead = ZSTD_decompressStream(zs->dstream, &outTmp, &zs->in);
+            if (ZSTD_isError(toRead)) {
+                return toRead;
+            }
 
-                ZSTD_outBuffer outTmp = {
-                        outBase, (size_t)MIN((U64)outLen, toDecompress), 0};
+            if (zs->seekTable.checksumFlag) {
+                XXH64_update(&zs->xxhState, outTmp.dst, outTmp.pos);
+            }
+            zs->decompressedOffset += outTmp.pos - prevOutPos;
 
-                size_t const ret =
-                        ZSTD_decompressStream(zds->dstream, &outTmp, input);
+            if (toRead == 0) {
+                /* frame complete */
 
-                if (ZSTD_isError(ret)) return ret;
-                if (ret == 0) {
-                    /* should not happen at this stage */
+                /* verify checksum */
+                if (zs->seekTable.checksumFlag &&
+                    (XXH64_digest(&zs->xxhState) & 0xFFFFFFFFU) !=
+                            zs->seekTable.entries[targetFrame].checksum) {
                     return ERROR(corruption_detected);
                 }
 
-                zds->compressedOffset += input->pos - prevInputPos;
-                zds->decompressedOffset += outTmp.pos;
-
-                if (jt->checksumFlag) {
-                    XXH64_update(&zds->xxhState, outTmp.dst, outTmp.pos);
+                if (zs->decompressedOffset < offset + len) {
+                    /* go back to the start and force a reset of the stream */
+                    targetFrame = ZSTD_seekable_offsetToFrame(zs, zs->decompressedOffset);
                 }
-
-                if (input->pos == input->size) {
-                    /* need more input */
-                    return MIN(
-                            ZSTD_DStreamInSize(),
-                            (size_t)(jt->entries[zds->curFrame + 1]
-                                             .cOffset -
-                                     zds->compressedOffset));
-                }
-            }
-
-            /* do actual decompression */
-            {
-                U64 const toDecompress =
-                        MIN(zds->targetEnd,
-                            jt->entries[zds->curFrame + 1].dOffset) -
-                        zds->decompressedOffset;
-                size_t const prevInputPos = input->pos;
-
-                ZSTD_outBuffer outTmp = {
-                        outBase, (size_t)MIN((U64)outLen, toDecompress), 0};
-
-                size_t const ret =
-                        ZSTD_decompressStream(zds->dstream, &outTmp, input);
-
-                if (ZSTD_isError(ret)) return ret;
-
-                zds->compressedOffset += input->pos - prevInputPos;
-                zds->decompressedOffset += outTmp.pos;
-
-                output->pos += outTmp.pos;
-
-                if (jt->checksumFlag) {
-                    XXH64_update(&zds->xxhState, outTmp.dst, outTmp.pos);
-                    if (ret == 0) {
-                        /* verify the checksum */
-                        U32 const digest = XXH64_digest(&zds->xxhState) & 0xFFFFFFFFU;
-                        if (digest != jt->entries[zds->curFrame].checksum) {
-                            return ERROR(checksum_wrong);
-                        }
-
-                        XXH64_reset(&zds->xxhState, 0);
-                    }
-                }
-
-                if (zds->decompressedOffset == zds->targetEnd) {
-                    /* done */
-                    zds->stage = zsds_done;
-                    return 0;
-                }
-
-                if (ret == 0) {
-                    /* frame is done */
-                    /* make sure this lines up with the expected frame border */
-                    if (zds->decompressedOffset !=
-                                jt->entries[zds->curFrame + 1].dOffset ||
-                        zds->compressedOffset !=
-                                jt->entries[zds->curFrame + 1].cOffset)
-                        return ERROR(corruption_detected);
-                    ZSTD_resetDStream(zds->dstream);
-                    zds->stage = zsds_seek;
-                    break;
-                }
-
-                /* need more input */
-                return MIN(ZSTD_DStreamInSize(), (size_t)(
-                        jt->entries[zds->curFrame + 1].cOffset -
-                        zds->compressedOffset));
-            }
-        }
-        case zsds_seek: {
-            U32 targetFrame;
-            if (zds->decompressedOffset < zds->targetStart ||
-                    zds->decompressedOffset >= zds->targetEnd) {
-                /* haven't started yet */
-                targetFrame = ZSTD_seekable_offsetToFrame(jt, zds->targetStart);
-            } else {
-                targetFrame = ZSTD_seekable_offsetToFrame(jt, zds->decompressedOffset);
-            }
-
-            zds->curFrame = targetFrame;
-
-            if (zds->compressedOffset == jt->entries[targetFrame].cOffset) {
-                zds->stage = zsds_decompress;
                 break;
             }
 
-            zds->nextSeek = jt->entries[targetFrame].cOffset;
-            zds->decompressedOffset = jt->entries[targetFrame].dOffset;
-            /* signal to user that a seek is required */
-            return ERROR(needSeek);
+            /* read in more data if we're done with this buffer */
+            if (zs->in.pos == zs->in.size) {
+                toRead = MIN(toRead, SEEKABLE_BUFF_SIZE);
+                CHECK_IO(zs->src.read(zs->src.opaque, zs->inBuff, toRead));
+                zs->in.size = toRead;
+                zs->in.pos = 0;
+            }
         }
-        case zsds_done:
-            return 0;
+    } while (zs->decompressedOffset != offset + len);
+
+    return len;
+}
+
+size_t ZSTD_seekable_decompressFrame(ZSTD_seekable* zs, void* dst, size_t dstSize, U32 frameIndex)
+{
+    if (frameIndex >= zs->seekTable.tableLen) {
+        return ERROR(frameIndex_tooLarge);
+    }
+
+    {
+        size_t const decompressedSize =
+                zs->seekTable.entries[frameIndex + 1].dOffset -
+                zs->seekTable.entries[frameIndex].dOffset;
+        if (dstSize < decompressedSize) {
+            return ERROR(dstSize_tooSmall);
         }
+        return ZSTD_seekable_decompress(
+                zs, dst, zs->seekTable.entries[frameIndex].dOffset,
+                decompressedSize);
     }
 }
diff --git a/lib/common/error_private.c b/lib/common/error_private.c
index f32c6abda..c94ea181c 100644
--- a/lib/common/error_private.c
+++ b/lib/common/error_private.c
@@ -39,7 +39,7 @@ const char* ERR_getErrorString(ERR_enum code)
     case PREFIX(dictionary_wrong): return "Dictionary mismatch";
     case PREFIX(dictionaryCreation_failed): return "Cannot create Dictionary from provided samples";
     case PREFIX(frameIndex_tooLarge): return "Frame index is too large";
-    case PREFIX(needSeek): return "Wrong file position, a seek is required to continue";
+    case PREFIX(seekableIO): return "An I/O error occurred when reading/seeking";
     case PREFIX(maxCode):
     default: return notErrorCode;
     }
diff --git a/lib/common/zstd_errors.h b/lib/common/zstd_errors.h
index d11c1ba21..de0fc8984 100644
--- a/lib/common/zstd_errors.h
+++ b/lib/common/zstd_errors.h
@@ -59,7 +59,7 @@ typedef enum {
   ZSTD_error_dictionary_wrong,
   ZSTD_error_dictionaryCreation_failed,
   ZSTD_error_frameIndex_tooLarge,
-  ZSTD_error_needSeek,
+  ZSTD_error_seekableIO,
   ZSTD_error_maxCode
 } ZSTD_ErrorCode;
 

From 98cf7fcb2ab311dc40a616cd15774813e1e8517d Mon Sep 17 00:00:00 2001
From: Sean Purcell 
Date: Tue, 18 Apr 2017 17:03:37 -0700
Subject: [PATCH 208/305] Update README

---
 lib/README.md | 8 ++++++++
 1 file changed, 8 insertions(+)

diff --git a/lib/README.md b/lib/README.md
index 3357e3d87..8fa60a20f 100644
--- a/lib/README.md
+++ b/lib/README.md
@@ -22,6 +22,14 @@ Some additional API may be useful if you're looking into advanced features :
                           They are not "stable", their definition may change in the future.
                           Only static linking is allowed.
 
+#### ZSTDMT API
+
+To enable building the multithreaded compression API, use the `make lib-mt` target.
+The header file zstdmt_compress.h (in compress/) provides the prototypes for the API.
+If linking a program that uses the ZSTDMT API against libzstd.a on a POSIX system,
+the -pthread flag must be provided to the compiler at link time.
+Note that these prototypes may still be used in a version built without multithread support,
+but they will be single threaded (i.e. no benefits will be given over the standard ZSTD API).
 
 #### Modular build
 

From 2c5514c759a747ea902a346a9a9911ce726272e7 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Tue, 18 Apr 2017 22:52:41 -0700
Subject: [PATCH 209/305] fixed ZSTDMT_initCStream_advanced()

Must use the new ZSTD_compressBegin_usingCDict_advanced()
to enforce correct frame parameters
---
 lib/compress/zstd_compress.c   | 2 +-
 lib/compress/zstdmt_compress.c | 2 +-
 tests/zstreamtest.c            | 6 ++++--
 3 files changed, 6 insertions(+), 4 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 9d9cb394f..29dc0b773 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -392,12 +392,12 @@ size_t ZSTD_copyCCtx_internal(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx,
 
     /* copy entropy tables */
     dstCCtx->flagStaticTables = srcCCtx->flagStaticTables;
-    dstCCtx->flagStaticHufTable = srcCCtx->flagStaticHufTable;
     if (srcCCtx->flagStaticTables) {
         memcpy(dstCCtx->litlengthCTable, srcCCtx->litlengthCTable, sizeof(dstCCtx->litlengthCTable));
         memcpy(dstCCtx->matchlengthCTable, srcCCtx->matchlengthCTable, sizeof(dstCCtx->matchlengthCTable));
         memcpy(dstCCtx->offcodeCTable, srcCCtx->offcodeCTable, sizeof(dstCCtx->offcodeCTable));
     }
+    dstCCtx->flagStaticHufTable = srcCCtx->flagStaticHufTable;
     if (srcCCtx->flagStaticHufTable) {
         memcpy(dstCCtx->hufTable, srcCCtx->hufTable, HUF_CTABLE_SIZE(255));
     }
diff --git a/lib/compress/zstdmt_compress.c b/lib/compress/zstdmt_compress.c
index 50c98ae93..04a448981 100644
--- a/lib/compress/zstdmt_compress.c
+++ b/lib/compress/zstdmt_compress.c
@@ -231,7 +231,7 @@ void ZSTDMT_compressChunk(void* jobDescription)
     DEBUGLOG(3, "job (first:%u) (last:%u) : dictSize %u, srcSize %u",
                  job->firstChunk, job->lastChunk, (U32)job->dictSize, (U32)job->srcSize);
     if (job->cdict) {  /* should only happen for first segment */
-        size_t const initError = ZSTD_compressBegin_usingCDict(job->cctx, job->cdict, job->fullFrameSize);
+        size_t const initError = ZSTD_compressBegin_usingCDict_advanced(job->cctx, job->cdict, job->params.fParams, job->fullFrameSize);
         if (job->cdict) DEBUGLOG(3, "using CDict ");
         if (ZSTD_isError(initError)) { job->cSize = initError; goto _endJob; }
     } else {  /* srcStart points at reloaded section */
diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c
index 6673208b0..4048a7df0 100644
--- a/tests/zstreamtest.c
+++ b/tests/zstreamtest.c
@@ -131,7 +131,7 @@ static buffer_t FUZ_createDictionary(const void* src, size_t srcSize, size_t blo
     }
     {   size_t const dictSize = ZDICT_trainFromBuffer(dict.start, requestedDictSize, src, blockSizes, (unsigned)nbBlocks);
         free(blockSizes);
-        if (ZDICT_isError(dictSize)) { free(dict.start); return (buffer_t){ NULL, 0, 0 }; }
+        if (ZDICT_isError(dictSize)) { free(dict.start); return g_nullBuffer; }
         dict.size = requestedDictSize;
         dict.filled = dictSize;
         return dict;   /* how to return dictSize ? */
@@ -972,6 +972,7 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp
                 params.fParams.checksumFlag = FUZ_rand(&lseed) & 1;
                 params.fParams.noDictIDFlag = FUZ_rand(&lseed) & 1;
                 params.fParams.contentSizeFlag = pledgedSrcSize>0;
+                DISPLAYLEVEL(5, "checksumFlag : %u \n", params.fParams.checksumFlag);
                 { size_t const initError = ZSTDMT_initCStream_advanced(zc, dict, dictSize, params, pledgedSrcSize);
                   CHECK (ZSTD_isError(initError),"ZSTDMT_initCStream_advanced error : %s", ZSTD_getErrorName(initError)); }
                 ZSTDMT_setMTCtxParameter(zc, ZSTDMT_p_overlapSectionLog, FUZ_rand(&lseed) % 12);
@@ -1023,9 +1024,9 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp
                     CHECK (ZSTD_isError(remainingToFlush), "ZSTDMT_endStream error : %s", ZSTD_getErrorName(remainingToFlush));
                     DISPLAYLEVEL(5, "endStream : remainingToFlush : %u \n", (U32)remainingToFlush);
             }   }
-            DISPLAYLEVEL(5, "Frame completed \n");
             crcOrig = XXH64_digest(&xxhState);
             cSize = outBuff.pos;
+            DISPLAYLEVEL(5, "Frame completed : %u bytes \n", (U32)cSize);
         }
 
         /* multi - fragments decompression test */
@@ -1046,6 +1047,7 @@ static int fuzzerTests_MT(U32 seed, U32 nbTests, unsigned startTest, double comp
                 DISPLAYLEVEL(5, "ZSTD_decompressStream input %u bytes \n", (U32)readCSrcSize);
                 decompressionResult = ZSTD_decompressStream(zd, &outBuff, &inBuff);
                 CHECK (ZSTD_isError(decompressionResult), "decompression error : %s", ZSTD_getErrorName(decompressionResult));
+                DISPLAYLEVEL(5, "inBuff.pos = %u \n", (U32)readCSrcSize);
             }
             CHECK (outBuff.pos != totalTestSize, "decompressed data : wrong size (%u != %u)", (U32)outBuff.pos, (U32)totalTestSize);
             CHECK (inBuff.pos != cSize, "compressed data should be fully read (%u != %u)", (U32)inBuff.pos, (U32)cSize);

From e847730452ba155888c193acf63f1038af0bfa81 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Tue, 18 Apr 2017 23:15:28 -0700
Subject: [PATCH 210/305] slightly refined README comments on lib-mt

---
 lib/README.md | 12 ++++++------
 1 file changed, 6 insertions(+), 6 deletions(-)

diff --git a/lib/README.md b/lib/README.md
index 8fa60a20f..79b6fd500 100644
--- a/lib/README.md
+++ b/lib/README.md
@@ -24,12 +24,12 @@ Some additional API may be useful if you're looking into advanced features :
 
 #### ZSTDMT API
 
-To enable building the multithreaded compression API, use the `make lib-mt` target.
-The header file zstdmt_compress.h (in compress/) provides the prototypes for the API.
-If linking a program that uses the ZSTDMT API against libzstd.a on a POSIX system,
-the -pthread flag must be provided to the compiler at link time.
-Note that these prototypes may still be used in a version built without multithread support,
-but they will be single threaded (i.e. no benefits will be given over the standard ZSTD API).
+To enable multithreaded compression within the library, invoke `make lib-mt` target.
+Prototypes are defined in header file `compress/zstdmt_compress.h`.
+When linking a program that uses ZSTDMT API against libzstd.a on a POSIX system,
+`-pthread` flag must be provided to the compiler and linker.
+Note : ZSTDMT prototypes can still be used with a library built without multithread support,
+but in this case, they will be single threaded only.
 
 #### Modular build
 

From 522df42e10710a543414f84d620a8b858645d0e2 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= 
Date: Wed, 19 Apr 2017 19:20:27 +0200
Subject: [PATCH 211/305] Add lzma and zlib support to cmake build system

cmake 2.8.9 needed for FindLibLZMA
---
 build/cmake/CMakeLists.txt          |  2 +-
 build/cmake/programs/CMakeLists.txt | 27 +++++++++++++++++++++++++++
 2 files changed, 28 insertions(+), 1 deletion(-)

diff --git a/build/cmake/CMakeLists.txt b/build/cmake/CMakeLists.txt
index 9ce52ed21..bf6890818 100644
--- a/build/cmake/CMakeLists.txt
+++ b/build/cmake/CMakeLists.txt
@@ -8,7 +8,7 @@
 # ################################################################
 
 PROJECT(zstd)
-CMAKE_MINIMUM_REQUIRED(VERSION 2.8.7)
+CMAKE_MINIMUM_REQUIRED(VERSION 2.8.9)
 SET(ZSTD_SOURCE_DIR "${CMAKE_SOURCE_DIR}/../..")
 
 #-----------------------------------------------------------------------------
diff --git a/build/cmake/programs/CMakeLists.txt b/build/cmake/programs/CMakeLists.txt
index 588fea41e..b6f79aba3 100644
--- a/build/cmake/programs/CMakeLists.txt
+++ b/build/cmake/programs/CMakeLists.txt
@@ -64,3 +64,30 @@ IF (ZSTD_MULTITHREAD_SUPPORT)
     ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd zstdmt COMMENT "Creating zstdmt symlink")
     INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/zstdmt DESTINATION "bin")
 ENDIF (ZSTD_MULTITHREAD_SUPPORT)
+
+OPTION(ZSTD_ZLIB_SUPPORT "ZLIB SUPPORT" OFF)
+OPTION(ZSTD_LZMA_SUPPORT "LZMA SUPPORT" OFF)
+
+IF (ZSTD_ZLIB_SUPPORT)
+    FIND_PACKAGE(ZLIB REQUIRED)
+
+    IF (ZLIB_FOUND)
+        INCLUDE_DIRECTORIES(${ZLIB_INCLUDE_DIRS})
+	TARGET_LINK_LIBRARIES(zstd ${ZLIB_LIBRARIES})
+        SET_TARGET_PROPERTIES(zstd PROPERTIES COMPILE_DEFINITIONS "ZSTD_GZCOMPRESS;ZSTD_GZDECOMPRESS")
+    ELSE ()
+        MESSAGE(SEND_ERROR "zlib library is missing")
+    ENDIF ()
+ENDIF ()
+
+IF (ZSTD_LZMA_SUPPORT)
+    FIND_PACKAGE(LibLZMA REQUIRED)
+
+    IF (LIBLZMA_FOUND)
+        INCLUDE_DIRECTORIES(${LIBLZMA_INCLUDE_DIRS})
+	TARGET_LINK_LIBRARIES(zstd ${LIBLZMA_LIBRARIES})
+	SET_TARGET_PROPERTIES(zstd PROPERTIES COMPILE_DEFINITIONS "ZSTD_LZMACOMPRESS;ZSTD_LZMADECOMPRESS")
+    ELSE ()
+        MESSAGE(SEND_ERROR "lzma library is missing")
+    ENDIF ()
+ENDIF ()

From fce21777bdce8695ab75f0ae781bfbca39b22d53 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= 
Date: Wed, 19 Apr 2017 19:22:17 +0200
Subject: [PATCH 212/305] Copy files during build phase, custom targets instead
 of commands

Previously some files were copied only during configure phase.
Custom targets seem nicer.
---
 build/cmake/programs/CMakeLists.txt | 14 ++++++++------
 1 file changed, 8 insertions(+), 6 deletions(-)

diff --git a/build/cmake/programs/CMakeLists.txt b/build/cmake/programs/CMakeLists.txt
index b6f79aba3..c3e68e01c 100644
--- a/build/cmake/programs/CMakeLists.txt
+++ b/build/cmake/programs/CMakeLists.txt
@@ -31,16 +31,18 @@ ENDIF (MSVC)
 
 ADD_EXECUTABLE(zstd ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c ${PROGRAMS_DIR}/bench.c ${PROGRAMS_DIR}/datagen.c ${PROGRAMS_DIR}/dibio.c ${PlatformDependResources})
 TARGET_LINK_LIBRARIES(zstd libzstd_shared)
-ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd zstdcat COMMENT "Creating zstdcat symlink")
-ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd unzstd COMMENT "Creating unzstd symlink")
+ADD_CUSTOM_TARGET(zstdcat ALL ${CMAKE_COMMAND} -E create_symlink zstd zstdcat DEPENDS zstd COMMENT "Creating zstdcat symlink")
+ADD_CUSTOM_TARGET(unzstd ALL ${CMAKE_COMMAND} -E create_symlink zstd unzstd DEPENDS zstd COMMENT "Creating unzstd symlink")
 INSTALL(TARGETS zstd RUNTIME DESTINATION "bin")
 INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/zstdcat DESTINATION "bin")
 INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/unzstd DESTINATION "bin")
 
 IF (UNIX)
-    FILE(COPY ${PROGRAMS_DIR}/zstd.1 DESTINATION .)
-    ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd.1 zstdcat.1 COMMENT "Creating zstdcat.1 symlink")
-    ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd.1 unzstd.1 COMMENT "Creating unzstd.1 symlink")
+    ADD_CUSTOM_TARGET(zstd.1 ALL
+        ${CMAKE_COMMAND} -E copy ${PROGRAMS_DIR}/zstd.1 .
+        COMMENT "Copying manpage zstd.1")
+    ADD_CUSTOM_TARGET(zstdcat.1 ALL ${CMAKE_COMMAND} -E create_symlink zstd.1 zstdcat.1 DEPENDS zstd.1 COMMENT "Creating zstdcat.1 symlink")
+    ADD_CUSTOM_TARGET(unzstd.1 ALL ${CMAKE_COMMAND} -E create_symlink zstd.1 unzstd.1 DEPENDS zstd.1 COMMENT "Creating unzstd.1 symlink")
     INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/zstd.1 DESTINATION "share/man/man1")
     INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/zstdcat.1 DESTINATION "share/man/man1")
     INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/unzstd.1 DESTINATION "share/man/man1")
@@ -61,7 +63,7 @@ IF (ZSTD_MULTITHREAD_SUPPORT)
         MESSAGE(SEND_ERROR "ZSTD currently does not support thread libraries other than pthreads")
     ENDIF()
 
-    ADD_CUSTOM_COMMAND(TARGET zstd POST_BUILD COMMAND ${CMAKE_COMMAND} -E create_symlink zstd zstdmt COMMENT "Creating zstdmt symlink")
+    ADD_CUSTOM_TARGET(zstdmt ALL ${CMAKE_COMMAND} -E create_symlink zstd zstdmt DEPENDS zstd COMMENT "Creating zstdmt symlink")
     INSTALL(FILES ${CMAKE_CURRENT_BINARY_DIR}/zstdmt DESTINATION "bin")
 ENDIF (ZSTD_MULTITHREAD_SUPPORT)
 

From cba4e79a9314b4bc216d25bf80b59f1d221c141f Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= 
Date: Wed, 19 Apr 2017 19:25:29 +0200
Subject: [PATCH 213/305] Create and install pkg-config file with cmake

---
 build/cmake/lib/CMakeLists.txt  | 12 ++++++++++++
 build/cmake/lib/pkgconfig.cmake |  1 +
 2 files changed, 13 insertions(+)
 create mode 100644 build/cmake/lib/pkgconfig.cmake

diff --git a/build/cmake/lib/CMakeLists.txt b/build/cmake/lib/CMakeLists.txt
index 7a345bf56..50618f4a8 100644
--- a/build/cmake/lib/CMakeLists.txt
+++ b/build/cmake/lib/CMakeLists.txt
@@ -130,8 +130,20 @@ IF (ZSTD_BUILD_STATIC)
 ENDIF (ZSTD_BUILD_STATIC)
 
 IF (UNIX)
+    # pkg-config
+    SET(PREFIX "${CMAKE_INSTALL_PREFIX}")
+    SET(LIBDIR "${CMAKE_INSTALL_PREFIX}/lib")
+    SET(INCLUDEDIR "${CMAKE_INSTALL_PREFIX}/include")
+    SET(VERSION "${LIBVER_MAJOR}.${LIBVER_MINOR}.${LIBVER_RELEASE}")
+    ADD_CUSTOM_TARGET(libzstd.pc ALL
+            ${CMAKE_COMMAND} -DIN="${LIBRARY_DIR}/libzstd.pc.in" -DOUT="libzstd.pc"
+            -DPREFIX="${PREFIX}" -DLIBDIR="${LIBDIR}" -DINCLUDEDIR="${INCLUDEDIR}" -DVERSION="${VERSION}"
+            -P "${CMAKE_SOURCE_DIR}/lib/pkgconfig.cmake"
+            COMMENT "Creating pkg-config file")
+
     # install target
     INSTALL(FILES ${LIBRARY_DIR}/zstd.h ${LIBRARY_DIR}/deprecated/zbuff.h ${LIBRARY_DIR}/dictBuilder/zdict.h DESTINATION "include")
+    INSTALL(FILES "${CMAKE_CURRENT_BINARY_DIR}/libzstd.pc" DESTINATION "share/pkgconfig")
     INSTALL(TARGETS libzstd_shared LIBRARY DESTINATION "lib")
     IF (ZSTD_BUILD_STATIC)
         INSTALL(TARGETS libzstd_static ARCHIVE DESTINATION "lib")
diff --git a/build/cmake/lib/pkgconfig.cmake b/build/cmake/lib/pkgconfig.cmake
new file mode 100644
index 000000000..5434ff75c
--- /dev/null
+++ b/build/cmake/lib/pkgconfig.cmake
@@ -0,0 +1 @@
+CONFIGURE_FILE("${IN}" "${OUT}" @ONLY)

From f49f760b41caea2ead424c8e8f2f8affef2a594b Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= 
Date: Wed, 19 Apr 2017 19:25:53 +0200
Subject: [PATCH 214/305] Test new cmake branches with Circle CI

---
 Makefile   | 2 +-
 circle.yml | 2 +-
 2 files changed, 2 insertions(+), 2 deletions(-)

diff --git a/Makefile b/Makefile
index 7e57e1680..ae2f68867 100644
--- a/Makefile
+++ b/Makefile
@@ -110,7 +110,7 @@ clean:
 #------------------------------------------------------------------------------
 ifneq (,$(filter $(shell uname),Linux Darwin GNU/kFreeBSD GNU FreeBSD DragonFly NetBSD))
 HOST_OS = POSIX
-CMAKE_PARAMS = -DZSTD_BUILD_CONTRIB:BOOL=ON -DZSTD_BUILD_STATIC:BOOL=ON -DZSTD_BUILD_TESTS:BOOL=ON
+CMAKE_PARAMS = -DZSTD_BUILD_CONTRIB:BOOL=ON -DZSTD_BUILD_STATIC:BOOL=ON -DZSTD_BUILD_TESTS:BOOL=ON -DZSTD_ZLIB_SUPPORT:BOOL=ON -DZSTD_LZMA_SUPPORT:BOOL=ON
 .PHONY: install uninstall travis-install clangtest gpptest armtest usan asan uasan
 
 install:
diff --git a/circle.yml b/circle.yml
index 298569d14..218e33bfc 100644
--- a/circle.yml
+++ b/circle.yml
@@ -3,7 +3,7 @@ dependencies:
     - sudo dpkg --add-architecture i386
     - sudo add-apt-repository -y ppa:ubuntu-toolchain-r/test; sudo apt-get -y -qq update
     - sudo apt-get -y install gcc-powerpc-linux-gnu gcc-arm-linux-gnueabi libc6-dev-armel-cross gcc-aarch64-linux-gnu libc6-dev-arm64-cross
-    - sudo apt-get -y install libstdc++-6-dev clang gcc g++ gcc-5 gcc-6
+    - sudo apt-get -y install libstdc++-6-dev clang gcc g++ gcc-5 gcc-6 zlib1g-dev liblzma-dev
     - sudo apt-get -y install linux-libc-dev:i386 libc6-dev-i386
 
 test:

From eb7371f1794378cad706734031279f4afbb29102 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Milan=20=C5=A0ev=C4=8D=C3=ADk?= 
Date: Wed, 19 Apr 2017 20:16:11 +0200
Subject: [PATCH 215/305] Change all SET_TARGET_PROPERTIES to SET_PROPERTY

SET_PROPERTY function can append to lists, whereas previously used
SET_TARGET_PROPERTIES cannot.
---
 build/cmake/contrib/pzstd/CMakeLists.txt |  4 ++--
 build/cmake/lib/CMakeLists.txt           |  4 ++--
 build/cmake/programs/CMakeLists.txt      | 12 ++++++------
 3 files changed, 10 insertions(+), 10 deletions(-)

diff --git a/build/cmake/contrib/pzstd/CMakeLists.txt b/build/cmake/contrib/pzstd/CMakeLists.txt
index b4fbe1689..71def02cd 100644
--- a/build/cmake/contrib/pzstd/CMakeLists.txt
+++ b/build/cmake/contrib/pzstd/CMakeLists.txt
@@ -21,8 +21,8 @@ SET(PZSTD_DIR ${ZSTD_SOURCE_DIR}/contrib/pzstd)
 INCLUDE_DIRECTORIES(${PROGRAMS_DIR} ${LIBRARY_DIR} ${LIBRARY_DIR}/common ${PZSTD_DIR})
 
 ADD_EXECUTABLE(pzstd ${PZSTD_DIR}/main.cpp ${PZSTD_DIR}/Options.cpp ${PZSTD_DIR}/Pzstd.cpp ${PZSTD_DIR}/SkippableFrame.cpp)
-SET_TARGET_PROPERTIES(pzstd PROPERTIES COMPILE_DEFINITIONS "NDEBUG")
-SET_TARGET_PROPERTIES(pzstd PROPERTIES COMPILE_OPTIONS "-Wno-shadow")
+SET_PROPERTY(TARGET pzstd APPEND PROPERTY COMPILE_DEFINITIONS "NDEBUG")
+SET_PROPERTY(TARGET pzstd APPEND PROPERTY COMPILE_OPTIONS "-Wno-shadow")
 
 SET(THREADS_PREFER_PTHREAD_FLAG ON)
 FIND_PACKAGE(Threads REQUIRED)
diff --git a/build/cmake/lib/CMakeLists.txt b/build/cmake/lib/CMakeLists.txt
index 50618f4a8..5e945ff12 100644
--- a/build/cmake/lib/CMakeLists.txt
+++ b/build/cmake/lib/CMakeLists.txt
@@ -97,9 +97,9 @@ ENDIF (ZSTD_BUILD_STATIC)
 
 # Add specific compile definitions for MSVC project
 IF (MSVC)
-    SET_TARGET_PROPERTIES(libzstd_shared PROPERTIES COMPILE_DEFINITIONS "ZSTD_DLL_EXPORT=1;ZSTD_HEAPMODE=0;_CONSOLE;_CRT_SECURE_NO_WARNINGS")
+    SET_PROPERTY(TARGET libzstd_shared APPEND PROPERTY COMPILE_DEFINITIONS "ZSTD_DLL_EXPORT=1;ZSTD_HEAPMODE=0;_CONSOLE;_CRT_SECURE_NO_WARNINGS")
     IF (ZSTD_BUILD_STATIC)
-        SET_TARGET_PROPERTIES(libzstd_static PROPERTIES COMPILE_DEFINITIONS "ZSTD_HEAPMODE=0;_CRT_SECURE_NO_WARNINGS")
+        SET_PROPERTY(TARGET libzstd_static APPEND PROPERTY COMPILE_DEFINITIONS "ZSTD_HEAPMODE=0;_CRT_SECURE_NO_WARNINGS")
     ENDIF (ZSTD_BUILD_STATIC)
 ENDIF (MSVC)
 
diff --git a/build/cmake/programs/CMakeLists.txt b/build/cmake/programs/CMakeLists.txt
index c3e68e01c..cb6aa9218 100644
--- a/build/cmake/programs/CMakeLists.txt
+++ b/build/cmake/programs/CMakeLists.txt
@@ -49,11 +49,11 @@ IF (UNIX)
 
     ADD_EXECUTABLE(zstd-frugal ${PROGRAMS_DIR}/zstdcli.c ${PROGRAMS_DIR}/fileio.c)
     TARGET_LINK_LIBRARIES(zstd-frugal libzstd_shared)
-    SET_TARGET_PROPERTIES(zstd-frugal PROPERTIES COMPILE_DEFINITIONS "ZSTD_NOBENCH;ZSTD_NODICT")
+    SET_PROPERTY(TARGET zstd-frugal APPEND PROPERTY COMPILE_DEFINITIONS "ZSTD_NOBENCH;ZSTD_NODICT")
 ENDIF (UNIX)
 
 IF (ZSTD_MULTITHREAD_SUPPORT)
-    SET_TARGET_PROPERTIES(zstd PROPERTIES COMPILE_DEFINITIONS "ZSTD_MULTITHREAD")
+    SET_PROPERTY(TARGET zstd APPEND PROPERTY COMPILE_DEFINITIONS "ZSTD_MULTITHREAD")
 
     SET(THREADS_PREFER_PTHREAD_FLAG ON)
     FIND_PACKAGE(Threads REQUIRED)
@@ -75,8 +75,8 @@ IF (ZSTD_ZLIB_SUPPORT)
 
     IF (ZLIB_FOUND)
         INCLUDE_DIRECTORIES(${ZLIB_INCLUDE_DIRS})
-	TARGET_LINK_LIBRARIES(zstd ${ZLIB_LIBRARIES})
-        SET_TARGET_PROPERTIES(zstd PROPERTIES COMPILE_DEFINITIONS "ZSTD_GZCOMPRESS;ZSTD_GZDECOMPRESS")
+        TARGET_LINK_LIBRARIES(zstd ${ZLIB_LIBRARIES})
+        SET_PROPERTY(TARGET zstd APPEND PROPERTY COMPILE_DEFINITIONS "ZSTD_GZCOMPRESS;ZSTD_GZDECOMPRESS")
     ELSE ()
         MESSAGE(SEND_ERROR "zlib library is missing")
     ENDIF ()
@@ -87,8 +87,8 @@ IF (ZSTD_LZMA_SUPPORT)
 
     IF (LIBLZMA_FOUND)
         INCLUDE_DIRECTORIES(${LIBLZMA_INCLUDE_DIRS})
-	TARGET_LINK_LIBRARIES(zstd ${LIBLZMA_LIBRARIES})
-	SET_TARGET_PROPERTIES(zstd PROPERTIES COMPILE_DEFINITIONS "ZSTD_LZMACOMPRESS;ZSTD_LZMADECOMPRESS")
+        TARGET_LINK_LIBRARIES(zstd ${LIBLZMA_LIBRARIES})
+        SET_PROPERTY(TARGET zstd APPEND PROPERTY COMPILE_DEFINITIONS "ZSTD_LZMACOMPRESS;ZSTD_LZMADECOMPRESS")
     ELSE ()
         MESSAGE(SEND_ERROR "lzma library is missing")
     ENDIF ()

From e348dad3053c0ffd801a2fc27a7b939570cbf737 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Thu, 20 Apr 2017 11:14:13 -0700
Subject: [PATCH 216/305] minor long line reformatting

---
 lib/compress/zstd_compress.c | 20 ++++++++++++++++----
 1 file changed, 16 insertions(+), 4 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 4ad978a4b..a95f4b930 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -510,8 +510,10 @@ static size_t ZSTD_compressLiterals (ZSTD_CCtx* zc,
     {   HUF_repeat repeat = zc->flagStaticHufTable;
         int const preferRepeat = zc->params.cParams.strategy < ZSTD_lazy ? srcSize <= 1024 : 0;
         if (repeat == HUF_repeat_valid && lhSize == 3) singleStream = 1;
-        cLitSize = singleStream ? HUF_compress1X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11, zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat)
-                                : HUF_compress4X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11, zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat);
+        cLitSize = singleStream ? HUF_compress1X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11,
+                                      zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat)
+                                : HUF_compress4X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11,
+                                      zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat);
         if (repeat != HUF_repeat_none) { hType = set_repeat; }    /* reused the existing table */
         else { zc->flagStaticHufTable = HUF_repeat_check; }       /* now have a table to reuse */
     }
@@ -856,7 +858,14 @@ static unsigned ZSTD_NbCommonBytes (register size_t val)
 #       elif defined(__GNUC__) && (__GNUC__ >= 3)
             return (__builtin_ctzll((U64)val) >> 3);
 #       else
-            static const int DeBruijnBytePos[64] = { 0, 0, 0, 0, 0, 1, 1, 2, 0, 3, 1, 3, 1, 4, 2, 7, 0, 2, 3, 6, 1, 5, 3, 5, 1, 3, 4, 4, 2, 5, 6, 7, 7, 0, 1, 2, 3, 3, 4, 6, 2, 6, 5, 5, 3, 4, 5, 6, 7, 1, 2, 4, 6, 4, 4, 5, 7, 2, 6, 5, 7, 6, 7, 7 };
+            static const int DeBruijnBytePos[64] = { 0, 0, 0, 0, 0, 1, 1, 2,
+                                                     0, 3, 1, 3, 1, 4, 2, 7,
+                                                     0, 2, 3, 6, 1, 5, 3, 5,
+                                                     1, 3, 4, 4, 2, 5, 6, 7,
+                                                     7, 0, 1, 2, 3, 3, 4, 6,
+                                                     2, 6, 5, 5, 3, 4, 5, 6,
+                                                     7, 1, 2, 4, 6, 4, 4, 5,
+                                                     7, 2, 6, 5, 7, 6, 7, 7 };
             return DeBruijnBytePos[((U64)((val & -(long long)val) * 0x0218A392CDABBD3FULL)) >> 58];
 #       endif
         } else { /* 32 bits */
@@ -867,7 +876,10 @@ static unsigned ZSTD_NbCommonBytes (register size_t val)
 #       elif defined(__GNUC__) && (__GNUC__ >= 3)
             return (__builtin_ctz((U32)val) >> 3);
 #       else
-            static const int DeBruijnBytePos[32] = { 0, 0, 3, 0, 3, 1, 3, 0, 3, 2, 2, 1, 3, 2, 0, 1, 3, 3, 1, 2, 2, 2, 2, 0, 3, 1, 2, 0, 1, 0, 1, 1 };
+            static const int DeBruijnBytePos[32] = { 0, 0, 3, 0, 3, 1, 3, 0,
+                                                     3, 2, 2, 1, 3, 2, 0, 1,
+                                                     3, 3, 1, 2, 2, 2, 2, 0,
+                                                     3, 1, 2, 0, 1, 0, 1, 1 };
             return DeBruijnBytePos[((U32)((val & -(S32)val) * 0x077CB531U)) >> 27];
 #       endif
         }

From c17e020c9aedbd421e19e3607a16d87559d29d97 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Thu, 20 Apr 2017 12:50:02 -0700
Subject: [PATCH 217/305] disable assert when compiling paramgrill

paramgrill is a benchmark calibration function.
Speed accuracy is critical, it cannot be altered by assert.
---
 lib/compress/zstd_compress.c | 60 ++++++++++++++++++++++++------------
 tests/Makefile               |  4 ++-
 2 files changed, 44 insertions(+), 20 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 891094e46..5f18121b3 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -1315,7 +1315,8 @@ void ZSTD_compressBlock_doubleFast_generic(ZSTD_CCtx* cctx,
         const BYTE* match = base + matchIndexS;
         hashLong[h2] = hashSmall[h] = current;   /* update hash tables */
 
-        if ((offset_1 > 0) & (MEM_read32(ip+1-offset_1) == MEM_read32(ip+1))) { /* note : by construction, offset_1 <= current */
+        assert(offset_1 <= current);   /* supposed guaranteed by construction */
+        if ((offset_1 > 0) & (MEM_read32(ip+1-offset_1) == MEM_read32(ip+1))) {
             mLength = ZSTD_count(ip+1+4, ip+1+4-offset_1, iend) + 4;
             ip++;
             ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH);
@@ -1893,7 +1894,11 @@ size_t ZSTD_HcFindBestMatch_generic (
         }
 
         /* save best solution */
-        if (currentMl > ml) { ml = currentMl; *offsetPtr = current - matchIndex + ZSTD_REP_MOVE; if (ip+currentMl == iLimit) break; /* best possible, and avoid read overflow*/ }
+        if (currentMl > ml) {
+            ml = currentMl;
+            *offsetPtr = current - matchIndex + ZSTD_REP_MOVE;
+            if (ip+currentMl == iLimit) break; /* best possible, avoids read overflow on next attempt */
+        }
 
         if (matchIndex <= minChain) break;
         matchIndex = NEXT_IN_CHAIN(matchIndex, chainMask);
@@ -2038,7 +2043,9 @@ void ZSTD_compressBlock_lazy_generic(ZSTD_CCtx* ctx,
 
         /* catch up */
         if (offset) {
-            while ((start>anchor) && (start>base+offset-ZSTD_REP_MOVE) && (start[-1] == start[-1-offset+ZSTD_REP_MOVE]))   /* only search for offset within prefix */
+            while ( (start > anchor)
+                 && (start > base+offset-ZSTD_REP_MOVE)
+                 && (start[-1] == start[-1-offset+ZSTD_REP_MOVE]) )  /* only search for offset within prefix */
                 { start--; matchLength++; }
             offset_2 = offset_1; offset_1 = (U32)(offset - ZSTD_REP_MOVE);
         }
@@ -2204,9 +2211,9 @@ void ZSTD_compressBlock_lazy_extDict_generic(ZSTD_CCtx* ctx,
                     if (MEM_read32(ip) == MEM_read32(repMatch)) {
                         /* repcode detected */
                         const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend;
-                        size_t repLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32;
-                        int gain2 = (int)(repLength * 4);
-                        int gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 1);
+                        size_t const repLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32;
+                        int const gain2 = (int)(repLength * 4);
+                        int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 1);
                         if ((repLength >= EQUAL_READ32) && (gain2 > gain1))
                             matchLength = repLength, offset = 0, start = ip;
                 }   }
@@ -2339,8 +2346,12 @@ typedef void (*ZSTD_blockCompressor) (ZSTD_CCtx* ctx, const void* src, size_t sr
 static ZSTD_blockCompressor ZSTD_selectBlockCompressor(ZSTD_strategy strat, int extDict)
 {
     static const ZSTD_blockCompressor blockCompressor[2][8] = {
-        { ZSTD_compressBlock_fast, ZSTD_compressBlock_doubleFast, ZSTD_compressBlock_greedy, ZSTD_compressBlock_lazy, ZSTD_compressBlock_lazy2, ZSTD_compressBlock_btlazy2, ZSTD_compressBlock_btopt, ZSTD_compressBlock_btopt2 },
-        { ZSTD_compressBlock_fast_extDict, ZSTD_compressBlock_doubleFast_extDict, ZSTD_compressBlock_greedy_extDict, ZSTD_compressBlock_lazy_extDict,ZSTD_compressBlock_lazy2_extDict, ZSTD_compressBlock_btlazy2_extDict, ZSTD_compressBlock_btopt_extDict, ZSTD_compressBlock_btopt2_extDict }
+        { ZSTD_compressBlock_fast, ZSTD_compressBlock_doubleFast, ZSTD_compressBlock_greedy,
+          ZSTD_compressBlock_lazy, ZSTD_compressBlock_lazy2, ZSTD_compressBlock_btlazy2,
+          ZSTD_compressBlock_btopt, ZSTD_compressBlock_btopt2 },
+        { ZSTD_compressBlock_fast_extDict, ZSTD_compressBlock_doubleFast_extDict, ZSTD_compressBlock_greedy_extDict,
+          ZSTD_compressBlock_lazy_extDict,ZSTD_compressBlock_lazy2_extDict, ZSTD_compressBlock_btlazy2_extDict,
+          ZSTD_compressBlock_btopt_extDict, ZSTD_compressBlock_btopt2_extDict }
     };
 
     return blockCompressor[extDict][(U32)strat];
@@ -2356,7 +2367,7 @@ static size_t ZSTD_compressBlock_internal(ZSTD_CCtx* zc, void* dst, size_t dstCa
     if (srcSize < MIN_CBLOCK_SIZE+ZSTD_blockHeaderSize+1) return 0;   /* don't even attempt compression below a certain srcSize */
     ZSTD_resetSeqStore(&(zc->seqStore));
     if (current > zc->nextToUpdate + 384)
-        zc->nextToUpdate = current - MIN(192, (U32)(current - zc->nextToUpdate - 384));   /* update tree not updated after finding very long rep matches */
+        zc->nextToUpdate = current - MIN(192, (U32)(current - zc->nextToUpdate - 384));   /* limited update after finding a very long match */
     blockCompressor(zc, src, srcSize);
     return ZSTD_compressSequences(zc, dst, dstCapacity, srcSize);
 }
@@ -2388,7 +2399,8 @@ static size_t ZSTD_compress_generic (ZSTD_CCtx* cctx,
         U32 const lastBlock = lastFrameChunk & (blockSize >= remaining);
         size_t cSize;
 
-        if (dstCapacity < ZSTD_blockHeaderSize + MIN_CBLOCK_SIZE) return ERROR(dstSize_tooSmall);   /* not enough space to store compressed block */
+        if (dstCapacity < ZSTD_blockHeaderSize + MIN_CBLOCK_SIZE)
+            return ERROR(dstSize_tooSmall);   /* not enough space to store compressed block */
         if (remaining < blockSize) blockSize = remaining;
 
         /* preemptive overflow correction */
@@ -2647,7 +2659,8 @@ static size_t ZSTD_loadZstdDictionary(ZSTD_CCtx* cctx, const void* dict, size_t
         if (FSE_isError(offcodeHeaderSize)) return ERROR(dictionary_corrupted);
         if (offcodeLog > OffFSELog) return ERROR(dictionary_corrupted);
         /* Defer checking offcodeMaxValue because we need to know the size of the dictionary content */
-        CHECK_E (FSE_buildCTable_wksp(cctx->offcodeCTable, offcodeNCount, offcodeMaxValue, offcodeLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted);
+        CHECK_E( FSE_buildCTable_wksp(cctx->offcodeCTable, offcodeNCount, offcodeMaxValue, offcodeLog, scratchBuffer, sizeof(scratchBuffer)),
+                 dictionary_corrupted);
         dictPtr += offcodeHeaderSize;
     }
 
@@ -2657,8 +2670,9 @@ static size_t ZSTD_loadZstdDictionary(ZSTD_CCtx* cctx, const void* dict, size_t
         if (FSE_isError(matchlengthHeaderSize)) return ERROR(dictionary_corrupted);
         if (matchlengthLog > MLFSELog) return ERROR(dictionary_corrupted);
         /* Every match length code must have non-zero probability */
-        CHECK_F (ZSTD_checkDictNCount(matchlengthNCount, matchlengthMaxValue, MaxML));
-        CHECK_E (FSE_buildCTable_wksp(cctx->matchlengthCTable, matchlengthNCount, matchlengthMaxValue, matchlengthLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted);
+        CHECK_F( ZSTD_checkDictNCount(matchlengthNCount, matchlengthMaxValue, MaxML));
+        CHECK_E( FSE_buildCTable_wksp(cctx->matchlengthCTable, matchlengthNCount, matchlengthMaxValue, matchlengthLog, scratchBuffer, sizeof(scratchBuffer)),
+                 dictionary_corrupted);
         dictPtr += matchlengthHeaderSize;
     }
 
@@ -2668,8 +2682,9 @@ static size_t ZSTD_loadZstdDictionary(ZSTD_CCtx* cctx, const void* dict, size_t
         if (FSE_isError(litlengthHeaderSize)) return ERROR(dictionary_corrupted);
         if (litlengthLog > LLFSELog) return ERROR(dictionary_corrupted);
         /* Every literal length code must have non-zero probability */
-        CHECK_F (ZSTD_checkDictNCount(litlengthNCount, litlengthMaxValue, MaxLL));
-        CHECK_E(FSE_buildCTable_wksp(cctx->litlengthCTable, litlengthNCount, litlengthMaxValue, litlengthLog, scratchBuffer, sizeof(scratchBuffer)), dictionary_corrupted);
+        CHECK_F( ZSTD_checkDictNCount(litlengthNCount, litlengthMaxValue, MaxLL));
+        CHECK_E( FSE_buildCTable_wksp(cctx->litlengthCTable, litlengthNCount, litlengthMaxValue, litlengthLog, scratchBuffer, sizeof(scratchBuffer)),
+                 dictionary_corrupted);
         dictPtr += litlengthHeaderSize;
     }
 
@@ -2829,7 +2844,8 @@ size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx,
     return ZSTD_compress_internal(ctx, dst, dstCapacity, src, srcSize, dict, dictSize, params);
 }
 
-size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, const void* dict, size_t dictSize, int compressionLevel)
+size_t ZSTD_compress_usingDict(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize,
+                               const void* dict, size_t dictSize, int compressionLevel)
 {
     ZSTD_parameters params = ZSTD_getParams(compressionLevel, srcSize, dict ? dictSize : 0);
     params.fParams.contentSizeFlag = 1;
@@ -3053,7 +3069,11 @@ size_t ZSTD_freeCStream(ZSTD_CStream* zcs)
 /*======   Initialization   ======*/
 
 size_t ZSTD_CStreamInSize(void)  { return ZSTD_BLOCKSIZE_ABSOLUTEMAX; }
-size_t ZSTD_CStreamOutSize(void) { return ZSTD_compressBound(ZSTD_BLOCKSIZE_ABSOLUTEMAX) + ZSTD_blockHeaderSize + 4 /* 32-bits hash */ ; }
+
+size_t ZSTD_CStreamOutSize(void)
+{
+    return ZSTD_compressBound(ZSTD_BLOCKSIZE_ABSOLUTEMAX) + ZSTD_blockHeaderSize + 4 /* 32-bits hash */ ;
+}
 
 static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize)
 {
@@ -3308,7 +3328,8 @@ size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output)
         /* flush whatever remains */
         size_t srcSize = 0;
         size_t sizeWritten = output->size - output->pos;
-        size_t const notEnded = ZSTD_compressStream_generic(zcs, ostart, &sizeWritten, &srcSize, &srcSize, zsf_end);  /* use a valid src address instead of NULL */
+        size_t const notEnded = ZSTD_compressStream_generic(zcs, ostart, &sizeWritten,
+                                     &srcSize /* use a valid src address instead of NULL */, &srcSize, zsf_end);
         size_t const remainingToFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize;
         op += sizeWritten;
         if (remainingToFlush) {
@@ -3318,7 +3339,8 @@ size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output)
         /* create epilogue */
         zcs->stage = zcss_final;
         zcs->outBuffContentSize = !notEnded ? 0 :
-            ZSTD_compressEnd(zcs->cctx, zcs->outBuff, zcs->outBuffSize, NULL, 0);  /* write epilogue, including final empty block, into outBuff */
+            /* write epilogue, including final empty block, into outBuff */
+            ZSTD_compressEnd(zcs->cctx, zcs->outBuff, zcs->outBuffSize, NULL, 0);
         if (ZSTD_isError(zcs->outBuffContentSize)) return zcs->outBuffContentSize;
     }
 
diff --git a/tests/Makefile b/tests/Makefile
index 70ef51878..f5260e50e 100644
--- a/tests/Makefile
+++ b/tests/Makefile
@@ -26,9 +26,10 @@ PYTHON ?= python3
 TESTARTEFACT := versionsTest namespaceTest
 
 
+DEBUGFLAG= -DZSTD_DEBUG=1
 CPPFLAGS+= -I$(ZSTDDIR) -I$(ZSTDDIR)/common -I$(ZSTDDIR)/compress \
            -I$(ZSTDDIR)/dictBuilder -I$(ZSTDDIR)/deprecated -I$(PRGDIR) \
-           -DZSTD_DEBUG=1
+           $(DEBUGFLAG)
 CFLAGS  ?= -O3
 CFLAGS  += -g -Wall -Wextra -Wcast-qual -Wcast-align -Wshadow \
            -Wstrict-aliasing=1 -Wswitch-enum -Wdeclaration-after-statement \
@@ -156,6 +157,7 @@ zstreamtest-dll : $(ZSTDDIR)/common/xxhash.c $(PRGDIR)/datagen.c zstreamtest.c
 	$(MAKE) -C $(ZSTDDIR) libzstd
 	$(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@$(EXT)
 
+paramgrill : DEBUGFLAG =
 paramgrill : $(ZSTD_FILES) $(PRGDIR)/datagen.c paramgrill.c
 	$(CC)      $(FLAGS) $^ -lm -o $@$(EXT)
 

From 1a96bec8dba9d8b8147259a703dd3b7f26584080 Mon Sep 17 00:00:00 2001
From: Michael Maltese 
Date: Thu, 20 Apr 2017 15:35:56 -0700
Subject: [PATCH 218/305] CMake: Set ZSTD_SOURCE_DIR from
 CMAKE_CURRENT_SOURCE_DIR

Instead of CMAKE_SOURCE_DIR, which is not correct when embedding Zstd
within a larger project.
---
 build/cmake/CMakeLists.txt | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/build/cmake/CMakeLists.txt b/build/cmake/CMakeLists.txt
index bf6890818..0aa7b3072 100644
--- a/build/cmake/CMakeLists.txt
+++ b/build/cmake/CMakeLists.txt
@@ -9,7 +9,7 @@
 
 PROJECT(zstd)
 CMAKE_MINIMUM_REQUIRED(VERSION 2.8.9)
-SET(ZSTD_SOURCE_DIR "${CMAKE_SOURCE_DIR}/../..")
+SET(ZSTD_SOURCE_DIR "${CMAKE_CURRENT_SOURCE_DIR}/../..")
 
 #-----------------------------------------------------------------------------
 # Add extra compilation flags

From 35186e65b04eb9844b63e15d1f5d1ed58bb497e6 Mon Sep 17 00:00:00 2001
From: Sean Purcell 
Date: Thu, 20 Apr 2017 16:48:54 -0700
Subject: [PATCH 219/305] Address comments and make sure all prototypes are
 rendered by gen_html

---
 contrib/seekable_format/zstd_seekable.h       | 7 ++-----
 contrib/seekable_format/zstdseek_decompress.c | 8 ++++----
 2 files changed, 6 insertions(+), 9 deletions(-)

diff --git a/contrib/seekable_format/zstd_seekable.h b/contrib/seekable_format/zstd_seekable.h
index 54b50fa50..71b4c0be7 100644
--- a/contrib/seekable_format/zstd_seekable.h
+++ b/contrib/seekable_format/zstd_seekable.h
@@ -16,8 +16,6 @@ static const unsigned ZSTD_seekTableFooterSize = 9;
 /* Limit the maximum size to avoid any potential issues storing the compressed size */
 #define ZSTD_SEEKABLE_MAX_FRAME_DECOMPRESSED_SIZE 0x80000000U
 
-#define ZSTD_SEEKABLE_FRAMEINDEX_TOOLARGE (0ULL-2)
-
 /*-****************************************************************************
 *  Seekable Format
 *
@@ -137,14 +135,14 @@ ZSTDLIB_API size_t ZSTD_seekable_initFile(ZSTD_seekable* zs, FILE* src);
 ZSTDLIB_API size_t ZSTD_seekable_decompress(ZSTD_seekable* zs, void* dst, size_t dstSize, unsigned long long offset);
 ZSTDLIB_API size_t ZSTD_seekable_decompressFrame(ZSTD_seekable* zs, void* dst, size_t dstSize, unsigned frameIndex);
 
+#define ZSTD_SEEKABLE_FRAMEINDEX_TOOLARGE (0ULL-2)
 /*===== Seek Table access functions =====*/
 ZSTDLIB_API unsigned ZSTD_seekable_getNumFrames(ZSTD_seekable* const zs);
 ZSTDLIB_API unsigned long long ZSTD_seekable_getFrameCompressedOffset(ZSTD_seekable* const zs, unsigned frameIndex);
 ZSTDLIB_API unsigned long long ZSTD_seekable_getFrameDecompressedOffset(ZSTD_seekable* const zs, unsigned frameIndex);
 ZSTDLIB_API size_t ZSTD_seekable_getFrameCompressedSize(ZSTD_seekable* const zs, unsigned frameIndex);
 ZSTDLIB_API size_t ZSTD_seekable_getFrameDecompressedSize(ZSTD_seekable* const zs, unsigned frameIndex);
-
-ZSTDLIB_API unsigned ZSTD_seekable_offsetToFrame(ZSTD_seekable* const zs, unsigned long long offset);
+ZSTDLIB_API unsigned ZSTD_seekable_offsetToFrameIndex(ZSTD_seekable* const zs, unsigned long long offset);
 
 /*===== Seekable advanced I/O API =====*/
 typedef int(ZSTD_seekable_read)(void* opaque, void* buffer, size_t n);
@@ -154,7 +152,6 @@ typedef struct {
     ZSTD_seekable_read* read;
     ZSTD_seekable_seek* seek;
 } ZSTD_seekable_customFile;
-
 ZSTDLIB_API size_t ZSTD_seekable_initAdvanced(ZSTD_seekable* zs, ZSTD_seekable_customFile src);
 
 #if defined (__cplusplus)
diff --git a/contrib/seekable_format/zstdseek_decompress.c b/contrib/seekable_format/zstdseek_decompress.c
index 9bcfea91b..a9e87577b 100644
--- a/contrib/seekable_format/zstdseek_decompress.c
+++ b/contrib/seekable_format/zstdseek_decompress.c
@@ -192,11 +192,11 @@ size_t ZSTD_seekable_free(ZSTD_seekable* zs)
     return 0;
 }
 
-/** ZSTD_seekable_offsetToFrame() :
+/** ZSTD_seekable_offsetToFrameIndex() :
  *  Performs a binary search to find the last frame with a decompressed offset
  *  <= pos
  *  @return : the frame's index */
-U32 ZSTD_seekable_offsetToFrame(ZSTD_seekable* const zs, U64 pos)
+U32 ZSTD_seekable_offsetToFrameIndex(ZSTD_seekable* const zs, U64 pos)
 {
     U32 lo = 0;
     U32 hi = zs->seekTable.tableLen;
@@ -373,7 +373,7 @@ size_t ZSTD_seekable_initAdvanced(ZSTD_seekable* zs, ZSTD_seekable_customFile sr
 
 size_t ZSTD_seekable_decompress(ZSTD_seekable* zs, void* dst, size_t len, U64 offset)
 {
-    U32 targetFrame = ZSTD_seekable_offsetToFrame(zs, offset);
+    U32 targetFrame = ZSTD_seekable_offsetToFrameIndex(zs, offset);
     do {
         /* check if we can continue from a previous decompress job */
         if (targetFrame != zs->curFrame || offset != zs->decompressedOffset) {
@@ -422,7 +422,7 @@ size_t ZSTD_seekable_decompress(ZSTD_seekable* zs, void* dst, size_t len, U64 of
 
                 if (zs->decompressedOffset < offset + len) {
                     /* go back to the start and force a reset of the stream */
-                    targetFrame = ZSTD_seekable_offsetToFrame(zs, zs->decompressedOffset);
+                    targetFrame = ZSTD_seekable_offsetToFrameIndex(zs, zs->decompressedOffset);
                 }
                 break;
             }

From e6fa70a0a1d8a229b0bf29218155cb3db53a03e7 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Thu, 20 Apr 2017 17:28:31 -0700
Subject: [PATCH 220/305] reorganized ZSTD_resetCCtx_internal()

clearer separation between variables and buffers
clearer buffers category
kept static buffers at the beginning, favoring cache locality
(it will be easier to add FSE tables there later)

This break a few assumptions that hashTable was always at the beginning.
This is fixed.
And remaining assumptions (namely that tables stand next to each other in memory)
are now tested with assert.
---
 lib/common/fse.h             |  4 +-
 lib/compress/zstd_compress.c | 98 ++++++++++++++++++++++--------------
 2 files changed, 62 insertions(+), 40 deletions(-)

diff --git a/lib/common/fse.h b/lib/common/fse.h
index 5e43215e3..4ecd8d741 100644
--- a/lib/common/fse.h
+++ b/lib/common/fse.h
@@ -550,9 +550,9 @@ MEM_STATIC void FSE_initCState2(FSE_CState_t* statePtr, const FSE_CTable* ct, U3
 
 MEM_STATIC void FSE_encodeSymbol(BIT_CStream_t* bitC, FSE_CState_t* statePtr, U32 symbol)
 {
-    const FSE_symbolCompressionTransform symbolTT = ((const FSE_symbolCompressionTransform*)(statePtr->symbolTT))[symbol];
+    FSE_symbolCompressionTransform const symbolTT = ((const FSE_symbolCompressionTransform*)(statePtr->symbolTT))[symbol];
     const U16* const stateTable = (const U16*)(statePtr->stateTable);
-    U32 nbBitsOut  = (U32)((statePtr->value + symbolTT.deltaNbBits) >> 16);
+    U32 const nbBitsOut  = (U32)((statePtr->value + symbolTT.deltaNbBits) >> 16);
     BIT_addBits(bitC, statePtr->value, nbBitsOut);
     statePtr->value = stateTable[ (statePtr->value >> nbBitsOut) + symbolTT.deltaFindState];
 }
diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 5f18121b3..e049aa076 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -282,9 +282,11 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
         void* ptr;
 
         /* Check if workSpace is large enough, alloc a new one if needed */
-        {   size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<workSpaceSize < neededSpace) {
                 zc->workSpaceSize = 0;
@@ -294,31 +296,37 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
                 zc->workSpaceSize = neededSpace;
         }   }
 
-        if (crp!=ZSTDcrp_noMemset) memset(zc->workSpace, 0, tableSpace);   /* reset tables only */
+        /* init params */
+        zc->params = params;
+        zc->blockSize = blockSize;
+        zc->frameContentSize = frameContentSize;
+        zc->consumedSrcSize = 0;
+
         XXH64_reset(&zc->xxhState, 0);
-        zc->hashLog3 = hashLog3;
-        zc->hashTable = (U32*)(zc->workSpace);
-        zc->chainTable = zc->hashTable + hSize;
-        zc->hashTable3 = zc->chainTable + chainSize;
-        ptr = zc->hashTable3 + h3Size;
-        zc->hufTable = (HUF_CElt*)ptr;
+        zc->stage = ZSTDcs_init;
+        zc->dictID = 0;
+        zc->loadedDictEnd = 0;
         zc->flagStaticTables = 0;
         zc->flagStaticHufTable = HUF_repeat_none;
-        ptr = ((U32*)ptr) + HUF_CTABLE_SIZE_U32(255);  /* note : HUF_CElt* is incomplete type, size is simulated using U32 */
-
         zc->nextToUpdate = 1;
         zc->nextSrc = NULL;
         zc->base = NULL;
         zc->dictBase = NULL;
         zc->dictLimit = 0;
         zc->lowLimit = 0;
-        zc->params = params;
-        zc->blockSize = blockSize;
-        zc->frameContentSize = frameContentSize;
-        zc->consumedSrcSize = 0;
         { int i; for (i=0; irep[i] = repStartValue[i]; }
+        zc->hashLog3 = hashLog3;
+        zc->seqStore.litLengthSum = 0;
 
+        ptr = zc->workSpace;
+
+        /* entropy space */
+        zc->hufTable = (HUF_CElt*)ptr;
+        ptr = (U32*)zc->hufTable + HUF_CTABLE_SIZE_U32(255);  /* note : HUF_CElt* is incomplete type, size is estimated via macro */
+
+        /* opt parser space */
         if ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) {
+            assert(((size_t)ptr & 3) == 0);  /* ensure ptr is properly aligned */
             zc->seqStore.litFreq = (U32*)ptr;
             zc->seqStore.litLengthFreq = zc->seqStore.litFreq + (1<seqStore.matchLengthFreq = zc->seqStore.litLengthFreq + (MaxLL+1);
@@ -328,8 +336,17 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
             ptr = zc->seqStore.matchTable + ZSTD_OPT_NUM+1;
             zc->seqStore.priceTable = (ZSTD_optimal_t*)ptr;
             ptr = zc->seqStore.priceTable + ZSTD_OPT_NUM+1;
-            zc->seqStore.litLengthSum = 0;
         }
+
+        /* table Space */
+        if (crp!=ZSTDcrp_noMemset) memset(ptr, 0, tableSpace);   /* reset tables only */
+        assert(((size_t)ptr & 3) == 0);  /* ensure ptr is properly aligned */
+        zc->hashTable = (U32*)(ptr);
+        zc->chainTable = zc->hashTable + hSize;
+        zc->hashTable3 = zc->chainTable + chainSize;
+        ptr = zc->hashTable3 + h3Size;
+
+        /* sequences storage */
         zc->seqStore.sequencesStart = (seqDef*)ptr;
         ptr = zc->seqStore.sequencesStart + maxNbSeq;
         zc->seqStore.llCode = (BYTE*) ptr;
@@ -337,10 +354,6 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
         zc->seqStore.ofCode = zc->seqStore.mlCode + maxNbSeq;
         zc->seqStore.litStart = zc->seqStore.ofCode + maxNbSeq;
 
-        zc->stage = ZSTDcs_init;
-        zc->dictID = 0;
-        zc->loadedDictEnd = 0;
-
         return 0;
     }
 }
@@ -373,10 +386,12 @@ size_t ZSTD_copyCCtx_internal(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx,
 
     /* copy tables */
     {   size_t const chainSize = (srcCCtx->params.cParams.strategy == ZSTD_fast) ? 0 : (1 << srcCCtx->params.cParams.chainLog);
-        size_t const hSize = ((size_t)1) << srcCCtx->params.cParams.hashLog;
+        size_t const hSize =  (size_t)1 << srcCCtx->params.cParams.hashLog;
         size_t const h3Size = (size_t)1 << srcCCtx->hashLog3;
         size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32);
-        memcpy(dstCCtx->workSpace, srcCCtx->workSpace, tableSpace);
+        assert((U32*)dstCCtx->chainTable == (U32*)dstCCtx->hashTable + hSize);  /* chainTable must follow hashTable */
+        assert((U32*)dstCCtx->hashTable3 == (U32*)dstCCtx->chainTable + chainSize);
+        memcpy(dstCCtx->hashTable, srcCCtx->hashTable, tableSpace);   /* presumes all tables follow each other */
     }
 
     /* copy dictionary offsets */
@@ -393,7 +408,7 @@ size_t ZSTD_copyCCtx_internal(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx,
     /* copy entropy tables */
     dstCCtx->flagStaticTables = srcCCtx->flagStaticTables;
     if (srcCCtx->flagStaticTables) {
-        memcpy(dstCCtx->litlengthCTable, srcCCtx->litlengthCTable, sizeof(dstCCtx->litlengthCTable));
+        memcpy(dstCCtx->litlengthCTable, srcCCtx->litlengthCTable, sizeof(dstCCtx->litlengthCTable));  /* depends on litlengthCTable being a table and not a pointer */
         memcpy(dstCCtx->matchlengthCTable, srcCCtx->matchlengthCTable, sizeof(dstCCtx->matchlengthCTable));
         memcpy(dstCCtx->offcodeCTable, srcCCtx->offcodeCTable, sizeof(dstCCtx->offcodeCTable));
     }
@@ -855,14 +870,20 @@ MEM_STATIC void ZSTD_storeSeq(seqStore_t* seqStorePtr, size_t litLength, const v
     seqStorePtr->lit += litLength;
 
     /* literal Length */
-    if (litLength>0xFFFF) { seqStorePtr->longLengthID = 1; seqStorePtr->longLengthPos = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); }
+    if (litLength>0xFFFF) {
+        seqStorePtr->longLengthID = 1;
+        seqStorePtr->longLengthPos = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart);
+    }
     seqStorePtr->sequences[0].litLength = (U16)litLength;
 
     /* match offset */
     seqStorePtr->sequences[0].offset = offsetCode + 1;
 
     /* match Length */
-    if (matchCode>0xFFFF) { seqStorePtr->longLengthID = 2; seqStorePtr->longLengthPos = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart); }
+    if (matchCode>0xFFFF) {
+        seqStorePtr->longLengthID = 2;
+        seqStorePtr->longLengthPos = (U32)(seqStorePtr->sequences - seqStorePtr->sequencesStart);
+    }
     seqStorePtr->sequences[0].matchLength = (U16)matchCode;
 
     seqStorePtr->sequences++;
@@ -976,7 +997,7 @@ static size_t ZSTD_count_2segments(const BYTE* ip, const BYTE* match, const BYTE
 ***************************************/
 static const U32 prime3bytes = 506832829U;
 static U32    ZSTD_hash3(U32 u, U32 h) { return ((u << (32-24)) * prime3bytes)  >> (32-h) ; }
-MEM_STATIC size_t ZSTD_hash3Ptr(const void* ptr, U32 h) { return ZSTD_hash3(MEM_readLE32(ptr), h); }   /* only in zstd_opt.h */
+MEM_STATIC size_t ZSTD_hash3Ptr(const void* ptr, U32 h) { return ZSTD_hash3(MEM_readLE32(ptr), h); } /* only in zstd_opt.h */
 
 static const U32 prime4bytes = 2654435761U;
 static U32    ZSTD_hash4(U32 u, U32 h) { return (u * prime4bytes) >> (32-h) ; }
@@ -1176,7 +1197,7 @@ static void ZSTD_compressBlock_fast_extDict_generic(ZSTD_CCtx* ctx,
         if ( (((U32)((dictLimit-1) - repIndex) >= 3) /* intentional underflow */ & (repIndex > lowestIndex))
            && (MEM_read32(repMatch) == MEM_read32(ip+1)) ) {
             const BYTE* repMatchEnd = repIndex < dictLimit ? dictEnd : iend;
-            mLength = ZSTD_count_2segments(ip+1+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repMatchEnd, lowPrefixPtr) + EQUAL_READ32;
+            mLength = ZSTD_count_2segments(ip+1+4, repMatch+4, iend, repMatchEnd, lowPrefixPtr) + 4;
             ip++;
             ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH);
         } else {
@@ -1188,7 +1209,7 @@ static void ZSTD_compressBlock_fast_extDict_generic(ZSTD_CCtx* ctx,
             {   const BYTE* matchEnd = matchIndex < dictLimit ? dictEnd : iend;
                 const BYTE* lowMatchPtr = matchIndex < dictLimit ? dictStart : lowPrefixPtr;
                 U32 offset;
-                mLength = ZSTD_count_2segments(ip+EQUAL_READ32, match+EQUAL_READ32, iend, matchEnd, lowPrefixPtr) + EQUAL_READ32;
+                mLength = ZSTD_count_2segments(ip+4, match+4, iend, matchEnd, lowPrefixPtr) + 4;
                 while (((ip>anchor) & (match>lowMatchPtr)) && (ip[-1] == match[-1])) { ip--; match--; mLength++; }   /* catch up */
                 offset = current - matchIndex;
                 offset_2 = offset_1;
@@ -1212,7 +1233,7 @@ static void ZSTD_compressBlock_fast_extDict_generic(ZSTD_CCtx* ctx,
                 if ( (((U32)((dictLimit-1) - repIndex2) >= 3) & (repIndex2 > lowestIndex))  /* intentional overflow */
                    && (MEM_read32(repMatch2) == MEM_read32(ip)) ) {
                     const BYTE* const repEnd2 = repIndex2 < dictLimit ? dictEnd : iend;
-                    size_t repLength2 = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch2+EQUAL_READ32, iend, repEnd2, lowPrefixPtr) + EQUAL_READ32;
+                    size_t const repLength2 = ZSTD_count_2segments(ip+4, repMatch2+4, iend, repEnd2, lowPrefixPtr) + 4;
                     U32 tmpOffset = offset_2; offset_2 = offset_1; offset_1 = tmpOffset;   /* swap offset_2 <=> offset_1 */
                     ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, repLength2-MINMATCH);
                     hashTable[ZSTD_hashPtr(ip, hBits, mls)] = current2;
@@ -1317,6 +1338,7 @@ void ZSTD_compressBlock_doubleFast_generic(ZSTD_CCtx* cctx,
 
         assert(offset_1 <= current);   /* supposed guaranteed by construction */
         if ((offset_1 > 0) & (MEM_read32(ip+1-offset_1) == MEM_read32(ip+1))) {
+            /* favor repcode */
             mLength = ZSTD_count(ip+1+4, ip+1+4-offset_1, iend) + 4;
             ip++;
             ZSTD_storeSeq(seqStorePtr, ip-anchor, anchor, 0, mLength-MINMATCH);
@@ -1327,15 +1349,15 @@ void ZSTD_compressBlock_doubleFast_generic(ZSTD_CCtx* cctx,
                 offset = (U32)(ip-matchLong);
                 while (((ip>anchor) & (matchLong>lowest)) && (ip[-1] == matchLong[-1])) { ip--; matchLong--; mLength++; } /* catch up */
             } else if ( (matchIndexS > lowestIndex) && (MEM_read32(match) == MEM_read32(ip)) ) {
-                size_t const h3 = ZSTD_hashPtr(ip+1, hBitsL, 8);
-                U32 const matchIndex3 = hashLong[h3];
-                const BYTE* match3 = base + matchIndex3;
-                hashLong[h3] = current + 1;
-                if ( (matchIndex3 > lowestIndex) && (MEM_read64(match3) == MEM_read64(ip+1)) ) {
-                    mLength = ZSTD_count(ip+9, match3+8, iend) + 8;
+                size_t const hl3 = ZSTD_hashPtr(ip+1, hBitsL, 8);
+                U32 const matchIndexL3 = hashLong[hl3];
+                const BYTE* matchL3 = base + matchIndexL3;
+                hashLong[hl3] = current + 1;
+                if ( (matchIndexL3 > lowestIndex) && (MEM_read64(matchL3) == MEM_read64(ip+1)) ) {
+                    mLength = ZSTD_count(ip+9, matchL3+8, iend) + 8;
                     ip++;
-                    offset = (U32)(ip-match3);
-                    while (((ip>anchor) & (match3>lowest)) && (ip[-1] == match3[-1])) { ip--; match3--; mLength++; } /* catch up */
+                    offset = (U32)(ip-matchL3);
+                    while (((ip>anchor) & (matchL3>lowest)) && (ip[-1] == matchL3[-1])) { ip--; matchL3--; mLength++; } /* catch up */
                 } else {
                     mLength = ZSTD_count(ip+4, match+4, iend) + 4;
                     offset = (U32)(ip-match);

From 7bb60b17d860e7a298dfb9984c15fa315a20a58c Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Thu, 20 Apr 2017 17:38:56 -0700
Subject: [PATCH 221/305] init entropy table pointers only once

per workSpace resize
---
 lib/compress/zstd_compress.c | 10 ++++++----
 1 file changed, 6 insertions(+), 4 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index e049aa076..ac0519fba 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -294,6 +294,12 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
                 zc->workSpace = ZSTD_malloc(neededSpace, zc->customMem);
                 if (zc->workSpace == NULL) return ERROR(memory_allocation);
                 zc->workSpaceSize = neededSpace;
+                ptr = zc->workSpace;
+
+                /* entropy space */
+                zc->hufTable = (HUF_CElt*)ptr;
+                ptr = (U32*)zc->hufTable + HUF_CTABLE_SIZE_U32(255);  /* note : HUF_CElt* is incomplete type, size is estimated via macro */
+
         }   }
 
         /* init params */
@@ -318,10 +324,6 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
         zc->hashLog3 = hashLog3;
         zc->seqStore.litLengthSum = 0;
 
-        ptr = zc->workSpace;
-
-        /* entropy space */
-        zc->hufTable = (HUF_CElt*)ptr;
         ptr = (U32*)zc->hufTable + HUF_CTABLE_SIZE_U32(255);  /* note : HUF_CElt* is incomplete type, size is estimated via macro */
 
         /* opt parser space */

From a34a39c1838552027342e859a5a3917db07ef700 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Thu, 20 Apr 2017 18:17:58 -0700
Subject: [PATCH 222/305] changed size evaluation of entropy tables

so that memcpy() does no longer depends on fse pointer being a static table
---
 lib/compress/zstd_compress.c | 11 ++++++++---
 1 file changed, 8 insertions(+), 3 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index ac0519fba..6855297fd 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -27,6 +27,11 @@ static const U32 g_searchStrength = 8;   /* control skip over incompressible dat
 #define HASH_READ_SIZE 8
 typedef enum { ZSTDcs_created=0, ZSTDcs_init, ZSTDcs_ongoing, ZSTDcs_ending } ZSTD_compressionStage_e;
 
+static size_t const offcodeCTable_size = FSE_CTABLE_SIZE_U32(OffFSELog, MaxOff) * sizeof(FSE_CTable);
+static size_t const matchlengthCTable_size = FSE_CTABLE_SIZE_U32(MLFSELog, MaxML) * sizeof(FSE_CTable);
+static size_t const litlengthCTable_size = FSE_CTABLE_SIZE_U32(LLFSELog, MaxLL) * sizeof(FSE_CTable);
+
+
 
 /*-*************************************
 *  Helper functions
@@ -410,9 +415,9 @@ size_t ZSTD_copyCCtx_internal(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx,
     /* copy entropy tables */
     dstCCtx->flagStaticTables = srcCCtx->flagStaticTables;
     if (srcCCtx->flagStaticTables) {
-        memcpy(dstCCtx->litlengthCTable, srcCCtx->litlengthCTable, sizeof(dstCCtx->litlengthCTable));  /* depends on litlengthCTable being a table and not a pointer */
-        memcpy(dstCCtx->matchlengthCTable, srcCCtx->matchlengthCTable, sizeof(dstCCtx->matchlengthCTable));
-        memcpy(dstCCtx->offcodeCTable, srcCCtx->offcodeCTable, sizeof(dstCCtx->offcodeCTable));
+        memcpy(dstCCtx->litlengthCTable, srcCCtx->litlengthCTable, litlengthCTable_size);
+        memcpy(dstCCtx->matchlengthCTable, srcCCtx->matchlengthCTable, matchlengthCTable_size);
+        memcpy(dstCCtx->offcodeCTable, srcCCtx->offcodeCTable, offcodeCTable_size);
     }
     dstCCtx->flagStaticHufTable = srcCCtx->flagStaticHufTable;
     if (srcCCtx->flagStaticHufTable) {

From 7f1fb955662d0df4425db0322b1fd3037edf42f1 Mon Sep 17 00:00:00 2001
From: Michael Maltese 
Date: Thu, 20 Apr 2017 15:46:44 -0700
Subject: [PATCH 223/305] CMake: namespace modules and set CMAKE_MODULE_PATH

---
 build/cmake/CMakeLists.txt                                   | 5 +++--
 ...aCompilationFlags.cmake => AddZstdCompilationFlags.cmake} | 4 ++--
 .../{GetLibraryVersion.cmake => GetZstdLibraryVersion.cmake} | 2 +-
 build/cmake/contrib/gen_html/CMakeLists.txt                  | 4 ++--
 build/cmake/lib/CMakeLists.txt                               | 4 ++--
 5 files changed, 10 insertions(+), 9 deletions(-)
 rename build/cmake/CMakeModules/{AddExtraCompilationFlags.cmake => AddZstdCompilationFlags.cmake} (98%)
 rename build/cmake/CMakeModules/{GetLibraryVersion.cmake => GetZstdLibraryVersion.cmake} (86%)

diff --git a/build/cmake/CMakeLists.txt b/build/cmake/CMakeLists.txt
index 0aa7b3072..5c4eca61c 100644
--- a/build/cmake/CMakeLists.txt
+++ b/build/cmake/CMakeLists.txt
@@ -10,12 +10,13 @@
 PROJECT(zstd)
 CMAKE_MINIMUM_REQUIRED(VERSION 2.8.9)
 SET(ZSTD_SOURCE_DIR "${CMAKE_CURRENT_SOURCE_DIR}/../..")
+LIST(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/CMakeModules")
 
 #-----------------------------------------------------------------------------
 # Add extra compilation flags
 #-----------------------------------------------------------------------------
-INCLUDE(CMakeModules/AddExtraCompilationFlags.cmake)
-ADD_EXTRA_COMPILATION_FLAGS()
+INCLUDE(AddZstdCompilationFlags)
+ADD_ZSTD_COMPILATION_FLAGS()
 
 #-----------------------------------------------------------------------------
 # Options
diff --git a/build/cmake/CMakeModules/AddExtraCompilationFlags.cmake b/build/cmake/CMakeModules/AddZstdCompilationFlags.cmake
similarity index 98%
rename from build/cmake/CMakeModules/AddExtraCompilationFlags.cmake
rename to build/cmake/CMakeModules/AddZstdCompilationFlags.cmake
index e099a01da..177db541c 100644
--- a/build/cmake/CMakeModules/AddExtraCompilationFlags.cmake
+++ b/build/cmake/CMakeModules/AddZstdCompilationFlags.cmake
@@ -19,7 +19,7 @@ function(EnableCompilerFlag _flag _C _CXX)
     endif ()
 endfunction()
 
-MACRO(ADD_EXTRA_COMPILATION_FLAGS)
+MACRO(ADD_ZSTD_COMPILATION_FLAGS)
     if (CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang" OR MINGW) #Not only UNIX but also WIN32 for MinGW
         #Set c++11 by default
         EnableCompilerFlag("-std=c++11" false true)
@@ -95,4 +95,4 @@ MACRO(ADD_EXTRA_COMPILATION_FLAGS)
     set(CMAKE_C_FLAGS_MINSIZEREL "${CMAKE_C_FLAGS_MINSIZEREL}" CACHE STRING "Updated flags" FORCE)
     set(CMAKE_C_FLAGS_RELWITHDEBINFO "${CMAKE_C_FLAGS_RELWITHDEBINFO}" CACHE STRING "Updated flags" FORCE)
 
-ENDMACRO(ADD_EXTRA_COMPILATION_FLAGS)
+ENDMACRO(ADD_ZSTD_COMPILATION_FLAGS)
diff --git a/build/cmake/CMakeModules/GetLibraryVersion.cmake b/build/cmake/CMakeModules/GetZstdLibraryVersion.cmake
similarity index 86%
rename from build/cmake/CMakeModules/GetLibraryVersion.cmake
rename to build/cmake/CMakeModules/GetZstdLibraryVersion.cmake
index 95d84a89f..8b6f394da 100644
--- a/build/cmake/CMakeModules/GetLibraryVersion.cmake
+++ b/build/cmake/CMakeModules/GetZstdLibraryVersion.cmake
@@ -1,4 +1,4 @@
-function(GetLibraryVersion _header _major _minor _release)
+function(GetZstdLibraryVersion _header _major _minor _release)
     # Read file content
     FILE(READ ${_header} CONTENT)
 
diff --git a/build/cmake/contrib/gen_html/CMakeLists.txt b/build/cmake/contrib/gen_html/CMakeLists.txt
index dff8c7a11..c10c62b54 100644
--- a/build/cmake/contrib/gen_html/CMakeLists.txt
+++ b/build/cmake/contrib/gen_html/CMakeLists.txt
@@ -11,7 +11,7 @@
 # ################################################################
 
 PROJECT(gen_html)
-INCLUDE(${CMAKE_SOURCE_DIR}/CMakeModules/GetLibraryVersion.cmake)
+INCLUDE(GetZstdLibraryVersion)
 
 SET(CMAKE_INCLUDE_CURRENT_DIR TRUE)
 
@@ -24,7 +24,7 @@ INCLUDE_DIRECTORIES(${PROGRAMS_DIR} ${LIBRARY_DIR} ${LIBRARY_DIR}/common ${GENHT
 
 ADD_EXECUTABLE(gen_html ${GENHTML_DIR}/gen_html.cpp)
 
-GetLibraryVersion(${LIBRARY_DIR}/zstd.h VMAJOR VMINOR VRELEASE)
+GetZstdLibraryVersion(${LIBRARY_DIR}/zstd.h VMAJOR VMINOR VRELEASE)
 SET(LIBVERSION "${VMAJOR}.${VMINOR}.${VRELEASE}")
 ADD_CUSTOM_TARGET(zstd_manual.html ALL
                   ${GENHTML_BINARY} "${LIBVERSION}" "${LIBRARY_DIR}/zstd.h" "${PROJECT_BINARY_DIR}/zstd_manual.html"
diff --git a/build/cmake/lib/CMakeLists.txt b/build/cmake/lib/CMakeLists.txt
index 5e945ff12..937ddbc50 100644
--- a/build/cmake/lib/CMakeLists.txt
+++ b/build/cmake/lib/CMakeLists.txt
@@ -11,7 +11,6 @@
 # ################################################################
 
 PROJECT(libzstd)
-INCLUDE(${CMAKE_SOURCE_DIR}/CMakeModules/GetLibraryVersion.cmake)
 
 SET(CMAKE_INCLUDE_CURRENT_DIR TRUE)
 OPTION(ZSTD_BUILD_STATIC "BUILD STATIC LIBRARIES" OFF)
@@ -21,7 +20,8 @@ SET(LIBRARY_DIR ${ZSTD_SOURCE_DIR}/lib)
 INCLUDE_DIRECTORIES(${LIBRARY_DIR} ${LIBRARY_DIR}/common)
 
 # Parse version
-GetLibraryVersion(${LIBRARY_DIR}/zstd.h LIBVER_MAJOR LIBVER_MINOR LIBVER_RELEASE)
+INCLUDE(GetZstdLibraryVersion)
+GetZstdLibraryVersion(${LIBRARY_DIR}/zstd.h LIBVER_MAJOR LIBVER_MINOR LIBVER_RELEASE)
 MESSAGE("ZSTD VERSION ${LIBVER_MAJOR}.${LIBVER_MINOR}.${LIBVER_RELEASE}")
 
 SET(Sources

From 554a13dd4b7b61920c9bd8bdc6df81f040f80871 Mon Sep 17 00:00:00 2001
From: Michael Maltese 
Date: Thu, 20 Apr 2017 15:47:31 -0700
Subject: [PATCH 224/305] CMake: various configure_file fixes to use
 CMAKE_CURRENT_SOURCE_DIR

---
 build/cmake/lib/CMakeLists.txt                 | 8 ++++----
 build/cmake/{ => lib}/cmake_uninstall.cmake.in | 0
 2 files changed, 4 insertions(+), 4 deletions(-)
 rename build/cmake/{ => lib}/cmake_uninstall.cmake.in (100%)

diff --git a/build/cmake/lib/CMakeLists.txt b/build/cmake/lib/CMakeLists.txt
index 937ddbc50..429d49449 100644
--- a/build/cmake/lib/CMakeLists.txt
+++ b/build/cmake/lib/CMakeLists.txt
@@ -138,7 +138,7 @@ IF (UNIX)
     ADD_CUSTOM_TARGET(libzstd.pc ALL
             ${CMAKE_COMMAND} -DIN="${LIBRARY_DIR}/libzstd.pc.in" -DOUT="libzstd.pc"
             -DPREFIX="${PREFIX}" -DLIBDIR="${LIBDIR}" -DINCLUDEDIR="${INCLUDEDIR}" -DVERSION="${VERSION}"
-            -P "${CMAKE_SOURCE_DIR}/lib/pkgconfig.cmake"
+            -P "${CMAKE_CURRENT_SOURCE_DIR}/pkgconfig.cmake"
             COMMENT "Creating pkg-config file")
 
     # install target
@@ -151,10 +151,10 @@ IF (UNIX)
 
     # uninstall target
     CONFIGURE_FILE(
-            "${CMAKE_SOURCE_DIR}/cmake_uninstall.cmake.in"
-            "${CMAKE_BINARY_DIR}/cmake_uninstall.cmake"
+            "${CMAKE_CURRENT_SOURCE_DIR}/cmake_uninstall.cmake.in"
+            "${CMAKE_CURRENT_BINARY_DIR}/cmake_uninstall.cmake"
             IMMEDIATE @ONLY)
 
     ADD_CUSTOM_TARGET(uninstall
-            COMMAND ${CMAKE_COMMAND} -P ${CMAKE_BINARY_DIR}/cmake_uninstall.cmake)
+            COMMAND ${CMAKE_COMMAND} -P ${CMAKE_CURRENT_BINARY_DIR}/cmake_uninstall.cmake)
 ENDIF (UNIX)
diff --git a/build/cmake/cmake_uninstall.cmake.in b/build/cmake/lib/cmake_uninstall.cmake.in
similarity index 100%
rename from build/cmake/cmake_uninstall.cmake.in
rename to build/cmake/lib/cmake_uninstall.cmake.in

From 377401f161e413962630e4f98a9e4eb55335cd7c Mon Sep 17 00:00:00 2001
From: Michael Maltese 
Date: Thu, 20 Apr 2017 15:54:17 -0700
Subject: [PATCH 225/305] CMake: don't recheck compile flags every time

Doesn't cause a problem when embedded within a larger project, but is
annoying.
---
 .../CMakeModules/AddZstdCompilationFlags.cmake    | 15 ++++++++-------
 1 file changed, 8 insertions(+), 7 deletions(-)

diff --git a/build/cmake/CMakeModules/AddZstdCompilationFlags.cmake b/build/cmake/CMakeModules/AddZstdCompilationFlags.cmake
index 177db541c..05ad86439 100644
--- a/build/cmake/CMakeModules/AddZstdCompilationFlags.cmake
+++ b/build/cmake/CMakeModules/AddZstdCompilationFlags.cmake
@@ -2,20 +2,21 @@ include(CheckCXXCompilerFlag)
 include(CheckCCompilerFlag)
 
 function(EnableCompilerFlag _flag _C _CXX)
-    message("Checking flag ${_flag}")
+    string(REGEX REPLACE "\\+" "PLUS" varname "${_flag}")
+    string(REGEX REPLACE "[^A-Za-z0-9]+" "_" varname "${varname}")
+    string(REGEX REPLACE "^_+" "" varname "${varname}")
+    string(TOUPPER "${varname}" varname)
     if (_C)
-        CHECK_C_COMPILER_FLAG(${_flag} C_FLAG)
-        if (C_FLAG)
+        CHECK_C_COMPILER_FLAG(${_flag} C_FLAG_${varname})
+        if (C_FLAG_${varname})
             set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${_flag}" CACHE INTERNAL "C Flags")
         endif ()
-        unset(C_FLAG CACHE)
     endif ()
     if (_CXX)
-        CHECK_CXX_COMPILER_FLAG(${_flag} CXX_FLAG)
-        if (CXX_FLAG)
+        CHECK_CXX_COMPILER_FLAG(${_flag} CXX_FLAG_${varname})
+        if (CXX_FLAG_${varname})
             set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${_flag}" CACHE INTERNAL "CXX Flags")
         endif ()
-        unset(CXX_FLAG CACHE)
     endif ()
 endfunction()
 

From 9eda436733bf41eca2c31e77b3b601aeca215786 Mon Sep 17 00:00:00 2001
From: Michael Maltese 
Date: Thu, 20 Apr 2017 15:55:48 -0700
Subject: [PATCH 226/305] CMake: don't modify global C_FLAGS and CXX_FLAGS

---
 .../AddZstdCompilationFlags.cmake             | 21 ++++---------------
 1 file changed, 4 insertions(+), 17 deletions(-)

diff --git a/build/cmake/CMakeModules/AddZstdCompilationFlags.cmake b/build/cmake/CMakeModules/AddZstdCompilationFlags.cmake
index 05ad86439..e812418e3 100644
--- a/build/cmake/CMakeModules/AddZstdCompilationFlags.cmake
+++ b/build/cmake/CMakeModules/AddZstdCompilationFlags.cmake
@@ -9,13 +9,13 @@ function(EnableCompilerFlag _flag _C _CXX)
     if (_C)
         CHECK_C_COMPILER_FLAG(${_flag} C_FLAG_${varname})
         if (C_FLAG_${varname})
-            set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${_flag}" CACHE INTERNAL "C Flags")
+            set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${_flag}" PARENT_SCOPE)
         endif ()
     endif ()
     if (_CXX)
         CHECK_CXX_COMPILER_FLAG(${_flag} CXX_FLAG_${varname})
         if (CXX_FLAG_${varname})
-            set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${_flag}" CACHE INTERNAL "CXX Flags")
+            set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${_flag}" PARENT_SCOPE)
         endif ()
     endif ()
 endfunction()
@@ -69,8 +69,7 @@ MACRO(ADD_ZSTD_COMPILATION_FLAGS)
         separate_arguments(${flag_var})
         list(REMOVE_DUPLICATES ${flag_var})
         string(REPLACE ";" " " ${flag_var} "${${flag_var}}")
-        set(${flag_var} "${${flag_var}}" CACHE STRING "common build flags" FORCE)
-    ENDFOREACH (flag_var)  
+    ENDFOREACH (flag_var)
 
     if (MSVC)
         # Replace /MT to /MD flag
@@ -81,19 +80,7 @@ MACRO(ADD_ZSTD_COMPILATION_FLAGS)
                  CMAKE_CXX_FLAGS_MINSIZEREL CMAKE_CXX_FLAGS_RELWITHDEBINFO)
             STRING(REGEX REPLACE "/MT" "/MD" ${flag_var} "${${flag_var}}")
             STRING(REGEX REPLACE "/O2" "/Ox" ${flag_var} "${${flag_var}}")
-        ENDFOREACH (flag_var)      
+        ENDFOREACH (flag_var)
     endif ()
 
-    set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}" CACHE STRING "Updated flags" FORCE)
-    set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG}" CACHE STRING "Updated flags" FORCE)
-    set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE}" CACHE STRING "Updated flags" FORCE)
-    set(CMAKE_CXX_FLAGS_MINSIZEREL "${CMAKE_CXX_FLAGS_MINSIZEREL}" CACHE STRING "Updated flags" FORCE)
-    set(CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO}" CACHE STRING "Updated flags" FORCE)
-
-    set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS}" CACHE STRING "Updated flags" FORCE)
-    set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG}" CACHE STRING "Updated flags" FORCE)
-    set(CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE}" CACHE STRING "Updated flags" FORCE)
-    set(CMAKE_C_FLAGS_MINSIZEREL "${CMAKE_C_FLAGS_MINSIZEREL}" CACHE STRING "Updated flags" FORCE)
-    set(CMAKE_C_FLAGS_RELWITHDEBINFO "${CMAKE_C_FLAGS_RELWITHDEBINFO}" CACHE STRING "Updated flags" FORCE)
-
 ENDMACRO(ADD_ZSTD_COMPILATION_FLAGS)

From 71ddeb67b1998cc48424dcc524d7e1f9bf568fd6 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Thu, 20 Apr 2017 22:54:54 -0700
Subject: [PATCH 227/305] made room in workspace for FSE tables

still need to be transferred from CCtx into workspace
---
 lib/compress/zstd_compress.c | 79 +++++++++++++++++----------------
 lib/zstd.h                   | 86 ++++++++++++++++++------------------
 2 files changed, 84 insertions(+), 81 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 6855297fd..4530cff9a 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -27,10 +27,11 @@ static const U32 g_searchStrength = 8;   /* control skip over incompressible dat
 #define HASH_READ_SIZE 8
 typedef enum { ZSTDcs_created=0, ZSTDcs_init, ZSTDcs_ongoing, ZSTDcs_ending } ZSTD_compressionStage_e;
 
+/* entropy tables always have same size */
+static size_t const hufCTable_size = HUF_CTABLE_SIZE(255);
+static size_t const litlengthCTable_size = FSE_CTABLE_SIZE_U32(LLFSELog, MaxLL) * sizeof(FSE_CTable);
 static size_t const offcodeCTable_size = FSE_CTABLE_SIZE_U32(OffFSELog, MaxOff) * sizeof(FSE_CTable);
 static size_t const matchlengthCTable_size = FSE_CTABLE_SIZE_U32(MLFSELog, MaxML) * sizeof(FSE_CTable);
-static size_t const litlengthCTable_size = FSE_CTABLE_SIZE_U32(LLFSELog, MaxLL) * sizeof(FSE_CTable);
-
 
 
 /*-*************************************
@@ -94,9 +95,9 @@ struct ZSTD_CCtx_s {
     U32* hashTable;
     U32* hashTable3;
     U32* chainTable;
-    HUF_CElt* hufTable;
-    U32 flagStaticTables;
-    HUF_repeat flagStaticHufTable;
+    HUF_repeat hufCTable_repeatMode;
+    HUF_CElt* hufCTable;
+    U32 fseCTables_ready;
     FSE_CTable offcodeCTable  [FSE_CTABLE_SIZE_U32(OffFSELog, MaxOff)];
     FSE_CTable matchlengthCTable[FSE_CTABLE_SIZE_U32(MLFSELog, MaxML)];
     FSE_CTable litlengthCTable  [FSE_CTABLE_SIZE_U32(LLFSELog, MaxLL)];
@@ -221,11 +222,13 @@ size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams)
     size_t const hSize = ((size_t)1) << cParams.hashLog;
     U32    const hashLog3 = (cParams.searchLength>3) ? 0 : MIN(ZSTD_HASHLOG3_MAX, cParams.windowLog);
     size_t const h3Size = ((size_t)1) << hashLog3;
+    size_t const entropySpace = hufCTable_size + litlengthCTable_size
+                              + offcodeCTable_size + matchlengthCTable_size;
     size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32);
 
     size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<params)) {
-            zc->flagStaticTables = 0;
-            zc->flagStaticHufTable = HUF_repeat_none;
+            zc->fseCTables_ready = 0;
+            zc->hufCTable_repeatMode = HUF_repeat_none;
             return ZSTD_continueCCtx(zc, params, frameContentSize);
         }
 
@@ -287,12 +290,12 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
         void* ptr;
 
         /* Check if workSpace is large enough, alloc a new one if needed */
-        {   size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<workSpaceSize < neededSpace) {
                 zc->workSpaceSize = 0;
                 ZSTD_free(zc->workSpace, zc->customMem);
@@ -302,8 +305,8 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
                 ptr = zc->workSpace;
 
                 /* entropy space */
-                zc->hufTable = (HUF_CElt*)ptr;
-                ptr = (U32*)zc->hufTable + HUF_CTABLE_SIZE_U32(255);  /* note : HUF_CElt* is incomplete type, size is estimated via macro */
+                zc->hufCTable = (HUF_CElt*)ptr;
+                ptr = (char*)zc->hufCTable + hufCTable_size;  /* note : HUF_CElt* is incomplete type, size is estimated via macro */
 
         }   }
 
@@ -317,8 +320,8 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
         zc->stage = ZSTDcs_init;
         zc->dictID = 0;
         zc->loadedDictEnd = 0;
-        zc->flagStaticTables = 0;
-        zc->flagStaticHufTable = HUF_repeat_none;
+        zc->fseCTables_ready = 0;
+        zc->hufCTable_repeatMode = HUF_repeat_none;
         zc->nextToUpdate = 1;
         zc->nextSrc = NULL;
         zc->base = NULL;
@@ -329,7 +332,7 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
         zc->hashLog3 = hashLog3;
         zc->seqStore.litLengthSum = 0;
 
-        ptr = (U32*)zc->hufTable + HUF_CTABLE_SIZE_U32(255);  /* note : HUF_CElt* is incomplete type, size is estimated via macro */
+        ptr = (char*)zc->hufCTable + hufCTable_size;  /* note : HUF_CElt* is incomplete type, size is estimated via macro */
 
         /* opt parser space */
         if ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) {
@@ -413,15 +416,15 @@ size_t ZSTD_copyCCtx_internal(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx,
     dstCCtx->dictID       = srcCCtx->dictID;
 
     /* copy entropy tables */
-    dstCCtx->flagStaticTables = srcCCtx->flagStaticTables;
-    if (srcCCtx->flagStaticTables) {
+    dstCCtx->fseCTables_ready = srcCCtx->fseCTables_ready;
+    if (srcCCtx->fseCTables_ready) {
         memcpy(dstCCtx->litlengthCTable, srcCCtx->litlengthCTable, litlengthCTable_size);
         memcpy(dstCCtx->matchlengthCTable, srcCCtx->matchlengthCTable, matchlengthCTable_size);
         memcpy(dstCCtx->offcodeCTable, srcCCtx->offcodeCTable, offcodeCTable_size);
     }
-    dstCCtx->flagStaticHufTable = srcCCtx->flagStaticHufTable;
-    if (srcCCtx->flagStaticHufTable) {
-        memcpy(dstCCtx->hufTable, srcCCtx->hufTable, HUF_CTABLE_SIZE(255));
+    dstCCtx->hufCTable_repeatMode = srcCCtx->hufCTable_repeatMode;
+    if (srcCCtx->hufCTable_repeatMode) {
+        memcpy(dstCCtx->hufCTable, srcCCtx->hufCTable, hufCTable_size);
     }
 
     return 0;
@@ -549,28 +552,28 @@ static size_t ZSTD_compressLiterals (ZSTD_CCtx* zc,
 
     /* small ? don't even attempt compression (speed opt) */
 #   define LITERAL_NOENTROPY 63
-    {   size_t const minLitSize = zc->flagStaticHufTable == HUF_repeat_valid ? 6 : LITERAL_NOENTROPY;
+    {   size_t const minLitSize = zc->hufCTable_repeatMode == HUF_repeat_valid ? 6 : LITERAL_NOENTROPY;
         if (srcSize <= minLitSize) return ZSTD_noCompressLiterals(dst, dstCapacity, src, srcSize);
     }
 
     if (dstCapacity < lhSize+1) return ERROR(dstSize_tooSmall);   /* not enough space for compression */
-    {   HUF_repeat repeat = zc->flagStaticHufTable;
+    {   HUF_repeat repeat = zc->hufCTable_repeatMode;
         int const preferRepeat = zc->params.cParams.strategy < ZSTD_lazy ? srcSize <= 1024 : 0;
         if (repeat == HUF_repeat_valid && lhSize == 3) singleStream = 1;
         cLitSize = singleStream ? HUF_compress1X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11,
-                                      zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat)
+                                      zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufCTable, &repeat, preferRepeat)
                                 : HUF_compress4X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11,
-                                      zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufTable, &repeat, preferRepeat);
+                                      zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufCTable, &repeat, preferRepeat);
         if (repeat != HUF_repeat_none) { hType = set_repeat; }    /* reused the existing table */
-        else { zc->flagStaticHufTable = HUF_repeat_check; }       /* now have a table to reuse */
+        else { zc->hufCTable_repeatMode = HUF_repeat_check; }       /* now have a table to reuse */
     }
 
     if ((cLitSize==0) | (cLitSize >= srcSize - minGain)) {
-        zc->flagStaticHufTable = HUF_repeat_none;
+        zc->hufCTable_repeatMode = HUF_repeat_none;
         return ZSTD_noCompressLiterals(dst, dstCapacity, src, srcSize);
     }
     if (cLitSize==1) {
-        zc->flagStaticHufTable = HUF_repeat_none;
+        zc->hufCTable_repeatMode = HUF_repeat_none;
         return ZSTD_compressRleLiteralsBlock(dst, dstCapacity, src, srcSize);
     }
 
@@ -694,7 +697,7 @@ MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc,
             *op++ = llCodeTable[0];
             FSE_buildCTable_rle(CTable_LitLength, (BYTE)max);
             LLtype = set_rle;
-        } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) {
+        } else if ((zc->fseCTables_ready) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) {
             LLtype = set_repeat;
         } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (LL_defaultNormLog-1)))) {
             FSE_buildCTable_wksp(CTable_LitLength, LL_defaultNorm, MaxLL, LL_defaultNormLog, scratchBuffer, sizeof(scratchBuffer));
@@ -718,7 +721,7 @@ MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc,
             *op++ = ofCodeTable[0];
             FSE_buildCTable_rle(CTable_OffsetBits, (BYTE)max);
             Offtype = set_rle;
-        } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) {
+        } else if ((zc->fseCTables_ready) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) {
             Offtype = set_repeat;
         } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (OF_defaultNormLog-1)))) {
             FSE_buildCTable_wksp(CTable_OffsetBits, OF_defaultNorm, MaxOff, OF_defaultNormLog, scratchBuffer, sizeof(scratchBuffer));
@@ -742,7 +745,7 @@ MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc,
             *op++ = *mlCodeTable;
             FSE_buildCTable_rle(CTable_MatchLength, (BYTE)max);
             MLtype = set_rle;
-        } else if ((zc->flagStaticTables) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) {
+        } else if ((zc->fseCTables_ready) && (nbSeq < MAX_SEQ_FOR_STATIC_FSE)) {
             MLtype = set_repeat;
         } else if ((nbSeq < MIN_SEQ_FOR_DYNAMIC_FSE) || (mostFrequent < (nbSeq >> (ML_defaultNormLog-1)))) {
             FSE_buildCTable_wksp(CTable_MatchLength, ML_defaultNorm, MaxML, ML_defaultNormLog, scratchBuffer, sizeof(scratchBuffer));
@@ -760,7 +763,7 @@ MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc,
     }   }
 
     *seqHead = (BYTE)((LLtype<<6) + (Offtype<<4) + (MLtype<<2));
-    zc->flagStaticTables = 0;
+    zc->fseCTables_ready = 0;
 
     /* Encoding Sequences */
     {   BIT_CStream_t blockStream;
@@ -839,7 +842,7 @@ _check_compressibility:
     {   size_t const minGain = ZSTD_minGain(srcSize);
         size_t const maxCSize = srcSize - minGain;
         if ((size_t)(op-ostart) >= maxCSize) {
-            zc->flagStaticHufTable = HUF_repeat_none;
+            zc->hufCTable_repeatMode = HUF_repeat_none;
             return 0;
     }   }
 
@@ -2678,7 +2681,7 @@ static size_t ZSTD_loadZstdDictionary(ZSTD_CCtx* cctx, const void* dict, size_t
     cctx->dictID = cctx->params.fParams.noDictIDFlag ? 0 :  MEM_readLE32(dictPtr);
     dictPtr += 4;
 
-    {   size_t const hufHeaderSize = HUF_readCTable(cctx->hufTable, 255, dictPtr, dictEnd-dictPtr);
+    {   size_t const hufHeaderSize = HUF_readCTable(cctx->hufCTable, 255, dictPtr, dictEnd-dictPtr);
         if (HUF_isError(hufHeaderSize)) return ERROR(dictionary_corrupted);
         dictPtr += hufHeaderSize;
     }
@@ -2738,8 +2741,8 @@ static size_t ZSTD_loadZstdDictionary(ZSTD_CCtx* cctx, const void* dict, size_t
                 if (cctx->rep[u] > dictContentSize) return ERROR(dictionary_corrupted);
         }   }
 
-        cctx->flagStaticTables = 1;
-        cctx->flagStaticHufTable = HUF_repeat_valid;
+        cctx->fseCTables_ready = 1;
+        cctx->hufCTable_repeatMode = HUF_repeat_valid;
         return ZSTD_loadDictionaryContent(cctx, dictPtr, dictContentSize);
     }
 }
diff --git a/lib/zstd.h b/lib/zstd.h
index 5101fd26c..a99e497fc 100644
--- a/lib/zstd.h
+++ b/lib/zstd.h
@@ -71,48 +71,48 @@ ZSTDLIB_API unsigned ZSTD_versionNumber(void);   /**< library version number; to
 *  Simple API
 ***************************************/
 /*! ZSTD_compress() :
-    Compresses `src` content as a single zstd compressed frame into already allocated `dst`.
-    Hint : compression runs faster if `dstCapacity` >=  `ZSTD_compressBound(srcSize)`.
-    @return : compressed size written into `dst` (<= `dstCapacity),
-              or an error code if it fails (which can be tested using ZSTD_isError()). */
+ *  Compresses `src` content as a single zstd compressed frame into already allocated `dst`.
+ *  Hint : compression runs faster if `dstCapacity` >=  `ZSTD_compressBound(srcSize)`.
+ *  @return : compressed size written into `dst` (<= `dstCapacity),
+ *            or an error code if it fails (which can be tested using ZSTD_isError()). */
 ZSTDLIB_API size_t ZSTD_compress( void* dst, size_t dstCapacity,
                             const void* src, size_t srcSize,
                                   int compressionLevel);
 
 /*! ZSTD_decompress() :
-    `compressedSize` : must be the _exact_ size of some number of compressed and/or skippable frames.
-    `dstCapacity` is an upper bound of originalSize.
-    If user cannot imply a maximum upper bound, it's better to use streaming mode to decompress data.
-    @return : the number of bytes decompressed into `dst` (<= `dstCapacity`),
-              or an errorCode if it fails (which can be tested using ZSTD_isError()). */
+ *  `compressedSize` : must be the _exact_ size of some number of compressed and/or skippable frames.
+ *  `dstCapacity` is an upper bound of originalSize.
+ *  If user cannot imply a maximum upper bound, it's better to use streaming mode to decompress data.
+ *  @return : the number of bytes decompressed into `dst` (<= `dstCapacity`),
+ *            or an errorCode if it fails (which can be tested using ZSTD_isError()). */
 ZSTDLIB_API size_t ZSTD_decompress( void* dst, size_t dstCapacity,
                               const void* src, size_t compressedSize);
 
 /*! ZSTD_getDecompressedSize() :
-*   NOTE: This function is planned to be obsolete, in favour of ZSTD_getFrameContentSize.
-*   ZSTD_getFrameContentSize functions the same way, returning the decompressed size of a single
-*   frame, but distinguishes empty frames from frames with an unknown size, or errors.
-*
-*   Additionally, ZSTD_findDecompressedSize can be used instead.  It can handle multiple
-*   concatenated frames in one buffer, and so is more general.
-*   As a result however, it requires more computation and entire frames to be passed to it,
-*   as opposed to ZSTD_getFrameContentSize which requires only a single frame's header.
-*
-*   'src' is the start of a zstd compressed frame.
-*   @return : content size to be decompressed, as a 64-bits value _if known_, 0 otherwise.
-*    note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode.
-*             When `return==0`, data to decompress could be any size.
-*             In which case, it's necessary to use streaming mode to decompress data.
-*             Optionally, application can still use ZSTD_decompress() while relying on implied limits.
-*             (For example, data may be necessarily cut into blocks <= 16 KB).
-*    note 2 : decompressed size is always present when compression is done with ZSTD_compress()
-*    note 3 : decompressed size can be very large (64-bits value),
-*             potentially larger than what local system can handle as a single memory segment.
-*             In which case, it's necessary to use streaming mode to decompress data.
-*    note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified.
-*             Always ensure result fits within application's authorized limits.
-*             Each application can set its own limits.
-*    note 5 : when `return==0`, if precise failure cause is needed, use ZSTD_getFrameParams() to know more. */
+ *  NOTE: This function is planned to be obsolete, in favour of ZSTD_getFrameContentSize.
+ *  ZSTD_getFrameContentSize functions the same way, returning the decompressed size of a single
+ *  frame, but distinguishes empty frames from frames with an unknown size, or errors.
+ *
+ *  Additionally, ZSTD_findDecompressedSize can be used instead.  It can handle multiple
+ *  concatenated frames in one buffer, and so is more general.
+ *  As a result however, it requires more computation and entire frames to be passed to it,
+ *  as opposed to ZSTD_getFrameContentSize which requires only a single frame's header.
+ *
+ *  'src' is the start of a zstd compressed frame.
+ *  @return : content size to be decompressed, as a 64-bits value _if known_, 0 otherwise.
+ *   note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode.
+ *            When `return==0`, data to decompress could be any size.
+ *            In which case, it's necessary to use streaming mode to decompress data.
+ *            Optionally, application can still use ZSTD_decompress() while relying on implied limits.
+ *            (For example, data may be necessarily cut into blocks <= 16 KB).
+ *   note 2 : decompressed size is always present when compression is done with ZSTD_compress()
+ *   note 3 : decompressed size can be very large (64-bits value),
+ *            potentially larger than what local system can handle as a single memory segment.
+ *            In which case, it's necessary to use streaming mode to decompress data.
+ *   note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified.
+ *            Always ensure result fits within application's authorized limits.
+ *            Each application can set its own limits.
+ *   note 5 : when `return==0`, if precise failure cause is needed, use ZSTD_getFrameParams() to know more. */
 ZSTDLIB_API unsigned long long ZSTD_getDecompressedSize(const void* src, size_t srcSize);
 
 
@@ -127,29 +127,29 @@ ZSTDLIB_API const char* ZSTD_getErrorName(size_t code);     /*!< provides readab
 *  Explicit memory management
 ***************************************/
 /*= Compression context
-*   When compressing many times,
-*   it is recommended to allocate a context just once, and re-use it for each successive compression operation.
-*   This will make workload friendlier for system's memory.
-*   Use one context per thread for parallel execution in multi-threaded environments. */
+ *  When compressing many times,
+ *  it is recommended to allocate a context just once, and re-use it for each successive compression operation.
+ *  This will make workload friendlier for system's memory.
+ *  Use one context per thread for parallel execution in multi-threaded environments. */
 typedef struct ZSTD_CCtx_s ZSTD_CCtx;
 ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx(void);
 ZSTDLIB_API size_t     ZSTD_freeCCtx(ZSTD_CCtx* cctx);
 
 /*! ZSTD_compressCCtx() :
-    Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). */
+ *  Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). */
 ZSTDLIB_API size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel);
 
 /*= Decompression context
-*   When decompressing many times,
-*   it is recommended to allocate a context just once, and re-use it for each successive compression operation.
-*   This will make workload friendlier for system's memory.
-*   Use one context per thread for parallel execution in multi-threaded environments. */
+ *  When decompressing many times,
+ *  it is recommended to allocate a context just once, and re-use it for each successive compression operation.
+ *  This will make workload friendlier for system's memory.
+ *  Use one context per thread for parallel execution in multi-threaded environments. */
 typedef struct ZSTD_DCtx_s ZSTD_DCtx;
 ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx(void);
 ZSTDLIB_API size_t     ZSTD_freeDCtx(ZSTD_DCtx* dctx);
 
 /*! ZSTD_decompressDCtx() :
-*   Same as ZSTD_decompress(), requires an allocated ZSTD_DCtx (see ZSTD_createDCtx()). */
+ *  Same as ZSTD_decompress(), requires an allocated ZSTD_DCtx (see ZSTD_createDCtx()). */
 ZSTDLIB_API size_t ZSTD_decompressDCtx(ZSTD_DCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
 
 

From 71aaa32c3c41a9a02de561b82bf0d3e8190744f0 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Thu, 20 Apr 2017 23:03:38 -0700
Subject: [PATCH 228/305] transferred FSE tables from CCtx into workspace

Saved 5 KB from CCtx
---
 lib/compress/zstd_compress.c | 19 ++++++++++++++-----
 1 file changed, 14 insertions(+), 5 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 4530cff9a..783d2661f 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -98,9 +98,9 @@ struct ZSTD_CCtx_s {
     HUF_repeat hufCTable_repeatMode;
     HUF_CElt* hufCTable;
     U32 fseCTables_ready;
-    FSE_CTable offcodeCTable  [FSE_CTABLE_SIZE_U32(OffFSELog, MaxOff)];
-    FSE_CTable matchlengthCTable[FSE_CTABLE_SIZE_U32(MLFSELog, MaxML)];
-    FSE_CTable litlengthCTable  [FSE_CTABLE_SIZE_U32(LLFSELog, MaxLL)];
+    FSE_CTable* offcodeCTable;
+    FSE_CTable* matchlengthCTable;
+    FSE_CTable* litlengthCTable;
     unsigned tmpCounters[HUF_WORKSPACE_SIZE_U32];
 };
 
@@ -307,7 +307,11 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
                 /* entropy space */
                 zc->hufCTable = (HUF_CElt*)ptr;
                 ptr = (char*)zc->hufCTable + hufCTable_size;  /* note : HUF_CElt* is incomplete type, size is estimated via macro */
-
+                zc->offcodeCTable = (FSE_CTable*) ptr;
+                ptr = (char*)ptr + offcodeCTable_size;
+                zc->matchlengthCTable = ptr;
+                ptr = (char*)ptr + matchlengthCTable_size;
+                zc->litlengthCTable = ptr;
         }   }
 
         /* init params */
@@ -332,7 +336,12 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
         zc->hashLog3 = hashLog3;
         zc->seqStore.litLengthSum = 0;
 
-        ptr = (char*)zc->hufCTable + hufCTable_size;  /* note : HUF_CElt* is incomplete type, size is estimated via macro */
+        /* ensure entropy tables are close together at the beginning */
+        assert((void*)zc->hufCTable == zc->workSpace);
+        assert((char*)zc->offcodeCTable == (char*)zc->hufCTable + hufCTable_size);
+        assert((char*)zc->matchlengthCTable == (char*)zc->offcodeCTable + offcodeCTable_size);
+        assert((char*)zc->litlengthCTable == (char*)zc->matchlengthCTable + matchlengthCTable_size);
+        ptr = (char*)zc->litlengthCTable + litlengthCTable_size;
 
         /* opt parser space */
         if ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) {

From a408645f500078c903bf0c73ec760bffc866ef0b Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Thu, 20 Apr 2017 23:09:39 -0700
Subject: [PATCH 229/305] made some room for entropy scratch space

---
 lib/compress/zstd_compress.c | 7 +++++--
 1 file changed, 5 insertions(+), 2 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 783d2661f..63266df93 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -32,6 +32,7 @@ static size_t const hufCTable_size = HUF_CTABLE_SIZE(255);
 static size_t const litlengthCTable_size = FSE_CTABLE_SIZE_U32(LLFSELog, MaxLL) * sizeof(FSE_CTable);
 static size_t const offcodeCTable_size = FSE_CTABLE_SIZE_U32(OffFSELog, MaxOff) * sizeof(FSE_CTable);
 static size_t const matchlengthCTable_size = FSE_CTABLE_SIZE_U32(MLFSELog, MaxML) * sizeof(FSE_CTable);
+static size_t const entropyScratchSpace = HUF_WORKSPACE_SIZE;
 
 
 /*-*************************************
@@ -223,7 +224,8 @@ size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams)
     U32    const hashLog3 = (cParams.searchLength>3) ? 0 : MIN(ZSTD_HASHLOG3_MAX, cParams.windowLog);
     size_t const h3Size = ((size_t)1) << hashLog3;
     size_t const entropySpace = hufCTable_size + litlengthCTable_size
-                              + offcodeCTable_size + matchlengthCTable_size;
+                              + offcodeCTable_size + matchlengthCTable_size
+                              + entropyScratchSpace;
     size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32);
 
     size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<
Date: Thu, 20 Apr 2017 23:21:19 -0700
Subject: [PATCH 230/305] transferred entropy scratch space from CCtx into
 workSpace

Saved 6 KB
---
 lib/compress/zstd_compress.c | 22 +++++++++++++---------
 1 file changed, 13 insertions(+), 9 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 63266df93..4e3760b54 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -32,7 +32,7 @@ static size_t const hufCTable_size = HUF_CTABLE_SIZE(255);
 static size_t const litlengthCTable_size = FSE_CTABLE_SIZE_U32(LLFSELog, MaxLL) * sizeof(FSE_CTable);
 static size_t const offcodeCTable_size = FSE_CTABLE_SIZE_U32(OffFSELog, MaxOff) * sizeof(FSE_CTable);
 static size_t const matchlengthCTable_size = FSE_CTABLE_SIZE_U32(MLFSELog, MaxML) * sizeof(FSE_CTable);
-static size_t const entropyScratchSpace = HUF_WORKSPACE_SIZE;
+static size_t const entropyScratchSpace_size = HUF_WORKSPACE_SIZE;
 
 
 /*-*************************************
@@ -102,7 +102,7 @@ struct ZSTD_CCtx_s {
     FSE_CTable* offcodeCTable;
     FSE_CTable* matchlengthCTable;
     FSE_CTable* litlengthCTable;
-    unsigned tmpCounters[HUF_WORKSPACE_SIZE_U32];
+    unsigned* tmpCounters;
 };
 
 ZSTD_CCtx* ZSTD_createCCtx(void)
@@ -225,7 +225,7 @@ size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams)
     size_t const h3Size = ((size_t)1) << hashLog3;
     size_t const entropySpace = hufCTable_size + litlengthCTable_size
                               + offcodeCTable_size + matchlengthCTable_size
-                              + entropyScratchSpace;
+                              + entropyScratchSpace_size;
     size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32);
 
     size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<hufCTable + hufCTable_size;  /* note : HUF_CElt* is incomplete type, size is estimated via macro */
                 zc->offcodeCTable = (FSE_CTable*) ptr;
                 ptr = (char*)ptr + offcodeCTable_size;
-                zc->matchlengthCTable = ptr;
+                zc->matchlengthCTable = (FSE_CTable*) ptr;
                 ptr = (char*)ptr + matchlengthCTable_size;
-                zc->litlengthCTable = ptr;
+                zc->litlengthCTable = (FSE_CTable*) ptr;
+                ptr = (char*)ptr + litlengthCTable_size;
+                assert(((size_t)ptr & 3) == 0);   /* ensure correct alignment */
+                zc->tmpCounters = (unsigned*) ptr;
         }   }
 
         /* init params */
@@ -344,7 +347,8 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
         assert((char*)zc->offcodeCTable == (char*)zc->hufCTable + hufCTable_size);
         assert((char*)zc->matchlengthCTable == (char*)zc->offcodeCTable + offcodeCTable_size);
         assert((char*)zc->litlengthCTable == (char*)zc->matchlengthCTable + matchlengthCTable_size);
-        ptr = (char*)zc->litlengthCTable + litlengthCTable_size;
+        assert((char*)zc->tmpCounters == (char*)zc->litlengthCTable + litlengthCTable_size);
+        ptr = (char*)zc->tmpCounters + entropyScratchSpace_size;
 
         /* opt parser space */
         if ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) {
@@ -573,9 +577,9 @@ static size_t ZSTD_compressLiterals (ZSTD_CCtx* zc,
         int const preferRepeat = zc->params.cParams.strategy < ZSTD_lazy ? srcSize <= 1024 : 0;
         if (repeat == HUF_repeat_valid && lhSize == 3) singleStream = 1;
         cLitSize = singleStream ? HUF_compress1X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11,
-                                      zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufCTable, &repeat, preferRepeat)
+                                      zc->tmpCounters, entropyScratchSpace_size, zc->hufCTable, &repeat, preferRepeat)
                                 : HUF_compress4X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11,
-                                      zc->tmpCounters, sizeof(zc->tmpCounters), zc->hufCTable, &repeat, preferRepeat);
+                                      zc->tmpCounters, entropyScratchSpace_size, zc->hufCTable, &repeat, preferRepeat);
         if (repeat != HUF_repeat_none) { hType = set_repeat; }    /* reused the existing table */
         else { zc->hufCTable_repeatMode = HUF_repeat_check; }       /* now have a table to reuse */
     }

From d0b1846cf47d3989ce7fb3afb12734c695dddb4b Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Fri, 21 Apr 2017 10:59:36 -0700
Subject: [PATCH 231/305] ignore more cmake build artefacts

---
 build/cmake/lib/.gitignore      | 2 ++
 build/cmake/programs/.gitignore | 2 ++
 2 files changed, 4 insertions(+)
 create mode 100644 build/cmake/lib/.gitignore

diff --git a/build/cmake/lib/.gitignore b/build/cmake/lib/.gitignore
new file mode 100644
index 000000000..a4444c8d3
--- /dev/null
+++ b/build/cmake/lib/.gitignore
@@ -0,0 +1,2 @@
+# cmake build artefact
+libzstd.pc
diff --git a/build/cmake/programs/.gitignore b/build/cmake/programs/.gitignore
index f04c5b429..ae3a8a356 100644
--- a/build/cmake/programs/.gitignore
+++ b/build/cmake/programs/.gitignore
@@ -1,3 +1,5 @@
 # produced by make
 zstd
 zstd-frugal
+unzstd
+zstdcat

From 230d7acc7d3d0b08ba2b09acf1bda93bb5c9aa62 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Fri, 21 Apr 2017 11:38:13 -0700
Subject: [PATCH 232/305] cli : add support for --threads=# command

updated documentation
add relevant test case
---
 programs/zstd.1    | 23 +++++++++++++----------
 programs/zstd.1.md | 41 +++++++++++++++++++++--------------------
 programs/zstdcli.c |  1 +
 tests/playTests.sh |  1 +
 4 files changed, 36 insertions(+), 30 deletions(-)

diff --git a/programs/zstd.1 b/programs/zstd.1
index b84ab3324..999dc8169 100644
--- a/programs/zstd.1
+++ b/programs/zstd.1
@@ -5,7 +5,7 @@
 \fBzstd\fR \- zstd, zstdmt, unzstd, zstdcat \- Compress or decompress \.zst files
 .
 .SH "SYNOPSIS"
-\fBzstd\fR [\fIOPTIONS\fR] [\-|] [\-o ]
+\fBzstd\fR [\fIOPTIONS\fR] [\-|\fIINPUT\-FILE\fR] [\-o \fIOUTPUT\-FILE\fR]
 .
 .P
 \fBzstdmt\fR is equivalent to \fBzstd \-T0\fR
@@ -100,8 +100,8 @@ Use FILEs as a training set to create a dictionary\. The training set should con
 unlocks high compression levels 20+ (maximum 22), using a lot more memory\. Note that decompression will also require more memory when using these levels\.
 .
 .TP
-\fB\-T#\fR
-Compress using # threads (default: 1)\. If \fB#\fR is 0, attempt to detect the number of physical CPU cores and compress with that many threads\. This modified does nothing if \fBzstd\fR was compiled without multithread support\.
+\fB\-T#\fR, \fB\-\-threads=#\fR
+Compress using \fB#\fR threads (default: 1)\. If \fB#\fR is 0, attempt to detect and use the number of physical CPU cores\. This modifier does nothing if \fBzstd\fR is compiled without multithread support\.
 .
 .TP
 \fB\-D file\fR
@@ -113,7 +113,7 @@ do not store dictionary ID within frame header (dictionary compression)\. The de
 .
 .TP
 \fB\-o file\fR
-save result into \fBfile\fR (only possible with a single INPUT\-FILE)
+save result into \fBfile\fR (only possible with a single \fIINPUT\-FILE\fR)
 .
 .TP
 \fB\-f\fR, \fB\-\-force\fR
@@ -172,7 +172,7 @@ use FILEs as training set to create a dictionary\. The training set should conta
 .
 .TP
 \fB\-o file\fR
-dictionary saved into \fBfile\fR (default: dictionary)
+dictionary saved into \fBfile\fR (default name: dictionary)
 .
 .TP
 \fB\-\-maxdict=#\fR
@@ -198,15 +198,18 @@ Example: \fB\-\-train \-\-cover=k=64,d=8 FILEs\fR\.
 If \fIsteps\fR is not specified, the default value of 32 is used\. If \fIk\fR is not specified, the \fIk\fR values in [16, 2048] are checked for each value of \fId\fR\. If \fId\fR is not specified, the values checked are [6, 8, \.\.\., 16]\.
 .
 .IP
-Runs the cover dictionary builder for each parameter set and saves the optimal parameters and dictionary\. Prints the optimal parameters and writes the optimal dictionary to the output file\. Supports multithreading if \fBzstd\fR is compiled with threading support\.
+Runs the cover dictionary builder for each parameter set and saves the optimal parameters and dictionary\. Prints optimal parameters and writes optimal dictionary into output file\. Supports multithreading if \fBzstd\fR is compiled with threading support\.
 .
 .IP
 The parameter \fIk\fR is more sensitive than \fId\fR, and is faster to optimize over\. Suggested use is to run with a \fIsteps\fR <= 32 with neither \fIk\fR nor \fId\fR set\. Once it completes, use the value of \fId\fR it selects with a higher \fIsteps\fR (in the range [256, 1024])\.
 .
 .IP
+Examples :
+.
+.IP
 \fBzstd \-\-train \-\-optimize\-cover FILEs\fR
 .
-.br
+.IP
 \fBzstd \-\-train \-\-optimize\-cover=d=d,steps=512 FILEs\fR
 .
 .SH "BENCHMARK"
@@ -233,9 +236,6 @@ set process priority to real\-time
 .
 .SH "ADVANCED COMPRESSION OPTIONS"
 .
-.SS "\-B#:"
-Select the size of each compression job\. This parameter is available only when multi\-threading is enabled\. Default value is \fB4 * windowSize\fR, which means it varies depending on compression level\. \fB\-B#\fR makes it possible to select a custom value\. Note that job size must respect a minimum value which is enforced transparently\. This minimum is either 1 MB, or \fBoverlapSize\fR, whichever is largest\.
-.
 .SS "\-\-zstd[=options]:"
 \fBzstd\fR provides 22 predefined compression levels\. The selected or default predefined compression level can be changed with advanced compression options\. The \fIoptions\fR are provided as a comma\-separated list\. You may specify only the options you want to change and the rest will be taken from the selected or default compression level\. The list of available \fIoptions\fR:
 .
@@ -310,6 +310,9 @@ Determine \fBoverlapSize\fR, amount of data reloaded from previous job\. This pa
 .IP
 The minimum \fIovlog\fR is 0, and the maximum is 9\. 0 means "no overlap", hence completely independent jobs\. 9 means "full overlap", meaning up to \fBwindowSize\fR is reloaded from previous job\. Reducing \fIovlog\fR by 1 reduces the amount of reload by a factor 2\. Default \fIovlog\fR is 6, which means "reload \fBwindowSize / 8\fR"\. Exception : the maximum compression level (22) has a default \fIovlog\fR of 9\.
 .
+.SS "\-B#:"
+Select the size of each compression job\. This parameter is available only when multi\-threading is enabled\. Default value is \fB4 * windowSize\fR, which means it varies depending on compression level\. \fB\-B#\fR makes it possible to select a custom value\. Note that job size must respect a minimum value which is enforced transparently\. This minimum is either 1 MB, or \fBoverlapSize\fR, whichever is largest\.
+.
 .SS "Example"
 The following parameters sets advanced compression options to those of predefined level 19 for files bigger than 256 KB:
 .
diff --git a/programs/zstd.1.md b/programs/zstd.1.md
index 4b918d8fd..f2d04d16f 100644
--- a/programs/zstd.1.md
+++ b/programs/zstd.1.md
@@ -4,7 +4,7 @@ zstd(1) -- zstd, zstdmt, unzstd, zstdcat - Compress or decompress .zst files
 SYNOPSIS
 --------
 
-`zstd` [*OPTIONS*] [-|<INPUT-FILE>] [-o <OUTPUT-FILE>]
+`zstd` [*OPTIONS*] [-|_INPUT-FILE_] [-o _OUTPUT-FILE_]
 
 `zstdmt` is equivalent to `zstd -T0`
 
@@ -101,11 +101,10 @@ the last one takes effect.
 * `--ultra`:
     unlocks high compression levels 20+ (maximum 22), using a lot more memory.
     Note that decompression will also require more memory when using these levels.
-* `-T#`:
-    Compress using # threads (default: 1).
-    If `#` is 0, attempt to detect the number of physical CPU cores and compress with
-    that many threads.
-    This modified does nothing if `zstd` was compiled without multithread support.
+* `-T#`, `--threads=#`:
+    Compress using `#` threads (default: 1).
+    If `#` is 0, attempt to detect and use the number of physical CPU cores.
+    This modifier does nothing if `zstd` is compiled without multithread support.
 * `-D file`:
     use `file` as Dictionary to compress or decompress FILE(s)
 * `--nodictID`:
@@ -113,7 +112,7 @@ the last one takes effect.
     The decoder will have to rely on implicit knowledge about which dictionary to use,
     it won't be able to check if it's correct.
 * `-o file`:
-    save result into `file` (only possible with a single INPUT-FILE)
+    save result into `file` (only possible with a single _INPUT-FILE_)
 * `-f`, `--force`:
     overwrite output without prompting, and (de)compress symbolic links
 * `-c`, `--stdout`:
@@ -164,7 +163,7 @@ Typical gains range from 10% (at 64KB) to x5 better (at <1KB).
     and weight typically 100x the target dictionary size
     (for example, 10 MB for a 100 KB dictionary).
 * `-o file`:
-    dictionary saved into `file` (default: dictionary)
+    dictionary saved into `file` (default name: dictionary)
 * `--maxdict=#`:
     limit dictionary to specified size (default : (112640)
 * `--dictID=#`:
@@ -198,9 +197,9 @@ Typical gains range from 10% (at 64KB) to x5 better (at <1KB).
     value of _d_.
     If _d_ is not specified, the values checked are [6, 8, ..., 16].
 
-    Runs the cover dictionary builder for each parameter set and saves the
-    optimal parameters and dictionary.
-    Prints the optimal parameters and writes the optimal dictionary to the output file.
+    Runs the cover dictionary builder for each parameter set
+    and saves the optimal parameters and dictionary.
+    Prints optimal parameters and writes optimal dictionary into output file.
     Supports multithreading if `zstd` is compiled with threading support.
 
     The parameter _k_ is more sensitive than _d_, and is faster to optimize over.
@@ -208,7 +207,10 @@ Typical gains range from 10% (at 64KB) to x5 better (at <1KB).
     Once it completes, use the value of _d_ it selects with a higher _steps_
     (in the range [256, 1024]).
 
-    `zstd --train --optimize-cover FILEs` 
+ Examples : + + `zstd --train --optimize-cover FILEs` + `zstd --train --optimize-cover=d=d,steps=512 FILEs` @@ -229,14 +231,6 @@ BENCHMARK ADVANCED COMPRESSION OPTIONS ---------------------------- -### -B#: -Select the size of each compression job. -This parameter is available only when multi-threading is enabled. -Default value is `4 * windowSize`, which means it varies depending on compression level. -`-B#` makes it possible to select a custom value. -Note that job size must respect a minimum value which is enforced transparently. -This minimum is either 1 MB, or `overlapSize`, whichever is largest. - ### --zstd[=options]: `zstd` provides 22 predefined compression levels. The selected or default predefined compression level can be changed with @@ -319,6 +313,13 @@ The list of available _options_: Default _ovlog_ is 6, which means "reload `windowSize / 8`". Exception : the maximum compression level (22) has a default _ovlog_ of 9. +### -B#: +Select the size of each compression job. +This parameter is available only when multi-threading is enabled. +Default value is `4 * windowSize`, which means it varies depending on compression level. +`-B#` makes it possible to select a custom value. +Note that job size must respect a minimum value which is enforced transparently. +This minimum is either 1 MB, or `overlapSize`, whichever is largest. ### Example The following parameters sets advanced compression options to those of diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 6cbe7383f..76f55de53 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -420,6 +420,7 @@ int main(int argCount, const char* argv[]) continue; } #endif + if (longCommandWArg(&argument, "--threads=")) { nbThreads = readU32FromChar(&argument); continue; } if (longCommandWArg(&argument, "--memlimit=")) { memLimit = readU32FromChar(&argument); continue; } if (longCommandWArg(&argument, "--memory=")) { memLimit = readU32FromChar(&argument); continue; } if (longCommandWArg(&argument, "--memlimit-decompress=")) { memLimit = readU32FromChar(&argument); continue; } diff --git a/tests/playTests.sh b/tests/playTests.sh index deeebee2e..a91e9e8bb 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -471,6 +471,7 @@ then $ECHO "\n**** zstdmt round-trip tests **** " roundTripTest -g4M "1 -T0" roundTripTest -g8M "3 -T2" + roundTripTest -g8000K "2 --threads=2" fileRoundTripTest -g4M "19 -T2 -B1M" else $ECHO "\n**** no multithreading, skipping zstdmt tests **** " From 11dc940e726012c8341679eeba44b6a876a49466 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Fri, 21 Apr 2017 12:23:06 -0700 Subject: [PATCH 233/305] Add parallel processing example for seekable API --- contrib/seekable_format/examples/.gitignore | 1 + contrib/seekable_format/examples/Makefile | 7 +- .../examples/parallel_processing.c | 195 ++++++++++++++++++ .../examples/seekable_decompression.c | 2 +- contrib/seekable_format/zstdseek_decompress.c | 7 +- 5 files changed, 206 insertions(+), 6 deletions(-) create mode 100644 contrib/seekable_format/examples/parallel_processing.c diff --git a/contrib/seekable_format/examples/.gitignore b/contrib/seekable_format/examples/.gitignore index 4ded45619..1e1661d80 100644 --- a/contrib/seekable_format/examples/.gitignore +++ b/contrib/seekable_format/examples/.gitignore @@ -1,2 +1,3 @@ seekable_compression seekable_decompression +parallel_processing diff --git a/contrib/seekable_format/examples/Makefile b/contrib/seekable_format/examples/Makefile index fcd1d9146..a77143889 100644 --- a/contrib/seekable_format/examples/Makefile +++ b/contrib/seekable_format/examples/Makefile @@ -21,7 +21,7 @@ SEEKABLE_OBJS = ../zstdseek_compress.c ../zstdseek_decompress.c default: all -all: seekable_compression seekable_decompression +all: seekable_compression seekable_decompression parallel_processing seekable_compression : seekable_compression.c $(SEEKABLE_OBJS) $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ @@ -29,7 +29,10 @@ seekable_compression : seekable_compression.c $(SEEKABLE_OBJS) seekable_decompression : seekable_decompression.c $(SEEKABLE_OBJS) $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ +parallel_processing : parallel_processing.c $(SEEKABLE_OBJS) + $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ -pthread + clean: @rm -f core *.o tmp* result* *.zst \ - seekable_compression seekable_decompression + seekable_compression seekable_decompression parallel_processing @echo Cleaning completed diff --git a/contrib/seekable_format/examples/parallel_processing.c b/contrib/seekable_format/examples/parallel_processing.c new file mode 100644 index 000000000..cf2d0d2af --- /dev/null +++ b/contrib/seekable_format/examples/parallel_processing.c @@ -0,0 +1,195 @@ +/** + * Copyright 2017-present, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the license found in the + * LICENSE-examples file in the root directory of this source tree. + */ + +/* + * A simple demo that sums up all the bytes in the file in parallel using + * seekable decompression and the zstd thread pool + */ + +#include // malloc, exit +#include // fprintf, perror, feof +#include // strerror +#include // errno +#define ZSTD_STATIC_LINKING_ONLY +#include // presumes zstd library is installed +#include +#if defined(WIN32) || defined(_WIN32) +# include +# define SLEEP(x) Sleep(x) +#else +# include +# define SLEEP(x) usleep(x * 1000) +#endif + +#include "pool.h" // use zstd thread pool for demo + +#include "zstd_seekable.h" + +#define MIN(a, b) ((a) < (b) ? (a) : (b)) + +static void* malloc_orDie(size_t size) +{ + void* const buff = malloc(size); + if (buff) return buff; + /* error */ + perror("malloc"); + exit(1); +} + +static void* realloc_orDie(void* ptr, size_t size) +{ + ptr = realloc(ptr, size); + if (ptr) return ptr; + /* error */ + perror("realloc"); + exit(1); +} + +static FILE* fopen_orDie(const char *filename, const char *instruction) +{ + FILE* const inFile = fopen(filename, instruction); + if (inFile) return inFile; + /* error */ + perror(filename); + exit(3); +} + +static size_t fread_orDie(void* buffer, size_t sizeToRead, FILE* file) +{ + size_t const readSize = fread(buffer, 1, sizeToRead, file); + if (readSize == sizeToRead) return readSize; /* good */ + if (feof(file)) return readSize; /* good, reached end of file */ + /* error */ + perror("fread"); + exit(4); +} + +static size_t fwrite_orDie(const void* buffer, size_t sizeToWrite, FILE* file) +{ + size_t const writtenSize = fwrite(buffer, 1, sizeToWrite, file); + if (writtenSize == sizeToWrite) return sizeToWrite; /* good */ + /* error */ + perror("fwrite"); + exit(5); +} + +static size_t fclose_orDie(FILE* file) +{ + if (!fclose(file)) return 0; + /* error */ + perror("fclose"); + exit(6); +} + +static void fseek_orDie(FILE* file, long int offset, int origin) { + if (!fseek(file, offset, origin)) { + if (!fflush(file)) return; + } + /* error */ + perror("fseek"); + exit(7); +} + +static const char* filename; + +struct sum_job { + const char* fname; + unsigned long long sum; + unsigned frameNb; + int done; +}; + +static void sumFrame(void* opaque) +{ + struct sum_job* job = (struct sum_job*)opaque; + job->done = 0; + + FILE* const fin = fopen_orDie(job->fname, "rb"); + + ZSTD_seekable* const seekable = ZSTD_seekable_create(); + if (seekable==NULL) { fprintf(stderr, "ZSTD_seekable_create() error \n"); exit(10); } + + size_t const initResult = ZSTD_seekable_initFile(seekable, fin); + if (ZSTD_isError(initResult)) { fprintf(stderr, "ZSTD_seekable_init() error : %s \n", ZSTD_getErrorName(initResult)); exit(11); } + + size_t const frameSize = ZSTD_seekable_getFrameDecompressedSize(seekable, job->frameNb); + unsigned char* data = malloc_orDie(frameSize); + + size_t result = ZSTD_seekable_decompressFrame(seekable, data, frameSize, job->frameNb); + if (ZSTD_isError(result)) { fprintf(stderr, "ZSTD_seekable_decompressFrame() error : %s \n", ZSTD_getErrorName(result)); exit(12); } + + unsigned long long sum = 0; + size_t i; + for (i = 0; i < frameSize; i++) { + sum += data[i]; + } + job->sum = sum; + job->done = 1; + + fclose(fin); + ZSTD_seekable_free(seekable); + free(data); +} + +static void sumFile_orDie(const char* fname, int nbThreads) +{ + POOL_ctx* pool = POOL_create(nbThreads, nbThreads); + if (pool == NULL) { fprintf(stderr, "POOL_create() error \n"); exit(9); } + + FILE* const fin = fopen_orDie(fname, "rb"); + + ZSTD_seekable* const seekable = ZSTD_seekable_create(); + if (seekable==NULL) { fprintf(stderr, "ZSTD_seekable_create() error \n"); exit(10); } + + size_t const initResult = ZSTD_seekable_initFile(seekable, fin); + if (ZSTD_isError(initResult)) { fprintf(stderr, "ZSTD_seekable_init() error : %s \n", ZSTD_getErrorName(initResult)); exit(11); } + + size_t const numFrames = ZSTD_seekable_getNumFrames(seekable); + struct sum_job* jobs = (struct sum_job*)malloc(numFrames * sizeof(struct sum_job)); + + size_t i; + for (i = 0; i < numFrames; i++) { + jobs[i] = (struct sum_job){ fname, 0, i, 0 }; + POOL_add(pool, sumFrame, &jobs[i]); + } + + unsigned long long total = 0; + + for (i = 0; i < numFrames; i++) { + while (!jobs[i].done) SLEEP(5); /* wake up every 5 milliseconds to check */ + total += jobs[i].sum; + } + + printf("Sum: %llu\n", total); + + POOL_free(pool); + ZSTD_seekable_free(seekable); + fclose(fin); + free(jobs); +} + + +int main(int argc, const char** argv) +{ + const char* const exeName = argv[0]; + + if (argc!=3) { + fprintf(stderr, "wrong arguments\n"); + fprintf(stderr, "usage:\n"); + fprintf(stderr, "%s FILE NB_THREADS\n", exeName); + return 1; + } + + { + const char* const inFilename = argv[1]; + int const nbThreads = atoi(argv[2]); + sumFile_orDie(inFilename, nbThreads); + } + + return 0; +} diff --git a/contrib/seekable_format/examples/seekable_decompression.c b/contrib/seekable_format/examples/seekable_decompression.c index d18def7cd..b765a7591 100644 --- a/contrib/seekable_format/examples/seekable_decompression.c +++ b/contrib/seekable_format/examples/seekable_decompression.c @@ -122,7 +122,7 @@ int main(int argc, const char** argv) if (argc!=4) { fprintf(stderr, "wrong arguments\n"); fprintf(stderr, "usage:\n"); - fprintf(stderr, "%s FILE\n", exeName); + fprintf(stderr, "%s FILE START END\n", exeName); return 1; } diff --git a/contrib/seekable_format/zstdseek_decompress.c b/contrib/seekable_format/zstdseek_decompress.c index a9e87577b..4a8b4e568 100644 --- a/contrib/seekable_format/zstdseek_decompress.c +++ b/contrib/seekable_format/zstdseek_decompress.c @@ -406,7 +406,8 @@ size_t ZSTD_seekable_decompress(ZSTD_seekable* zs, void* dst, size_t len, U64 of } if (zs->seekTable.checksumFlag) { - XXH64_update(&zs->xxhState, outTmp.dst, outTmp.pos); + XXH64_update(&zs->xxhState, (BYTE*)outTmp.dst + prevOutPos, + outTmp.pos - prevOutPos); } zs->decompressedOffset += outTmp.pos - prevOutPos; @@ -454,7 +455,7 @@ size_t ZSTD_seekable_decompressFrame(ZSTD_seekable* zs, void* dst, size_t dstSiz return ERROR(dstSize_tooSmall); } return ZSTD_seekable_decompress( - zs, dst, zs->seekTable.entries[frameIndex].dOffset, - decompressedSize); + zs, dst, decompressedSize, + zs->seekTable.entries[frameIndex].dOffset); } } From 4de86329576ab5b5f4c7e190ea573243c11866b5 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Mon, 24 Apr 2017 16:48:25 -0700 Subject: [PATCH 234/305] Add LZ4 compress/decompress support to CLI --- programs/Makefile | 19 +++++- programs/fileio.c | 154 ++++++++++++++++++++++++++++++++++++++++++++- programs/fileio.h | 3 +- programs/zstdcli.c | 6 ++ tests/playTests.sh | 41 +++++++++++- 5 files changed, 216 insertions(+), 7 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index 4ab4dfd62..b3763484e 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -116,6 +116,16 @@ LZMALD = -llzma else LZMA_MSG := $(NO_LZMA_MSG) endif +# lz4 detection +NO_LZ4_MSG := ==> no liblz4, building zstd without .lz4 support +HAVE_LZ4 := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_lz4$(EXT) -x c - -llz4 2> $(VOID) && rm have_lz4$(EXT) && echo 1 || echo 0) +ifeq ($(HAVE_LZ4), 1) +LZ4_MSG := ==> building zstd with .lz4 compression support +LZ4CPP = -DZSTD_LZ4COMPRESS -DZSTD_LZ4DECOMPRESS +LZ4LD = -llz4 +else +LZ4_MSG := $(NO_LZ4_MSG) +endif MD2ROFF =ronn MD2ROFF_FLAGS = --roff --warnings --manual="User Commands" --organization="zstd $(ZSTD_VERSION)" @@ -135,11 +145,16 @@ zstd-nogz : ZLIB_MSG := $(NO_ZLIB_MSG) zstd-nogz : LZMA_MSG := $(NO_LZMA_MSG) xzstd : CPPFLAGS += $(ZLIBCPP) $(LZMACPP) xzstd : LDFLAGS += $(ZLIBLD) $(LZMALD) -zstd zstd-nogz xzstd : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) -zstd zstd-nogz xzstd : $(ZSTDLIB_FILES) zstdcli.o fileio.o bench.o datagen.o dibio.o +xzstd : LZ4_MSG := $(NO_LZMA_MSG) +zstd4 : CPPFLAGS += $(ZLIBCPP) $(LZ4CPP) +zstd4 : LDFLAGS += $(ZLIBLD) $(LZ4LD) +zstd4 : LZMA_MSG := $(NO_LZMA_MSG) +zstd zstd-nogz xzstd zstd4 : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) +zstd zstd-nogz xzstd zstd4 : $(ZSTDLIB_FILES) zstdcli.o fileio.o bench.o datagen.o dibio.o @echo "$(THREAD_MSG)" @echo "$(ZLIB_MSG)" @echo "$(LZMA_MSG)" + @echo "$(LZ4_MSG)" ifneq (,$(filter Windows%,$(OS))) windres/generate_res.bat endif diff --git a/programs/fileio.c b/programs/fileio.c index 9bca20664..1b54256ec 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -53,6 +53,11 @@ # include #endif +#define LZ4_MAGICNUMBER 0x184D2204 +#if defined(ZSTD_LZ4COMPRESS) || defined(ZSTD_LZ4DECOMPRESS) +# include +#endif + /*-************************************* * Constants @@ -514,6 +519,77 @@ static unsigned long long FIO_compressLzmaFrame(cRess_t* ress, const char* srcFi } #endif +#ifdef ZSTD_LZ4COMPRESS +static int FIO_LZ4_GetBlockSize_FromBlockId (int id) { return (1 << (8 + (2 * id))); } +static unsigned long long FIO_compressLz4Frame(cRess_t* ress, const char* srcFileName, U64 const srcFileSize, int compressionLevel, U64* readsize) +{ + unsigned long long inFileSize = 0, outFileSize = 0; + + LZ4F_preferences_t prefs; + LZ4F_cctx* ctx; + + LZ4F_errorCode_t const errorCode = LZ4F_createCompressionContext(&ctx, LZ4F_VERSION); + if (LZ4F_isError(errorCode)) EXM_THROW(31, "zstd: failed to create lz4 compression context"); + + memset(&prefs, 0, sizeof(prefs)); + + prefs.autoFlush = 1; + prefs.compressionLevel = compressionLevel; + prefs.frameInfo.blockMode = LZ4F_blockIndependent; /* stick to defaults for lz4 cli */ + prefs.frameInfo.blockSizeID = LZ4F_max4MB; + prefs.frameInfo.contentChecksumFlag = (LZ4F_contentChecksum_t)g_checksumFlag; + prefs.frameInfo.contentSize = srcFileSize; + + { + size_t blockSize = FIO_LZ4_GetBlockSize_FromBlockId(LZ4F_max4MB); + size_t readSize; + size_t headerSize = LZ4F_compressBegin(ctx, ress->dstBuffer, ress->dstBufferSize, &prefs); + if (LZ4F_isError(headerSize)) EXM_THROW(33, "File header generation failed : %s", LZ4F_getErrorName(headerSize)); + { size_t const sizeCheck = fwrite(ress->dstBuffer, 1, headerSize, ress->dstFile); + if (sizeCheck!=headerSize) EXM_THROW(34, "Write error : cannot write header"); } + outFileSize += headerSize; + + /* Read first block */ + readSize = fread(ress->srcBuffer, (size_t)1, (size_t)blockSize, ress->srcFile); + inFileSize += readSize; + + /* Main Loop */ + while (readSize>0) { + size_t outSize; + + /* Compress Block */ + outSize = LZ4F_compressUpdate(ctx, ress->dstBuffer, ress->dstBufferSize, ress->srcBuffer, readSize, NULL); + if (LZ4F_isError(outSize)) EXM_THROW(35, "zstd: %s: lz4 compression failed : %s", srcFileName, LZ4F_getErrorName(outSize)); + outFileSize += outSize; + if (!srcFileSize) DISPLAYUPDATE(2, "\rRead : %u MB ==> %.2f%%", (U32)(inFileSize>>20), (double)outFileSize/inFileSize*100) + else DISPLAYUPDATE(2, "\rRead : %u / %u MB ==> %.2f%%", (U32)(inFileSize>>20), (U32)(srcFileSize>>20), (double)outFileSize/inFileSize*100); + + /* Write Block */ + { size_t const sizeCheck = fwrite(ress->dstBuffer, 1, outSize, ress->dstFile); + if (sizeCheck!=outSize) EXM_THROW(36, "Write error : cannot write compressed block"); } + + /* Read next block */ + readSize = fread(ress->srcBuffer, (size_t)1, (size_t)blockSize, ress->srcFile); + inFileSize += readSize; + } + if (ferror(ress->srcFile)) EXM_THROW(37, "Error reading %s ", srcFileName); + + /* End of Stream mark */ + headerSize = LZ4F_compressEnd(ctx, ress->dstBuffer, ress->dstBufferSize, NULL); + if (LZ4F_isError(headerSize)) EXM_THROW(38, "zstd: %s: lz4 end of file generation failed : %s", srcFileName, LZ4F_getErrorName(headerSize)); + + { size_t const sizeCheck = fwrite(ress->dstBuffer, 1, headerSize, ress->dstFile); + if (sizeCheck!=headerSize) EXM_THROW(39, "Write error : cannot write end of stream"); } + outFileSize += headerSize; + } + + *readsize = inFileSize; + LZ4F_freeCompressionContext(ctx); + + return outFileSize; +} +#endif + /*! FIO_compressFilename_internal() : * same as FIO_compressFilename_extRess(), with `ress.desFile` already opened. @@ -547,6 +623,13 @@ static int FIO_compressFilename_internal(cRess_t ress, #else (void)compressionLevel; EXM_THROW(20, "zstd: %s: file cannot be compressed as xz/lzma (zstd compiled without ZSTD_LZMACOMPRESS) -- ignored \n", srcFileName); +#endif + case FIO_lz4Compression: +#ifdef ZSTD_LZ4COMPRESS + compressedfilesize = FIO_compressLz4Frame(&ress, srcFileName, fileSize, compressionLevel, &readsize); +#else + (void)compressionLevel; + EXM_THROW(20, "zstd: %s: file cannot be compressed as lz4 (zstd compiled without ZSTD_LZ4COMPRESS) -- ignored \n", srcFileName); #endif goto finish; } @@ -1039,6 +1122,66 @@ static unsigned long long FIO_decompressLzmaFrame(dRess_t* ress, FILE* srcFile, } #endif +#ifdef ZSTD_LZ4DECOMPRESS +static unsigned long long FIO_decompressLz4Frame(dRess_t* ress, FILE* srcFile, const char* srcFileName) +{ + unsigned long long filesize = 0; + LZ4F_errorCode_t nextToLoad; + LZ4F_dctx* dCtx; + LZ4F_errorCode_t const errorCode = LZ4F_createDecompressionContext(&dCtx, LZ4F_VERSION); + + if (LZ4F_isError(errorCode)) EXM_THROW(61, "zstd: failed to create lz4 decompression context"); + + /* Init feed with magic number (already consumed from FILE* sFile) */ + { size_t inSize = 4; + size_t outSize= 0; + MEM_writeLE32(ress->srcBuffer, LZ4_MAGICNUMBER); + nextToLoad = LZ4F_decompress(dCtx, ress->dstBuffer, &outSize, ress->srcBuffer, &inSize, NULL); + if (LZ4F_isError(nextToLoad)) EXM_THROW(62, "zstd: %s: lz4 header error : %s", srcFileName, LZ4F_getErrorName(nextToLoad)); + } + + /* Main Loop */ + for (;nextToLoad;) { + size_t readSize; + size_t pos = 0; + size_t decodedBytes = ress->dstBufferSize; + + /* Read input */ + if (nextToLoad > ress->srcBufferSize) nextToLoad = ress->srcBufferSize; + readSize = fread(ress->srcBuffer, 1, nextToLoad, srcFile); + if (!readSize) break; /* reached end of file or stream */ + + while ((pos < readSize) || (decodedBytes == ress->dstBufferSize)) { /* still to read, or still to flush */ + /* Decode Input (at least partially) */ + size_t remaining = readSize - pos; + decodedBytes = ress->dstBufferSize; + nextToLoad = LZ4F_decompress(dCtx, ress->dstBuffer, &decodedBytes, (char*)(ress->srcBuffer)+pos, &remaining, NULL); + if (LZ4F_isError(nextToLoad)) EXM_THROW(66, "zstd: %s: decompression error : %s", srcFileName, LZ4F_getErrorName(nextToLoad)); + pos += remaining; + + /* Write Block */ + if (decodedBytes) { + if (fwrite(ress->dstBuffer, 1, decodedBytes, ress->dstFile) != decodedBytes) EXM_THROW(63, "Write error : cannot write to output file"); + filesize += decodedBytes; + DISPLAYUPDATE(2, "\rDecompressed : %u MB ", (unsigned)(filesize>>20)); + } + + if (!nextToLoad) break; + } + } + /* can be out because readSize == 0, which could be an fread() error */ + if (ferror(srcFile)) EXM_THROW(67, "zstd: %s: read error", srcFileName); + + if (nextToLoad!=0) EXM_THROW(68, "zstd: %s: unfinished stream", srcFileName); + + LZ4F_freeDecompressionContext(dCtx); + ress->srcBufferLoaded = 0; /* LZ4F will go to the frame boundary */ + + return filesize; +} +#endif + + /** FIO_decompressSrcFile() : Decompression `srcFileName` into `ress.dstFile` @@ -1090,6 +1233,15 @@ static int FIO_decompressSrcFile(dRess_t ress, const char* dstFileName, const ch #else DISPLAYLEVEL(1, "zstd: %s: xz/lzma file cannot be uncompressed (zstd compiled without ZSTD_LZMADECOMPRESS) -- ignored \n", srcFileName); return 1; +#endif + } else if (MEM_readLE32(buf) == LZ4_MAGICNUMBER) { +#ifdef ZSTD_LZ4DECOMPRESS + unsigned long long const result = FIO_decompressLz4Frame(&ress, srcFile, srcFileName); + if (result == 0) return 1; + filesize += result; +#else + DISPLAYLEVEL(1, "zstd: %s: lz4 file cannot be uncompressed (zstd compiled without ZSTD_LZ4DECOMPRESS) -- ignored \n", srcFileName); + return 1; #endif } else { if (!ZSTD_isFrame(ress.srcBuffer, toRead)) { @@ -1199,7 +1351,7 @@ int FIO_decompressMultipleFilenames(const char** srcNamesTable, unsigned nbFiles dstFileName = (char*)malloc(dfnSize); if (dstFileName==NULL) EXM_THROW(74, "not enough memory for dstFileName"); } - if (sfnSize <= suffixSize || (strcmp(suffixPtr, GZ_EXTENSION) && strcmp(suffixPtr, XZ_EXTENSION) && strcmp(suffixPtr, ZSTD_EXTENSION) && strcmp(suffixPtr, LZMA_EXTENSION))) { + if (sfnSize <= suffixSize || (strcmp(suffixPtr, GZ_EXTENSION) && strcmp(suffixPtr, XZ_EXTENSION) && strcmp(suffixPtr, ZSTD_EXTENSION) && strcmp(suffixPtr, LZMA_EXTENSION) && strcmp(suffixPtr, LZ4_EXTENSION))) { DISPLAYLEVEL(1, "zstd: %s: unknown suffix (%s/%s/%s/%s expected) -- ignored \n", srcFileName, GZ_EXTENSION, XZ_EXTENSION, ZSTD_EXTENSION, LZMA_EXTENSION); skippedFiles++; continue; diff --git a/programs/fileio.h b/programs/fileio.h index 0dd58d625..65da98d7f 100644 --- a/programs/fileio.h +++ b/programs/fileio.h @@ -33,12 +33,13 @@ extern "C" { #define XZ_EXTENSION ".xz" #define GZ_EXTENSION ".gz" #define ZSTD_EXTENSION ".zst" +#define LZ4_EXTENSION ".lz4" /*-************************************* * Types ***************************************/ -typedef enum { FIO_zstdCompression, FIO_gzipCompression, FIO_xzCompression, FIO_lzmaCompression } FIO_compressionType_t; +typedef enum { FIO_zstdCompression, FIO_gzipCompression, FIO_xzCompression, FIO_lzmaCompression, FIO_lz4Compression } FIO_compressionType_t; /*-************************************* diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 76f55de53..79bc84877 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -136,6 +136,9 @@ static int usage_advanced(const char* programName) DISPLAY( "--format=xz : compress files to the .xz format \n"); DISPLAY( "--format=lzma : compress files to the .lzma format \n"); #endif +#ifdef ZSTD_LZ4COMPRESS + DISPLAY( "--format=lz4 : compress files to the .lz4 format \n"); +#endif #ifndef ZSTD_NODECOMPRESS DISPLAY( "--test : test compressed file integrity \n"); #if ZSTD_SPARSE_DEFAULT @@ -404,6 +407,9 @@ int main(int argCount, const char* argv[]) if (!strcmp(argument, "--format=lzma")) { suffix = LZMA_EXTENSION; FIO_setCompressionType(FIO_lzmaCompression); continue; } if (!strcmp(argument, "--format=xz")) { suffix = XZ_EXTENSION; FIO_setCompressionType(FIO_xzCompression); continue; } #endif +#ifdef ZSTD_LZ4COMPRESS + if (!strcmp(argument, "--format=lz4")) { suffix = LZ4_EXTENSION; FIO_setCompressionType(FIO_lz4Compression); continue; } +#endif /* long commands with arguments */ #ifndef ZSTD_NODICT diff --git a/tests/playTests.sh b/tests/playTests.sh index a91e9e8bb..369506c2e 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -403,7 +403,7 @@ if [ $GZIPMODE -eq 1 ]; then $ZSTD -f --format=gzip tmp $ZSTD -f tmp cat tmp.gz tmp.zst tmp.gz tmp.zst | $ZSTD -d -f -o tmp - head -c -1 tmp.gz | $ZSTD -t && die "incomplete frame not detected !" + head -c -1 tmp.gz | $ZSTD -t > $INTOVOID && die "incomplete frame not detected !" rm tmp* else $ECHO "gzip mode not supported" @@ -445,13 +445,48 @@ if [ $LZMAMODE -eq 1 ]; then $ZSTD -f --format=lzma tmp $ZSTD -f tmp cat tmp.xz tmp.lzma tmp.zst tmp.lzma tmp.xz tmp.zst | $ZSTD -d -f -o tmp - head -c -1 tmp.xz | $ZSTD -t && die "incomplete frame not detected !" - head -c -1 tmp.lzma | $ZSTD -t && die "incomplete frame not detected !" + head -c -1 tmp.xz | $ZSTD -t > $INTOVOID && die "incomplete frame not detected !" + head -c -1 tmp.lzma | $ZSTD -t > $INTOVOID && die "incomplete frame not detected !" rm tmp* else $ECHO "xz mode not supported" fi +$ECHO "\n**** lz4 compatibility tests **** " + +LZ4MODE=1 +$ZSTD --format=lz4 -V || LZ4MODE=0 +if [ $LZ4MODE -eq 1 ]; then + $ECHO "lz4 support detected" + LZ4EXE=1 + lz4 -V || LZ4EXE=0 + if [ $LZ4EXE -eq 1 ]; then + ./datagen > tmp + $ZSTD --format=lz4 -f tmp + lz4 -t -v tmp.lz4 + lz4 -f tmp + $ZSTD -d -f -v tmp.lz4 + rm tmp* + else + $ECHO "lz4 binary not detected" + fi +else + $ECHO "lz4 mode not supported" +fi + + +$ECHO "\n**** lz4 frame tests **** " + +if [ $LZ4MODE -eq 1 ]; then + ./datagen > tmp + $ZSTD -f --format=lz4 tmp + $ZSTD -f tmp + cat tmp.lz4 tmp.zst tmp.lz4 tmp.zst | $ZSTD -d -f -o tmp + head -c -1 tmp.lz4 | $ZSTD -t > $INTOVOID && die "incomplete frame not detected !" + rm tmp* +else + $ECHO "lz4 mode not supported" +fi $ECHO "\n**** zstd round-trip tests **** " From 2c4b6fe6b3d53eff421841200dead4dc314bf077 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Tue, 25 Apr 2017 11:00:54 -0700 Subject: [PATCH 235/305] Make lz4 compression/decompression compatible with library r123 --- programs/Makefile | 2 +- programs/fileio.c | 15 +++++++++------ 2 files changed, 10 insertions(+), 7 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index b3763484e..2619614c5 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -118,7 +118,7 @@ LZMA_MSG := $(NO_LZMA_MSG) endif # lz4 detection NO_LZ4_MSG := ==> no liblz4, building zstd without .lz4 support -HAVE_LZ4 := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_lz4$(EXT) -x c - -llz4 2> $(VOID) && rm have_lz4$(EXT) && echo 1 || echo 0) +HAVE_LZ4 := $(shell printf '\#include \n\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_lz4$(EXT) -x c - -llz4 2> $(VOID) && rm have_lz4$(EXT) && echo 1 || echo 0) ifeq ($(HAVE_LZ4), 1) LZ4_MSG := ==> building zstd with .lz4 compression support LZ4CPP = -DZSTD_LZ4COMPRESS -DZSTD_LZ4DECOMPRESS diff --git a/programs/fileio.c b/programs/fileio.c index 1b54256ec..2bf5cb209 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -56,6 +56,7 @@ #define LZ4_MAGICNUMBER 0x184D2204 #if defined(ZSTD_LZ4COMPRESS) || defined(ZSTD_LZ4DECOMPRESS) # include +# include #endif @@ -526,7 +527,7 @@ static unsigned long long FIO_compressLz4Frame(cRess_t* ress, const char* srcFil unsigned long long inFileSize = 0, outFileSize = 0; LZ4F_preferences_t prefs; - LZ4F_cctx* ctx; + LZ4F_compressionContext_t ctx; LZ4F_errorCode_t const errorCode = LZ4F_createCompressionContext(&ctx, LZ4F_VERSION); if (LZ4F_isError(errorCode)) EXM_THROW(31, "zstd: failed to create lz4 compression context"); @@ -535,13 +536,15 @@ static unsigned long long FIO_compressLz4Frame(cRess_t* ress, const char* srcFil prefs.autoFlush = 1; prefs.compressionLevel = compressionLevel; - prefs.frameInfo.blockMode = LZ4F_blockIndependent; /* stick to defaults for lz4 cli */ - prefs.frameInfo.blockSizeID = LZ4F_max4MB; - prefs.frameInfo.contentChecksumFlag = (LZ4F_contentChecksum_t)g_checksumFlag; + prefs.frameInfo.blockMode = blockIndependent; /* stick to defaults for lz4 cli */ + prefs.frameInfo.blockSizeID = max4MB; + prefs.frameInfo.contentChecksumFlag = (contentChecksum_t)g_checksumFlag; +#if LZ4_VERSION_NUMBER >= 10600 prefs.frameInfo.contentSize = srcFileSize; +#endif { - size_t blockSize = FIO_LZ4_GetBlockSize_FromBlockId(LZ4F_max4MB); + size_t blockSize = FIO_LZ4_GetBlockSize_FromBlockId(max4MB); size_t readSize; size_t headerSize = LZ4F_compressBegin(ctx, ress->dstBuffer, ress->dstBufferSize, &prefs); if (LZ4F_isError(headerSize)) EXM_THROW(33, "File header generation failed : %s", LZ4F_getErrorName(headerSize)); @@ -1127,7 +1130,7 @@ static unsigned long long FIO_decompressLz4Frame(dRess_t* ress, FILE* srcFile, c { unsigned long long filesize = 0; LZ4F_errorCode_t nextToLoad; - LZ4F_dctx* dCtx; + LZ4F_decompressionContext_t dCtx; LZ4F_errorCode_t const errorCode = LZ4F_createDecompressionContext(&dCtx, LZ4F_VERSION); if (LZ4F_isError(errorCode)) EXM_THROW(61, "zstd: failed to create lz4 decompression context"); From 65e2cda7aaa4e1504ee8d9f9b01460348c8d1e3b Mon Sep 17 00:00:00 2001 From: niXman Date: Wed, 26 Apr 2017 13:04:04 +0300 Subject: [PATCH 236/305] unchecked argv[idx] access --- examples/simple_compression.c | 3 ++- examples/streaming_compression.c | 3 ++- examples/streaming_decompression.c | 3 ++- 3 files changed, 6 insertions(+), 3 deletions(-) diff --git a/examples/simple_compression.c b/examples/simple_compression.c index 9d448712e..ab1131475 100644 --- a/examples/simple_compression.c +++ b/examples/simple_compression.c @@ -116,7 +116,6 @@ static char* createOutFilename_orDie(const char* filename) int main(int argc, const char** argv) { const char* const exeName = argv[0]; - const char* const inFilename = argv[1]; if (argc!=2) { printf("wrong arguments\n"); @@ -125,6 +124,8 @@ int main(int argc, const char** argv) return 1; } + const char* const inFilename = argv[1]; + char* const outFilename = createOutFilename_orDie(inFilename); compress_orDie(inFilename, outFilename); free(outFilename); diff --git a/examples/streaming_compression.c b/examples/streaming_compression.c index 4c2c1a1d8..24ad15bd6 100644 --- a/examples/streaming_compression.c +++ b/examples/streaming_compression.c @@ -112,7 +112,6 @@ static const char* createOutFilename_orDie(const char* filename) int main(int argc, const char** argv) { const char* const exeName = argv[0]; - const char* const inFilename = argv[1]; if (argc!=2) { printf("wrong arguments\n"); @@ -121,6 +120,8 @@ int main(int argc, const char** argv) return 1; } + const char* const inFilename = argv[1]; + const char* const outFilename = createOutFilename_orDie(inFilename); compressFile_orDie(inFilename, outFilename, 1); diff --git a/examples/streaming_decompression.c b/examples/streaming_decompression.c index 400aa673d..bb2d80987 100644 --- a/examples/streaming_decompression.c +++ b/examples/streaming_decompression.c @@ -99,7 +99,6 @@ static void decompressFile_orDie(const char* fname) int main(int argc, const char** argv) { const char* const exeName = argv[0]; - const char* const inFilename = argv[1]; if (argc!=2) { fprintf(stderr, "wrong arguments\n"); @@ -108,6 +107,8 @@ int main(int argc, const char** argv) return 1; } + const char* const inFilename = argv[1]; + decompressFile_orDie(inFilename); return 0; } From eab41c1872fbaf99a949bc082c3169a754d7d147 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Wed, 26 Apr 2017 10:17:38 -0700 Subject: [PATCH 237/305] Fix LZ4 wrapper deprecation warnings --- programs/fileio.c | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/programs/fileio.c b/programs/fileio.c index 2bf5cb209..70d35ad67 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -534,17 +534,22 @@ static unsigned long long FIO_compressLz4Frame(cRess_t* ress, const char* srcFil memset(&prefs, 0, sizeof(prefs)); +#if LZ4_VERSION_NUMBER <= 10600 +#define LZ4F_blockIndependent blockIndependent +#define LZ4F_max4MB max4MB +#endif + prefs.autoFlush = 1; prefs.compressionLevel = compressionLevel; - prefs.frameInfo.blockMode = blockIndependent; /* stick to defaults for lz4 cli */ - prefs.frameInfo.blockSizeID = max4MB; + prefs.frameInfo.blockMode = LZ4F_blockIndependent; /* stick to defaults for lz4 cli */ + prefs.frameInfo.blockSizeID = LZ4F_max4MB; prefs.frameInfo.contentChecksumFlag = (contentChecksum_t)g_checksumFlag; #if LZ4_VERSION_NUMBER >= 10600 prefs.frameInfo.contentSize = srcFileSize; #endif { - size_t blockSize = FIO_LZ4_GetBlockSize_FromBlockId(max4MB); + size_t blockSize = FIO_LZ4_GetBlockSize_FromBlockId(LZ4F_max4MB); size_t readSize; size_t headerSize = LZ4F_compressBegin(ctx, ress->dstBuffer, ress->dstBufferSize, &prefs); if (LZ4F_isError(headerSize)) EXM_THROW(33, "File header generation failed : %s", LZ4F_getErrorName(headerSize)); From e42afbc6fa8fee6e93e7216a146a861b6af4684c Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 26 Apr 2017 11:39:35 -0700 Subject: [PATCH 238/305] Comply with suggested comments by @terrelln created FSE_CTABLE_SIZE() and FSE_DTABLE_SIZE() --- doc/zstd_manual.html | 82 ++++++++++++++++++------------------ lib/common/fse.h | 4 ++ lib/common/zstd_internal.h | 1 - lib/compress/zstd_compress.c | 66 ++++++++++++++--------------- 4 files changed, 78 insertions(+), 75 deletions(-) diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index d267cfc3c..73e200b4e 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -57,46 +57,46 @@
size_t ZSTD_compress( void* dst, size_t dstCapacity,
                 const void* src, size_t srcSize,
                       int compressionLevel);
-

Compresses `src` content as a single zstd compressed frame into already allocated `dst`. - Hint : compression runs faster if `dstCapacity` >= `ZSTD_compressBound(srcSize)`. - @return : compressed size written into `dst` (<= `dstCapacity), - or an error code if it fails (which can be tested using ZSTD_isError()). +

Compresses `src` content as a single zstd compressed frame into already allocated `dst`. + Hint : compression runs faster if `dstCapacity` >= `ZSTD_compressBound(srcSize)`. + @return : compressed size written into `dst` (<= `dstCapacity), + or an error code if it fails (which can be tested using ZSTD_isError()).


size_t ZSTD_decompress( void* dst, size_t dstCapacity,
                   const void* src, size_t compressedSize);
-

`compressedSize` : must be the _exact_ size of some number of compressed and/or skippable frames. - `dstCapacity` is an upper bound of originalSize. - If user cannot imply a maximum upper bound, it's better to use streaming mode to decompress data. - @return : the number of bytes decompressed into `dst` (<= `dstCapacity`), - or an errorCode if it fails (which can be tested using ZSTD_isError()). +

`compressedSize` : must be the _exact_ size of some number of compressed and/or skippable frames. + `dstCapacity` is an upper bound of originalSize. + If user cannot imply a maximum upper bound, it's better to use streaming mode to decompress data. + @return : the number of bytes decompressed into `dst` (<= `dstCapacity`), + or an errorCode if it fails (which can be tested using ZSTD_isError()).


unsigned long long ZSTD_getDecompressedSize(const void* src, size_t srcSize);
-

NOTE: This function is planned to be obsolete, in favour of ZSTD_getFrameContentSize. - ZSTD_getFrameContentSize functions the same way, returning the decompressed size of a single - frame, but distinguishes empty frames from frames with an unknown size, or errors. +

NOTE: This function is planned to be obsolete, in favour of ZSTD_getFrameContentSize. + ZSTD_getFrameContentSize functions the same way, returning the decompressed size of a single + frame, but distinguishes empty frames from frames with an unknown size, or errors. - Additionally, ZSTD_findDecompressedSize can be used instead. It can handle multiple - concatenated frames in one buffer, and so is more general. - As a result however, it requires more computation and entire frames to be passed to it, - as opposed to ZSTD_getFrameContentSize which requires only a single frame's header. + Additionally, ZSTD_findDecompressedSize can be used instead. It can handle multiple + concatenated frames in one buffer, and so is more general. + As a result however, it requires more computation and entire frames to be passed to it, + as opposed to ZSTD_getFrameContentSize which requires only a single frame's header. - 'src' is the start of a zstd compressed frame. - @return : content size to be decompressed, as a 64-bits value _if known_, 0 otherwise. - note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode. - When `return==0`, data to decompress could be any size. - In which case, it's necessary to use streaming mode to decompress data. - Optionally, application can still use ZSTD_decompress() while relying on implied limits. - (For example, data may be necessarily cut into blocks <= 16 KB). - note 2 : decompressed size is always present when compression is done with ZSTD_compress() - note 3 : decompressed size can be very large (64-bits value), - potentially larger than what local system can handle as a single memory segment. - In which case, it's necessary to use streaming mode to decompress data. - note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified. - Always ensure result fits within application's authorized limits. - Each application can set its own limits. - note 5 : when `return==0`, if precise failure cause is needed, use ZSTD_getFrameParams() to know more. + 'src' is the start of a zstd compressed frame. + @return : content size to be decompressed, as a 64-bits value _if known_, 0 otherwise. + note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode. + When `return==0`, data to decompress could be any size. + In which case, it's necessary to use streaming mode to decompress data. + Optionally, application can still use ZSTD_decompress() while relying on implied limits. + (For example, data may be necessarily cut into blocks <= 16 KB). + note 2 : decompressed size is always present when compression is done with ZSTD_compress() + note 3 : decompressed size can be very large (64-bits value), + potentially larger than what local system can handle as a single memory segment. + In which case, it's necessary to use streaming mode to decompress data. + note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified. + Always ensure result fits within application's authorized limits. + Each application can set its own limits. + note 5 : when `return==0`, if precise failure cause is needed, use ZSTD_getFrameParams() to know more.


Helper functions

int         ZSTD_maxCLevel(void);               /*!< maximum compression level available */
@@ -106,28 +106,28 @@ const char* ZSTD_getErrorName(size_t code);     /*!< provides readable strin
 

Explicit memory management


 
-

Compression context

   When compressing many times,
-   it is recommended to allocate a context just once, and re-use it for each successive compression operation.
-   This will make workload friendlier for system's memory.
-   Use one context per thread for parallel execution in multi-threaded environments. 
+

Compression context

  When compressing many times,
+  it is recommended to allocate a context just once, and re-use it for each successive compression operation.
+  This will make workload friendlier for system's memory.
+  Use one context per thread for parallel execution in multi-threaded environments. 
 
typedef struct ZSTD_CCtx_s ZSTD_CCtx;
 ZSTD_CCtx* ZSTD_createCCtx(void);
 size_t     ZSTD_freeCCtx(ZSTD_CCtx* cctx);
 

size_t ZSTD_compressCCtx(ZSTD_CCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize, int compressionLevel);
-

Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()). +

Same as ZSTD_compress(), requires an allocated ZSTD_CCtx (see ZSTD_createCCtx()).


-

Decompression context

   When decompressing many times,
-   it is recommended to allocate a context just once, and re-use it for each successive compression operation.
-   This will make workload friendlier for system's memory.
-   Use one context per thread for parallel execution in multi-threaded environments. 
+

Decompression context

  When decompressing many times,
+  it is recommended to allocate a context just once, and re-use it for each successive compression operation.
+  This will make workload friendlier for system's memory.
+  Use one context per thread for parallel execution in multi-threaded environments. 
 
typedef struct ZSTD_DCtx_s ZSTD_DCtx;
 ZSTD_DCtx* ZSTD_createDCtx(void);
 size_t     ZSTD_freeDCtx(ZSTD_DCtx* dctx);
 

size_t ZSTD_decompressDCtx(ZSTD_DCtx* ctx, void* dst, size_t dstCapacity, const void* src, size_t srcSize);
-

Same as ZSTD_decompress(), requires an allocated ZSTD_DCtx (see ZSTD_createDCtx()). +

Same as ZSTD_decompress(), requires an allocated ZSTD_DCtx (see ZSTD_createDCtx()).


Simple dictionary API


diff --git a/lib/common/fse.h b/lib/common/fse.h
index 4ecd8d741..6d5d41def 100644
--- a/lib/common/fse.h
+++ b/lib/common/fse.h
@@ -316,6 +316,10 @@ If there is an error, the function will return an error code, which can be teste
 #define FSE_CTABLE_SIZE_U32(maxTableLog, maxSymbolValue)   (1 + (1<<(maxTableLog-1)) + ((maxSymbolValue+1)*2))
 #define FSE_DTABLE_SIZE_U32(maxTableLog)                   (1 + (1<litlengthCTable = (FSE_CTable*) ptr;
                 ptr = (char*)ptr + litlengthCTable_size;
                 assert(((size_t)ptr & 3) == 0);   /* ensure correct alignment */
-                zc->tmpCounters = (unsigned*) ptr;
+                zc->entropyScratchSpace = (unsigned*) ptr;
         }   }
 
         /* init params */
@@ -347,8 +347,8 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc,
         assert((char*)zc->offcodeCTable == (char*)zc->hufCTable + hufCTable_size);
         assert((char*)zc->matchlengthCTable == (char*)zc->offcodeCTable + offcodeCTable_size);
         assert((char*)zc->litlengthCTable == (char*)zc->matchlengthCTable + matchlengthCTable_size);
-        assert((char*)zc->tmpCounters == (char*)zc->litlengthCTable + litlengthCTable_size);
-        ptr = (char*)zc->tmpCounters + entropyScratchSpace_size;
+        assert((char*)zc->entropyScratchSpace == (char*)zc->litlengthCTable + litlengthCTable_size);
+        ptr = (char*)zc->entropyScratchSpace + entropyScratchSpace_size;
 
         /* opt parser space */
         if ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) {
@@ -577,9 +577,9 @@ static size_t ZSTD_compressLiterals (ZSTD_CCtx* zc,
         int const preferRepeat = zc->params.cParams.strategy < ZSTD_lazy ? srcSize <= 1024 : 0;
         if (repeat == HUF_repeat_valid && lhSize == 3) singleStream = 1;
         cLitSize = singleStream ? HUF_compress1X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11,
-                                      zc->tmpCounters, entropyScratchSpace_size, zc->hufCTable, &repeat, preferRepeat)
+                                      zc->entropyScratchSpace, entropyScratchSpace_size, zc->hufCTable, &repeat, preferRepeat)
                                 : HUF_compress4X_repeat(ostart+lhSize, dstCapacity-lhSize, src, srcSize, 255, 11,
-                                      zc->tmpCounters, entropyScratchSpace_size, zc->hufCTable, &repeat, preferRepeat);
+                                      zc->entropyScratchSpace, entropyScratchSpace_size, zc->hufCTable, &repeat, preferRepeat);
         if (repeat != HUF_repeat_none) { hType = set_repeat; }    /* reused the existing table */
         else { zc->hufCTable_repeatMode = HUF_repeat_check; }       /* now have a table to reuse */
     }
@@ -708,7 +708,7 @@ MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc,
 
     /* CTable for Literal Lengths */
     {   U32 max = MaxLL;
-        size_t const mostFrequent = FSE_countFast_wksp(count, &max, llCodeTable, nbSeq, zc->tmpCounters);
+        size_t const mostFrequent = FSE_countFast_wksp(count, &max, llCodeTable, nbSeq, zc->entropyScratchSpace);
         if ((mostFrequent == nbSeq) && (nbSeq > 2)) {
             *op++ = llCodeTable[0];
             FSE_buildCTable_rle(CTable_LitLength, (BYTE)max);
@@ -732,7 +732,7 @@ MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc,
 
     /* CTable for Offsets */
     {   U32 max = MaxOff;
-        size_t const mostFrequent = FSE_countFast_wksp(count, &max, ofCodeTable, nbSeq, zc->tmpCounters);
+        size_t const mostFrequent = FSE_countFast_wksp(count, &max, ofCodeTable, nbSeq, zc->entropyScratchSpace);
         if ((mostFrequent == nbSeq) && (nbSeq > 2)) {
             *op++ = ofCodeTable[0];
             FSE_buildCTable_rle(CTable_OffsetBits, (BYTE)max);
@@ -756,7 +756,7 @@ MEM_STATIC size_t ZSTD_compressSequences (ZSTD_CCtx* zc,
 
     /* CTable for MatchLengths */
     {   U32 max = MaxML;
-        size_t const mostFrequent = FSE_countFast_wksp(count, &max, mlCodeTable, nbSeq, zc->tmpCounters);
+        size_t const mostFrequent = FSE_countFast_wksp(count, &max, mlCodeTable, nbSeq, zc->entropyScratchSpace);
         if ((mostFrequent == nbSeq) && (nbSeq > 2)) {
             *op++ = *mlCodeTable;
             FSE_buildCTable_rle(CTable_MatchLength, (BYTE)max);
@@ -1564,7 +1564,7 @@ static void ZSTD_compressBlock_doubleFast_extDict_generic(ZSTD_CCtx* ctx,
                 if ( (((U32)((dictLimit-1) - repIndex2) >= 3) & (repIndex2 > lowestIndex))  /* intentional overflow */
                    && (MEM_read32(repMatch2) == MEM_read32(ip)) ) {
                     const BYTE* const repEnd2 = repIndex2 < dictLimit ? dictEnd : iend;
-                    size_t const repLength2 = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch2+EQUAL_READ32, iend, repEnd2, lowPrefixPtr) + EQUAL_READ32;
+                    size_t const repLength2 = ZSTD_count_2segments(ip+4, repMatch2+4, iend, repEnd2, lowPrefixPtr) + 4;
                     U32 tmpOffset = offset_2; offset_2 = offset_1; offset_1 = tmpOffset;   /* swap offset_2 <=> offset_1 */
                     ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, repLength2-MINMATCH);
                     hashSmall[ZSTD_hashPtr(ip, hBitsS, mls)] = current2;
@@ -1923,7 +1923,7 @@ size_t ZSTD_HcFindBestMatch_generic (
     const U32 current = (U32)(ip-base);
     const U32 minChain = current > chainSize ? current - chainSize : 0;
     int nbAttempts=maxNbAttempts;
-    size_t ml=EQUAL_READ32-1;
+    size_t ml=4-1;
 
     /* HC4 match finder */
     U32 matchIndex = ZSTD_insertAndFindFirstIndex (zc, ip, mls);
@@ -1938,7 +1938,7 @@ size_t ZSTD_HcFindBestMatch_generic (
         } else {
             match = dictBase + matchIndex;
             if (MEM_read32(match) == MEM_read32(ip))   /* assumption : matchIndex <= dictLimit-4 (by table construction) */
-                currentMl = ZSTD_count_2segments(ip+EQUAL_READ32, match+EQUAL_READ32, iLimit, dictEnd, prefixStart) + EQUAL_READ32;
+                currentMl = ZSTD_count_2segments(ip+4, match+4, iLimit, dictEnd, prefixStart) + 4;
         }
 
         /* save best solution */
@@ -2032,7 +2032,7 @@ void ZSTD_compressBlock_lazy_generic(ZSTD_CCtx* ctx,
         /* check repCode */
         if ((offset_1>0) & (MEM_read32(ip+1) == MEM_read32(ip+1 - offset_1))) {
             /* repcode : we take it */
-            matchLength = ZSTD_count(ip+1+EQUAL_READ32, ip+1+EQUAL_READ32-offset_1, iend) + EQUAL_READ32;
+            matchLength = ZSTD_count(ip+1+4, ip+1+4-offset_1, iend) + 4;
             if (depth==0) goto _storeSequence;
         }
 
@@ -2043,7 +2043,7 @@ void ZSTD_compressBlock_lazy_generic(ZSTD_CCtx* ctx,
                 matchLength = ml2, start = ip, offset=offsetFound;
         }
 
-        if (matchLength < EQUAL_READ32) {
+        if (matchLength < 4) {
             ip += ((ip-anchor) >> g_searchStrength) + 1;   /* jump faster over incompressible sections */
             continue;
         }
@@ -2053,17 +2053,17 @@ void ZSTD_compressBlock_lazy_generic(ZSTD_CCtx* ctx,
         while (ip0) & (MEM_read32(ip) == MEM_read32(ip - offset_1)))) {
-                size_t const mlRep = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_1, iend) + EQUAL_READ32;
+                size_t const mlRep = ZSTD_count(ip+4, ip+4-offset_1, iend) + 4;
                 int const gain2 = (int)(mlRep * 3);
                 int const gain1 = (int)(matchLength*3 - ZSTD_highbit32((U32)offset+1) + 1);
-                if ((mlRep >= EQUAL_READ32) && (gain2 > gain1))
+                if ((mlRep >= 4) && (gain2 > gain1))
                     matchLength = mlRep, offset = 0, start = ip;
             }
             {   size_t offset2=99999999;
                 size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls);
                 int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1));   /* raw approx */
                 int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 4);
-                if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) {
+                if ((ml2 >= 4) && (gain2 > gain1)) {
                     matchLength = ml2, offset = offset2, start = ip;
                     continue;   /* search a better one */
             }   }
@@ -2072,17 +2072,17 @@ void ZSTD_compressBlock_lazy_generic(ZSTD_CCtx* ctx,
             if ((depth==2) && (ip0) & (MEM_read32(ip) == MEM_read32(ip - offset_1)))) {
-                    size_t const ml2 = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_1, iend) + EQUAL_READ32;
+                    size_t const ml2 = ZSTD_count(ip+4, ip+4-offset_1, iend) + 4;
                     int const gain2 = (int)(ml2 * 4);
                     int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 1);
-                    if ((ml2 >= EQUAL_READ32) && (gain2 > gain1))
+                    if ((ml2 >= 4) && (gain2 > gain1))
                         matchLength = ml2, offset = 0, start = ip;
                 }
                 {   size_t offset2=99999999;
                     size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls);
                     int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1));   /* raw approx */
                     int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 7);
-                    if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) {
+                    if ((ml2 >= 4) && (gain2 > gain1)) {
                         matchLength = ml2, offset = offset2, start = ip;
                         continue;
             }   }   }
@@ -2110,7 +2110,7 @@ _storeSequence:
              && ((offset_2>0)
              & (MEM_read32(ip) == MEM_read32(ip - offset_2)) )) {
             /* store sequence */
-            matchLength = ZSTD_count(ip+EQUAL_READ32, ip+EQUAL_READ32-offset_2, iend) + EQUAL_READ32;
+            matchLength = ZSTD_count(ip+4, ip+4-offset_2, iend) + 4;
             offset = offset_2; offset_2 = offset_1; offset_1 = (U32)offset; /* swap repcodes */
             ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, matchLength-MINMATCH);
             ip += matchLength;
@@ -2199,7 +2199,7 @@ void ZSTD_compressBlock_lazy_extDict_generic(ZSTD_CCtx* ctx,
             if (MEM_read32(ip+1) == MEM_read32(repMatch)) {
                 /* repcode detected we should take it */
                 const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend;
-                matchLength = ZSTD_count_2segments(ip+1+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32;
+                matchLength = ZSTD_count_2segments(ip+1+4, repMatch+4, iend, repEnd, prefixStart) + 4;
                 if (depth==0) goto _storeSequence;
         }   }
 
@@ -2210,7 +2210,7 @@ void ZSTD_compressBlock_lazy_extDict_generic(ZSTD_CCtx* ctx,
                 matchLength = ml2, start = ip, offset=offsetFound;
         }
 
-         if (matchLength < EQUAL_READ32) {
+         if (matchLength < 4) {
             ip += ((ip-anchor) >> g_searchStrength) + 1;   /* jump faster over incompressible sections */
             continue;
         }
@@ -2229,10 +2229,10 @@ void ZSTD_compressBlock_lazy_extDict_generic(ZSTD_CCtx* ctx,
                 if (MEM_read32(ip) == MEM_read32(repMatch)) {
                     /* repcode detected */
                     const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend;
-                    size_t const repLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32;
+                    size_t const repLength = ZSTD_count_2segments(ip+4, repMatch+4, iend, repEnd, prefixStart) + 4;
                     int const gain2 = (int)(repLength * 3);
                     int const gain1 = (int)(matchLength*3 - ZSTD_highbit32((U32)offset+1) + 1);
-                    if ((repLength >= EQUAL_READ32) && (gain2 > gain1))
+                    if ((repLength >= 4) && (gain2 > gain1))
                         matchLength = repLength, offset = 0, start = ip;
             }   }
 
@@ -2241,7 +2241,7 @@ void ZSTD_compressBlock_lazy_extDict_generic(ZSTD_CCtx* ctx,
                 size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls);
                 int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1));   /* raw approx */
                 int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 4);
-                if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) {
+                if ((ml2 >= 4) && (gain2 > gain1)) {
                     matchLength = ml2, offset = offset2, start = ip;
                     continue;   /* search a better one */
             }   }
@@ -2259,10 +2259,10 @@ void ZSTD_compressBlock_lazy_extDict_generic(ZSTD_CCtx* ctx,
                     if (MEM_read32(ip) == MEM_read32(repMatch)) {
                         /* repcode detected */
                         const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend;
-                        size_t const repLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32;
+                        size_t const repLength = ZSTD_count_2segments(ip+4, repMatch+4, iend, repEnd, prefixStart) + 4;
                         int const gain2 = (int)(repLength * 4);
                         int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 1);
-                        if ((repLength >= EQUAL_READ32) && (gain2 > gain1))
+                        if ((repLength >= 4) && (gain2 > gain1))
                             matchLength = repLength, offset = 0, start = ip;
                 }   }
 
@@ -2271,7 +2271,7 @@ void ZSTD_compressBlock_lazy_extDict_generic(ZSTD_CCtx* ctx,
                     size_t const ml2 = searchMax(ctx, ip, iend, &offset2, maxSearches, mls);
                     int const gain2 = (int)(ml2*4 - ZSTD_highbit32((U32)offset2+1));   /* raw approx */
                     int const gain1 = (int)(matchLength*4 - ZSTD_highbit32((U32)offset+1) + 7);
-                    if ((ml2 >= EQUAL_READ32) && (gain2 > gain1)) {
+                    if ((ml2 >= 4) && (gain2 > gain1)) {
                         matchLength = ml2, offset = offset2, start = ip;
                         continue;
             }   }   }
@@ -2303,7 +2303,7 @@ _storeSequence:
             if (MEM_read32(ip) == MEM_read32(repMatch)) {
                 /* repcode detected we should take it */
                 const BYTE* const repEnd = repIndex < dictLimit ? dictEnd : iend;
-                matchLength = ZSTD_count_2segments(ip+EQUAL_READ32, repMatch+EQUAL_READ32, iend, repEnd, prefixStart) + EQUAL_READ32;
+                matchLength = ZSTD_count_2segments(ip+4, repMatch+4, iend, repEnd, prefixStart) + 4;
                 offset = offset_2; offset_2 = offset_1; offset_1 = (U32)offset;   /* swap offset history */
                 ZSTD_storeSeq(seqStorePtr, 0, anchor, 0, matchLength-MINMATCH);
                 ip += matchLength;

From 5087c1761f1dbe4158a4333e6a4b437aacb6d34b Mon Sep 17 00:00:00 2001
From: Nick Terrell 
Date: Wed, 26 Apr 2017 13:06:38 -0700
Subject: [PATCH 239/305] Rename ZSTD_create*() to ZSTD_init*()

---
 contrib/linux-kernel/include/linux/zstd.h  | 85 +++++++++++-----------
 contrib/linux-kernel/lib/zstd/compress.c   | 18 ++---
 contrib/linux-kernel/lib/zstd/decompress.c | 20 ++---
 contrib/linux-kernel/test/UserlandTest.cpp | 46 ++++++------
 4 files changed, 85 insertions(+), 84 deletions(-)

diff --git a/contrib/linux-kernel/include/linux/zstd.h b/contrib/linux-kernel/include/linux/zstd.h
index 38b2bca69..ee7bd8207 100644
--- a/contrib/linux-kernel/include/linux/zstd.h
+++ b/contrib/linux-kernel/include/linux/zstd.h
@@ -202,7 +202,7 @@ ZSTD_parameters ZSTD_getParams(int compressionLevel,
  **************************************/
 
 /**
- * ZSTD_CCtxWorkspaceBound() - the amount of memory needed to create a ZSTD_CCtx
+ * ZSTD_CCtxWorkspaceBound() - amount of memory needed to initialize a ZSTD_CCtx
  * @cParams: The compression parameters to be used for compression.
  *
  * If multiple compression parameters might be used, the caller must call
@@ -210,7 +210,7 @@ ZSTD_parameters ZSTD_getParams(int compressionLevel,
  * size.
  *
  * Return:   A lower bound on the size of the workspace that is passed to
- *           ZSTD_createCCtx().
+ *           ZSTD_initCCtx().
  */
 size_t ZSTD_CCtxWorkspaceBound(ZSTD_compressionParameters cParams);
 
@@ -222,7 +222,7 @@ size_t ZSTD_CCtxWorkspaceBound(ZSTD_compressionParameters cParams);
  */
 typedef struct ZSTD_CCtx_s ZSTD_CCtx;
 /**
- * ZSTD_createCCtx() - create a zstd compression context
+ * ZSTD_initCCtx() - initialize a zstd compression context
  * @workspace:     The workspace to emplace the context into. It must outlive
  *                 the returned context.
  * @workspaceSize: The size of workspace. Use ZSTD_CCtxWorkspaceBound() to
@@ -230,12 +230,12 @@ typedef struct ZSTD_CCtx_s ZSTD_CCtx;
  *
  * Return:         A compression context emplaced into workspace.
  */
-ZSTD_CCtx *ZSTD_createCCtx(void *workspace, size_t workspaceSize);
+ZSTD_CCtx *ZSTD_initCCtx(void *workspace, size_t workspaceSize);
 
 /**
  * ZSTD_compressCCtx() - compress src into dst
- * @ctx:         The context. Must have been created with a workspace at least
- *               as large as ZSTD_CCtxWorkspaceBound(params.cParams).
+ * @ctx:         The context. Must have been initialized with a workspace at
+ *               least as large as ZSTD_CCtxWorkspaceBound(params.cParams).
  * @dst:         The buffer to compress src into.
  * @dstCapacity: The size of the destination buffer. May be any size, but
  *               ZSTD_compressBound(srcSize) is guaranteed to be large enough.
@@ -250,10 +250,10 @@ size_t ZSTD_compressCCtx(ZSTD_CCtx *ctx, void *dst, size_t dstCapacity,
 	const void *src, size_t srcSize, ZSTD_parameters params);
 
 /**
- * ZSTD_DCtxWorkspaceBound() - the amount of memory needed to create a ZSTD_DCtx
+ * ZSTD_DCtxWorkspaceBound() - amount of memory needed to initialize a ZSTD_DCtx
  *
  * Return: A lower bound on the size of the workspace that is passed to
- *         ZSTD_createDCtx().
+ *         ZSTD_initDCtx().
  */
 size_t ZSTD_DCtxWorkspaceBound(void);
 
@@ -265,7 +265,7 @@ size_t ZSTD_DCtxWorkspaceBound(void);
  */
 typedef struct ZSTD_DCtx_s ZSTD_DCtx;
 /**
- * ZSTD_createDCtx() - create a zstd decompression context
+ * ZSTD_initDCtx() - initialize a zstd decompression context
  * @workspace:     The workspace to emplace the context into. It must outlive
  *                 the returned context.
  * @workspaceSize: The size of workspace. Use ZSTD_DCtxWorkspaceBound() to
@@ -273,7 +273,7 @@ typedef struct ZSTD_DCtx_s ZSTD_DCtx;
  *
  * Return:         A decompression context emplaced into workspace.
  */
-ZSTD_DCtx *ZSTD_createDCtx(void *workspace, size_t workspaceSize);
+ZSTD_DCtx *ZSTD_initDCtx(void *workspace, size_t workspaceSize);
 
 /**
  * ZSTD_decompressDCtx() - decompress zstd compressed src into dst
@@ -298,8 +298,8 @@ size_t ZSTD_decompressDCtx(ZSTD_DCtx *ctx, void *dst, size_t dstCapacity,
 
 /**
  * ZSTD_compress_usingDict() - compress src into dst using a dictionary
- * @ctx:         The context. Must have been created with a workspace at least
- *               as large as ZSTD_CCtxWorkspaceBound(params.cParams).
+ * @ctx:         The context. Must have been initialized with a workspace at
+ *               least as large as ZSTD_CCtxWorkspaceBound(params.cParams).
  * @dst:         The buffer to compress src into.
  * @dstCapacity: The size of the destination buffer. May be any size, but
  *               ZSTD_compressBound(srcSize) is guaranteed to be large enough.
@@ -344,11 +344,11 @@ size_t ZSTD_decompress_usingDict(ZSTD_DCtx *ctx, void *dst, size_t dstCapacity,
  ***************************/
 
 /**
- * ZSTD_CDictWorkspaceBound() - amount of memory needed to create a ZSTD_CDict
+ * ZSTD_CDictWorkspaceBound() - memory needed to initialize a ZSTD_CDict
  * @cParams: The compression parameters to be used for compression.
  *
  * Return:   A lower bound on the size of the workspace that is passed to
- *           ZSTD_createCDict().
+ *           ZSTD_initCDict().
  */
 size_t ZSTD_CDictWorkspaceBound(ZSTD_compressionParameters cParams);
 
@@ -358,7 +358,7 @@ size_t ZSTD_CDictWorkspaceBound(ZSTD_compressionParameters cParams);
 typedef struct ZSTD_CDict_s ZSTD_CDict;
 
 /**
- * ZSTD_createCDict() - create a digested dictionary for compression
+ * ZSTD_initCDict() - initialize a digested dictionary for compression
  * @dictBuffer:    The dictionary to digest. The buffer is referenced by the
  *                 ZSTD_CDict so it must outlive the returned ZSTD_CDict.
  * @dictSize:      The size of the dictionary.
@@ -373,14 +373,15 @@ typedef struct ZSTD_CDict_s ZSTD_CDict;
  *
  * Return:         The digested dictionary emplaced into workspace.
  */
-ZSTD_CDict *ZSTD_createCDict(const void *dictBuffer, size_t dictSize,
+ZSTD_CDict *ZSTD_initCDict(const void *dictBuffer, size_t dictSize,
 	ZSTD_parameters params, void *workspace, size_t workspaceSize);
 
 /**
  * ZSTD_compress_usingCDict() - compress src into dst using a ZSTD_CDict
- * @ctx:         The context. Must have been created with a workspace at least
- *               as large as ZSTD_CCtxWorkspaceBound(cParams) where cParams are
- *               the compression parameters used to create cdict.
+ * @ctx:         The context. Must have been initialized with a workspace at
+ *               least as large as ZSTD_CCtxWorkspaceBound(cParams) where
+ *               cParams are the compression parameters used to initialize the
+ *               cdict.
  * @dst:         The buffer to compress src into.
  * @dstCapacity: The size of the destination buffer. May be any size, but
  *               ZSTD_compressBound(srcSize) is guaranteed to be large enough.
@@ -400,10 +401,10 @@ size_t ZSTD_compress_usingCDict(ZSTD_CCtx *cctx, void *dst, size_t dstCapacity,
 
 
 /**
- * ZSTD_DDictWorkspaceBound() - amount of memory needed to create a ZSTD_DDict
+ * ZSTD_DDictWorkspaceBound() - memory needed to initialize a ZSTD_DDict
  *
  * Return:  A lower bound on the size of the workspace that is passed to
- *          ZSTD_createDDict().
+ *          ZSTD_initDDict().
  */
 size_t ZSTD_DDictWorkspaceBound(void);
 
@@ -413,7 +414,7 @@ size_t ZSTD_DDictWorkspaceBound(void);
 typedef struct ZSTD_DDict_s ZSTD_DDict;
 
 /**
- * ZSTD_createDDict() - create a digested dictionary for decompression
+ * ZSTD_initDDict() - initialize a digested dictionary for decompression
  * @dictBuffer:    The dictionary to digest. The buffer is referenced by the
  *                 ZSTD_DDict so it must outlive the returned ZSTD_DDict.
  * @dictSize:      The size of the dictionary.
@@ -427,7 +428,7 @@ typedef struct ZSTD_DDict_s ZSTD_DDict;
  *
  * Return:         The digested dictionary emplaced into workspace.
  */
-ZSTD_DDict *ZSTD_createDDict(const void *dictBuffer, size_t dictSize,
+ZSTD_DDict *ZSTD_initDDict(const void *dictBuffer, size_t dictSize,
 	void *workspace, size_t workspaceSize);
 
 /**
@@ -487,7 +488,7 @@ typedef struct ZSTD_outBuffer_s {
  * Streaming compression - HowTo
  *
  * A ZSTD_CStream object is required to track streaming operation.
- * Use ZSTD_createCStream() to create and initialize a ZSTD_CStream object.
+ * Use ZSTD_initCStream() to initialize a ZSTD_CStream object.
  * ZSTD_CStream objects can be reused multiple times on consecutive compression
  * operations. It is recommended to re-use ZSTD_CStream in situations where many
  * streaming operations will be achieved consecutively. Use one separate
@@ -515,11 +516,11 @@ typedef struct ZSTD_outBuffer_s {
  ******************************************************************************/
 
 /**
- * ZSTD_CStreamWorkspaceBound() - memory needed to create a ZSTD_CStream
+ * ZSTD_CStreamWorkspaceBound() - memory needed to initialize a ZSTD_CStream
  * @cParams: The compression parameters to be used for compression.
  *
  * Return:   A lower bound on the size of the workspace that is passed to
- *           ZSTD_createCStream() and ZSTD_createCStream_usingCDict().
+ *           ZSTD_initCStream() and ZSTD_initCStream_usingCDict().
  */
 size_t ZSTD_CStreamWorkspaceBound(ZSTD_compressionParameters cParams);
 
@@ -530,7 +531,7 @@ typedef struct ZSTD_CStream_s ZSTD_CStream;
 
 /*===== ZSTD_CStream management functions =====*/
 /**
- * ZSTD_createCStream() - create a zstd streaming compression context
+ * ZSTD_initCStream() - initialize a zstd streaming compression context
  * @params:         The zstd compression parameters.
  * @pledgedSrcSize: If params.fParams.contentSizeFlag == 1 then the caller must
  *                  pass the source size (zero means empty source). Otherwise,
@@ -544,23 +545,23 @@ typedef struct ZSTD_CStream_s ZSTD_CStream;
  *
  * Return:          The zstd streaming compression context.
  */
-ZSTD_CStream *ZSTD_createCStream(ZSTD_parameters params,
+ZSTD_CStream *ZSTD_initCStream(ZSTD_parameters params,
 	unsigned long long pledgedSrcSize, void *workspace,
 	size_t workspaceSize);
 
 /**
- * ZSTD_createCStream_usingCDict() - create a zstd streaming compression context
+ * ZSTD_initCStream_usingCDict() - initialize a streaming compression context
  * @cdict:          The digested dictionary to use for compression.
  * @pledgedSrcSize: Optionally the source size, or zero if unknown.
  * @workspace:      The workspace to emplace the context into. It must outlive
  *                  the returned context.
  * @workspaceSize:  The size of workspace. Call ZSTD_CStreamWorkspaceBound()
- *                  with the cParams used to create the cdict to determine how
- *                  large the workspace must be.
+ *                  with the cParams used to initialize the cdict to determine
+ *                  how large the workspace must be.
  *
  * Return:          The zstd streaming compression context.
  */
-ZSTD_CStream *ZSTD_createCStream_usingCDict(const ZSTD_CDict *cdict,
+ZSTD_CStream *ZSTD_initCStream_usingCDict(const ZSTD_CDict *cdict,
 	unsigned long long pledgedSrcSize, void *workspace,
 	size_t workspaceSize);
 
@@ -646,7 +647,7 @@ size_t ZSTD_CStreamOutSize(void);
  * Streaming decompression - HowTo
  *
  * A ZSTD_DStream object is required to track streaming operations.
- * Use ZSTD_createDStream() to initialize a ZSTD_DStream object.
+ * Use ZSTD_initDStream() to initialize a ZSTD_DStream object.
  * ZSTD_DStream objects can be re-used multiple times.
  *
  * Use ZSTD_decompressStream() repetitively to consume your input.
@@ -660,11 +661,11 @@ size_t ZSTD_CStreamOutSize(void);
  ******************************************************************************/
 
 /**
- * ZSTD_DStreamWorkspaceBound() - memory needed to create a ZSTD_DStream
+ * ZSTD_DStreamWorkspaceBound() - memory needed to initialize a ZSTD_DStream
  * @maxWindowSize: The maximum window size allowed for compressed frames.
  *
  * Return:         A lower bound on the size of the workspace that is passed to
- *                 ZSTD_createDStream() and ZSTD_createDStream_usingDDict().
+ *                 ZSTD_initDStream() and ZSTD_initDStream_usingDDict().
  */
 size_t ZSTD_DStreamWorkspaceBound(size_t maxWindowSize);
 
@@ -674,7 +675,7 @@ size_t ZSTD_DStreamWorkspaceBound(size_t maxWindowSize);
 typedef struct ZSTD_DStream_s ZSTD_DStream;
 /*===== ZSTD_DStream management functions =====*/
 /**
- * ZSTD_createDStream() - create a zstd streaming decompression context
+ * ZSTD_initDStream() - initialize a zstd streaming decompression context
  * @maxWindowSize: The maximum window size allowed for compressed frames.
  * @workspace:     The workspace to emplace the context into. It must outlive
  *                 the returned context.
@@ -684,10 +685,10 @@ typedef struct ZSTD_DStream_s ZSTD_DStream;
  *
  * Return:         The zstd streaming decompression context.
  */
-ZSTD_DStream *ZSTD_createDStream(size_t maxWindowSize, void *workspace,
+ZSTD_DStream *ZSTD_initDStream(size_t maxWindowSize, void *workspace,
 	size_t workspaceSize);
 /**
- * ZSTD_createDStream_usingDDict() - create zstd streaming decompression context
+ * ZSTD_initDStream_usingDDict() - initialize streaming decompression context
  * @maxWindowSize: The maximum window size allowed for compressed frames.
  * @ddict:         The digested dictionary to use for decompression.
  * @workspace:     The workspace to emplace the context into. It must outlive
@@ -698,7 +699,7 @@ ZSTD_DStream *ZSTD_createDStream(size_t maxWindowSize, void *workspace,
  *
  * Return:         The zstd streaming decompression context.
  */
-ZSTD_DStream *ZSTD_createDStream_usingDDict(size_t maxWindowSize,
+ZSTD_DStream *ZSTD_initDStream_usingDDict(size_t maxWindowSize,
 	const ZSTD_DDict *ddict, void *workspace, size_t workspaceSize);
 
 /*===== Streaming decompression functions =====*/
@@ -953,7 +954,7 @@ size_t ZSTD_getFrameParams(ZSTD_frameParams *fparamsPtr, const void *src,
  * Buffer-less streaming compression (synchronous mode)
  *
  * A ZSTD_CCtx object is required to track streaming operations.
- * Use ZSTD_createCCtx() to create a context.
+ * Use ZSTD_initCCtx() to initialize a context.
  * ZSTD_CCtx object can be re-used multiple times within successive compression
  * operations.
  *
@@ -1015,7 +1016,7 @@ size_t ZSTD_compressEnd(ZSTD_CCtx *cctx, void *dst, size_t dstCapacity,
  * Buffer-less streaming decompression (synchronous mode)
  *
  * A ZSTD_DCtx object is required to track streaming operations.
- * Use ZSTD_createDCtx() to create a context.
+ * Use ZSTD_initDCtx() to initialize a context.
  * A ZSTD_DCtx object can be re-used multiple times.
  *
  * First typical operation is to retrieve frame parameters, using
@@ -1116,7 +1117,7 @@ ZSTD_nextInputType_e ZSTD_nextInputType(ZSTD_DCtx *dctx);
  *
  * A few rules to respect:
  * - Compressing and decompressing require a context structure
- *   + Use ZSTD_createCCtx() and ZSTD_createDCtx()
+ *   + Use ZSTD_initCCtx() and ZSTD_initDCtx()
  * - It is necessary to init context before starting
  *   + compression : ZSTD_compressBegin()
  *   + decompression : ZSTD_decompressBegin()
diff --git a/contrib/linux-kernel/lib/zstd/compress.c b/contrib/linux-kernel/lib/zstd/compress.c
index 00d18069a..5f6d955a4 100644
--- a/contrib/linux-kernel/lib/zstd/compress.c
+++ b/contrib/linux-kernel/lib/zstd/compress.c
@@ -116,7 +116,7 @@ static ZSTD_CCtx* ZSTD_createCCtx_advanced(ZSTD_customMem customMem)
 	return cctx;
 }
 
-ZSTD_CCtx* ZSTD_createCCtx(void* workspace, size_t workspaceSize)
+ZSTD_CCtx* ZSTD_initCCtx(void* workspace, size_t workspaceSize)
 {
 	ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize);
 	ZSTD_CCtx* cctx = ZSTD_createCCtx_advanced(stackMem);
@@ -2819,7 +2819,7 @@ static ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dict
 	}
 }
 
-ZSTD_CDict* ZSTD_createCDict(const void* dict, size_t dictSize, ZSTD_parameters params, void* workspace, size_t workspaceSize)
+ZSTD_CDict* ZSTD_initCDict(const void* dict, size_t dictSize, ZSTD_parameters params, void* workspace, size_t workspaceSize)
 {
 	ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize);
 	return ZSTD_createCDict_advanced(dict, dictSize, 1, params, stackMem);
@@ -3010,7 +3010,7 @@ static size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs,
 	return ZSTD_resetCStream_internal(zcs, pledgedSrcSize);
 }
 
-ZSTD_CStream* ZSTD_createCStream(ZSTD_parameters params, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize)
+ZSTD_CStream* ZSTD_initCStream(ZSTD_parameters params, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize)
 {
 	ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize);
 	ZSTD_CStream* const zcs = ZSTD_createCStream_advanced(stackMem);
@@ -3021,10 +3021,10 @@ ZSTD_CStream* ZSTD_createCStream(ZSTD_parameters params, unsigned long long pled
 	return zcs;
 }
 
-ZSTD_CStream* ZSTD_createCStream_usingCDict(const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize)
+ZSTD_CStream* ZSTD_initCStream_usingCDict(const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, void* workspace, size_t workspaceSize)
 {
 	ZSTD_parameters const params = ZSTD_getParamsFromCDict(cdict);
-	ZSTD_CStream* const zcs = ZSTD_createCStream(params, pledgedSrcSize, workspace, workspaceSize);
+	ZSTD_CStream* const zcs = ZSTD_initCStream(params, pledgedSrcSize, workspace, workspaceSize);
 	if (zcs) {
 		zcs->cdict = cdict;
 		if (ZSTD_isError(ZSTD_resetCStream_internal(zcs, pledgedSrcSize))) {
@@ -3346,17 +3346,17 @@ EXPORT_SYMBOL(ZSTD_maxCLevel);
 EXPORT_SYMBOL(ZSTD_compressBound);
 
 EXPORT_SYMBOL(ZSTD_CCtxWorkspaceBound);
-EXPORT_SYMBOL(ZSTD_createCCtx);
+EXPORT_SYMBOL(ZSTD_initCCtx);
 EXPORT_SYMBOL(ZSTD_compressCCtx);
 EXPORT_SYMBOL(ZSTD_compress_usingDict);
 
 EXPORT_SYMBOL(ZSTD_CDictWorkspaceBound);
-EXPORT_SYMBOL(ZSTD_createCDict);
+EXPORT_SYMBOL(ZSTD_initCDict);
 EXPORT_SYMBOL(ZSTD_compress_usingCDict);
 
 EXPORT_SYMBOL(ZSTD_CStreamWorkspaceBound);
-EXPORT_SYMBOL(ZSTD_createCStream);
-EXPORT_SYMBOL(ZSTD_createCStream_usingCDict);
+EXPORT_SYMBOL(ZSTD_initCStream);
+EXPORT_SYMBOL(ZSTD_initCStream_usingCDict);
 EXPORT_SYMBOL(ZSTD_resetCStream);
 EXPORT_SYMBOL(ZSTD_compressStream);
 EXPORT_SYMBOL(ZSTD_flushStream);
diff --git a/contrib/linux-kernel/lib/zstd/decompress.c b/contrib/linux-kernel/lib/zstd/decompress.c
index 3fcef0105..94f5fd560 100644
--- a/contrib/linux-kernel/lib/zstd/decompress.c
+++ b/contrib/linux-kernel/lib/zstd/decompress.c
@@ -130,7 +130,7 @@ ZSTD_DCtx* ZSTD_createDCtx_advanced(ZSTD_customMem customMem)
 	return dctx;
 }
 
-ZSTD_DCtx* ZSTD_createDCtx(void* workspace, size_t workspaceSize)
+ZSTD_DCtx* ZSTD_initDCtx(void* workspace, size_t workspaceSize)
 {
 	ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize);
 	return ZSTD_createDCtx_advanced(stackMem);
@@ -1968,11 +1968,11 @@ static ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize,
 	}
 }
 
-/*! ZSTD_createDDict() :
+/*! ZSTD_initDDict() :
 *   Create a digested dictionary, to start decompression without startup delay.
 *   `dict` content is copied inside DDict.
 *   Consequently, `dict` can be released after `ZSTD_DDict` creation */
-ZSTD_DDict* ZSTD_createDDict(const void* dict, size_t dictSize, void* workspace, size_t workspaceSize)
+ZSTD_DDict* ZSTD_initDDict(const void* dict, size_t dictSize, void* workspace, size_t workspaceSize)
 {
 	ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize);
 	return ZSTD_createDDict_advanced(dict, dictSize, 1, stackMem);
@@ -2100,7 +2100,7 @@ static ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem)
 	return zds;
 }
 
-ZSTD_DStream* ZSTD_createDStream(size_t maxWindowSize, void* workspace, size_t workspaceSize)
+ZSTD_DStream* ZSTD_initDStream(size_t maxWindowSize, void* workspace, size_t workspaceSize)
 {
 	ZSTD_customMem const stackMem = ZSTD_initStack(workspace, workspaceSize);
 	ZSTD_DStream* zds = ZSTD_createDStream_advanced(stackMem);
@@ -2117,9 +2117,9 @@ ZSTD_DStream* ZSTD_createDStream(size_t maxWindowSize, void* workspace, size_t w
 	return zds;
 }
 
-ZSTD_DStream* ZSTD_createDStream_usingDDict(size_t maxWindowSize, const ZSTD_DDict* ddict, void* workspace, size_t workspaceSize)
+ZSTD_DStream* ZSTD_initDStream_usingDDict(size_t maxWindowSize, const ZSTD_DDict* ddict, void* workspace, size_t workspaceSize)
 {
-	ZSTD_DStream* zds = ZSTD_createDStream(maxWindowSize, workspace, workspaceSize);
+	ZSTD_DStream* zds = ZSTD_initDStream(maxWindowSize, workspace, workspaceSize);
 	if (zds) {
 		zds->ddict = ddict;
 	}
@@ -2337,17 +2337,17 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
 }
 
 EXPORT_SYMBOL(ZSTD_DCtxWorkspaceBound);
-EXPORT_SYMBOL(ZSTD_createDCtx);
+EXPORT_SYMBOL(ZSTD_initDCtx);
 EXPORT_SYMBOL(ZSTD_decompressDCtx);
 EXPORT_SYMBOL(ZSTD_decompress_usingDict);
 
 EXPORT_SYMBOL(ZSTD_DDictWorkspaceBound);
-EXPORT_SYMBOL(ZSTD_createDDict);
+EXPORT_SYMBOL(ZSTD_initDDict);
 EXPORT_SYMBOL(ZSTD_decompress_usingDDict);
 
 EXPORT_SYMBOL(ZSTD_DStreamWorkspaceBound);
-EXPORT_SYMBOL(ZSTD_createDStream);
-EXPORT_SYMBOL(ZSTD_createDStream_usingDDict);
+EXPORT_SYMBOL(ZSTD_initDStream);
+EXPORT_SYMBOL(ZSTD_initDStream_usingDDict);
 EXPORT_SYMBOL(ZSTD_resetDStream);
 EXPORT_SYMBOL(ZSTD_decompressStream);
 EXPORT_SYMBOL(ZSTD_DStreamInSize);
diff --git a/contrib/linux-kernel/test/UserlandTest.cpp b/contrib/linux-kernel/test/UserlandTest.cpp
index 556abcdaf..73b30be4c 100644
--- a/contrib/linux-kernel/test/UserlandTest.cpp
+++ b/contrib/linux-kernel/test/UserlandTest.cpp
@@ -20,7 +20,7 @@ createCCtx(ZSTD_compressionParameters cParams) {
   size_t const workspaceSize = ZSTD_CCtxWorkspaceBound(cParams);
   void *workspace = malloc(workspaceSize);
   std::unique_ptr cctx{
-      ZSTD_createCCtx(workspace, workspaceSize), WorkspaceDeleter{workspace}};
+      ZSTD_initCCtx(workspace, workspaceSize), WorkspaceDeleter{workspace}};
   if (!cctx) {
     throw std::runtime_error{"Bad cctx"};
   }
@@ -39,7 +39,7 @@ createDCtx() {
   size_t const workspaceSize = ZSTD_DCtxWorkspaceBound();
   void *workspace = malloc(workspaceSize);
   std::unique_ptr dctx{
-      ZSTD_createDCtx(workspace, workspaceSize), WorkspaceDeleter{workspace}};
+      ZSTD_initDCtx(workspace, workspaceSize), WorkspaceDeleter{workspace}};
   if (!dctx) {
     throw std::runtime_error{"Bad dctx"};
   }
@@ -51,8 +51,8 @@ createCDict(std::string const& dict, ZSTD_parameters params) {
   size_t const workspaceSize = ZSTD_CDictWorkspaceBound(params.cParams);
   void *workspace = malloc(workspaceSize);
   std::unique_ptr cdict{
-      ZSTD_createCDict(dict.data(), dict.size(), params, workspace,
-                       workspaceSize),
+      ZSTD_initCDict(dict.data(), dict.size(), params, workspace,
+                     workspaceSize),
       WorkspaceDeleter{workspace}};
   if (!cdict) {
     throw std::runtime_error{"Bad cdict"};
@@ -71,7 +71,7 @@ createDDict(std::string const& dict) {
   size_t const workspaceSize = ZSTD_DDictWorkspaceBound();
   void *workspace = malloc(workspaceSize);
   std::unique_ptr ddict{
-      ZSTD_createDDict(dict.data(), dict.size(), workspace, workspaceSize),
+      ZSTD_initDDict(dict.data(), dict.size(), workspace, workspaceSize),
       WorkspaceDeleter{workspace}};
   if (!ddict) {
     throw std::runtime_error{"Bad ddict"};
@@ -84,7 +84,7 @@ createCStream(ZSTD_parameters params, unsigned long long pledgedSrcSize = 0) {
   size_t const workspaceSize = ZSTD_CStreamWorkspaceBound(params.cParams);
   void *workspace = malloc(workspaceSize);
   std::unique_ptr zcs{
-      ZSTD_createCStream(params, pledgedSrcSize, workspace, workspaceSize)};
+      ZSTD_initCStream(params, pledgedSrcSize, workspace, workspaceSize)};
   if (!zcs) {
     throw std::runtime_error{"bad cstream"};
   }
@@ -97,8 +97,8 @@ createCStream(ZSTD_compressionParameters cParams, ZSTD_CDict const &cdict,
   size_t const workspaceSize = ZSTD_CStreamWorkspaceBound(cParams);
   void *workspace = malloc(workspaceSize);
   std::unique_ptr zcs{
-      ZSTD_createCStream_usingCDict(&cdict, pledgedSrcSize, workspace,
-                                    workspaceSize)};
+      ZSTD_initCStream_usingCDict(&cdict, pledgedSrcSize, workspace,
+                                  workspaceSize)};
   if (!zcs) {
     throw std::runtime_error{"bad cstream"};
   }
@@ -118,9 +118,9 @@ createDStream(size_t maxWindowSize = (1ULL << ZSTD_WINDOWLOG_MAX),
   void *workspace = malloc(workspaceSize);
   std::unique_ptr zds{
       ddict == nullptr
-          ? ZSTD_createDStream(maxWindowSize, workspace, workspaceSize)
-          : ZSTD_createDStream_usingDDict(maxWindowSize, ddict, workspace,
-                                          workspaceSize)};
+          ? ZSTD_initDStream(maxWindowSize, workspace, workspaceSize)
+          : ZSTD_initDStream_usingDDict(maxWindowSize, ddict, workspace,
+                                        workspaceSize)};
   if (!zds) {
     throw std::runtime_error{"bad dstream"};
   }
@@ -333,7 +333,7 @@ TEST(Block, PreprocessedZstdDict) {
   EXPECT_EQ(kData, decompressed);
 }
 
-TEST(Block, RecreateCCtx) {
+TEST(Block, ReinitializeCCtx) {
   auto cctx = createCCtx(1);
   {
     auto const compressed = compress(*cctx, kData, 1);
@@ -346,7 +346,7 @@ TEST(Block, RecreateCCtx) {
   auto raw = cctx.release();
   auto params = ZSTD_getParams(1, 0, 0);
   cctx.reset(
-      ZSTD_createCCtx(d.memory, ZSTD_CCtxWorkspaceBound(params.cParams)));
+      ZSTD_initCCtx(d.memory, ZSTD_CCtxWorkspaceBound(params.cParams)));
   // Repeat
   {
     auto const compressed = compress(*cctx, kData, 1);
@@ -356,7 +356,7 @@ TEST(Block, RecreateCCtx) {
   }
 }
 
-TEST(Block, RecreateDCtx) {
+TEST(Block, ReinitializeDCtx) {
   auto dctx = createDCtx();
   {
     auto cctx = createCCtx(1);
@@ -367,7 +367,7 @@ TEST(Block, RecreateDCtx) {
   // Create the cctx with the same memory
   auto d = dctx.get_deleter();
   auto raw = dctx.release();
-  dctx.reset(ZSTD_createDCtx(d.memory, ZSTD_DCtxWorkspaceBound()));
+  dctx.reset(ZSTD_initDCtx(d.memory, ZSTD_DCtxWorkspaceBound()));
   // Repeat
   {
     auto cctx = createCCtx(1);
@@ -486,24 +486,24 @@ TEST(Stream, Flush) {
 
 TEST(API, Symbols) {
   TEST_SYMBOL(ZSTD_CCtxWorkspaceBound);
-  TEST_SYMBOL(ZSTD_createCCtx);
+  TEST_SYMBOL(ZSTD_initCCtx);
   TEST_SYMBOL(ZSTD_compressCCtx);
   TEST_SYMBOL(ZSTD_compress_usingDict);
   TEST_SYMBOL(ZSTD_DCtxWorkspaceBound);
-  TEST_SYMBOL(ZSTD_createDCtx);
+  TEST_SYMBOL(ZSTD_initDCtx);
   TEST_SYMBOL(ZSTD_decompressDCtx);
   TEST_SYMBOL(ZSTD_decompress_usingDict);
 
   TEST_SYMBOL(ZSTD_CDictWorkspaceBound);
-  TEST_SYMBOL(ZSTD_createCDict);
+  TEST_SYMBOL(ZSTD_initCDict);
   TEST_SYMBOL(ZSTD_compress_usingCDict);
   TEST_SYMBOL(ZSTD_DDictWorkspaceBound);
-  TEST_SYMBOL(ZSTD_createDDict);
+  TEST_SYMBOL(ZSTD_initDDict);
   TEST_SYMBOL(ZSTD_decompress_usingDDict);
 
   TEST_SYMBOL(ZSTD_CStreamWorkspaceBound);
-  TEST_SYMBOL(ZSTD_createCStream);
-  TEST_SYMBOL(ZSTD_createCStream_usingCDict);
+  TEST_SYMBOL(ZSTD_initCStream);
+  TEST_SYMBOL(ZSTD_initCStream_usingCDict);
   TEST_SYMBOL(ZSTD_resetCStream);
   TEST_SYMBOL(ZSTD_compressStream);
   TEST_SYMBOL(ZSTD_flushStream);
@@ -511,8 +511,8 @@ TEST(API, Symbols) {
   TEST_SYMBOL(ZSTD_CStreamInSize);
   TEST_SYMBOL(ZSTD_CStreamOutSize);
   TEST_SYMBOL(ZSTD_DStreamWorkspaceBound);
-  TEST_SYMBOL(ZSTD_createDStream);
-  TEST_SYMBOL(ZSTD_createDStream_usingDDict);
+  TEST_SYMBOL(ZSTD_initDStream);
+  TEST_SYMBOL(ZSTD_initDStream_usingDDict);
   TEST_SYMBOL(ZSTD_resetDStream);
   TEST_SYMBOL(ZSTD_decompressStream);
   TEST_SYMBOL(ZSTD_DStreamInSize);

From 768df129d2c81c5c82b444225f62983e2cf3423e Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Wed, 26 Apr 2017 15:42:10 -0700
Subject: [PATCH 240/305] changed ZSTD_compressBegin_usingCDict()

No longer takes `pledgedSrcSize` as argument
this is in line with similar functions ZSTD_compress_usingCDict()
and ZSTD_initCStream_usingCDict().
---
 doc/zstd_manual.html         | 2 +-
 lib/compress/zstd_compress.c | 7 +++----
 lib/zstd.h                   | 2 +-
 3 files changed, 5 insertions(+), 6 deletions(-)

diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html
index 73e200b4e..0ef9b8385 100644
--- a/doc/zstd_manual.html
+++ b/doc/zstd_manual.html
@@ -561,7 +561,7 @@ size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds);
 

Buffer-less streaming compression functions

size_t ZSTD_compressBegin(ZSTD_CCtx* cctx, int compressionLevel);
 size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel);
 size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */
-size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: fail if cdict==NULL. pledgedSrcSize can be 0, indicating unknown size.  For 0 size frames, use compressBegin_advanced */
+size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict); /**< note: fails if cdict==NULL */
 size_t ZSTD_compressBegin_usingCDict_advanced(ZSTD_CCtx* const cctx, const ZSTD_CDict* const cdict, ZSTD_frameParameters const fParams, unsigned long long const pledgedSrcSize);   /* compression parameters are already set within cdict. pledgedSrcSize=0 means null-size */
 size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**<  note: if pledgedSrcSize can be 0, indicating unknown size.  if it is non-zero, it must be accurate.  for 0 size frames, use compressBegin_advanced */
 

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 9a2d194ad..e4d82661e 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -3023,11 +3023,10 @@ size_t ZSTD_compressBegin_usingCDict_advanced( /* ZSTD_compressBegin_usingCDict() : * pledgedSrcSize=0 means "unknown" * if pledgedSrcSize>0, it will enable contentSizeFlag */ -size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize) +size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict) { - ZSTD_frameParameters fParams = { 1 /*content*/, 0 /*checksum*/, 0 /*noDictID*/ }; - fParams.contentSizeFlag = (pledgedSrcSize > 0); - return ZSTD_compressBegin_usingCDict_advanced(cctx, cdict, fParams, pledgedSrcSize); + ZSTD_frameParameters const fParams = { 0 /*content*/, 0 /*checksum*/, 0 /*noDictID*/ }; + return ZSTD_compressBegin_usingCDict_advanced(cctx, cdict, fParams, 0); } /*! ZSTD_compress_usingCDict() : diff --git a/lib/zstd.h b/lib/zstd.h index a99e497fc..158ff66a1 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -659,7 +659,7 @@ ZSTDLIB_API size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds); ZSTDLIB_API size_t ZSTD_compressBegin(ZSTD_CCtx* cctx, int compressionLevel); ZSTDLIB_API size_t ZSTD_compressBegin_usingDict(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, int compressionLevel); ZSTDLIB_API size_t ZSTD_compressBegin_advanced(ZSTD_CCtx* cctx, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */ -ZSTDLIB_API size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize); /**< note: fail if cdict==NULL. pledgedSrcSize can be 0, indicating unknown size. For 0 size frames, use compressBegin_advanced */ +ZSTDLIB_API size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict); /**< note: fails if cdict==NULL */ ZSTDLIB_API size_t ZSTD_compressBegin_usingCDict_advanced(ZSTD_CCtx* const cctx, const ZSTD_CDict* const cdict, ZSTD_frameParameters const fParams, unsigned long long const pledgedSrcSize); /* compression parameters are already set within cdict. pledgedSrcSize=0 means null-size */ ZSTDLIB_API size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned long long pledgedSrcSize); /**< note: if pledgedSrcSize can be 0, indicating unknown size. if it is non-zero, it must be accurate. for 0 size frames, use compressBegin_advanced */ From 31533bacce265904ecc376206ebf52f4cbb2e0b5 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 00:29:04 -0700 Subject: [PATCH 241/305] Changed ZSTD_createCDict_advanced() It now only uses compressionParameters as argument. It produces many changes throughout user code, though hopefully they tend to be simple : just provide the cParams part from existing ZSTD_parameters. Some programs might depend on ZSTD_createCDict_advanced() to pass frame parameters. This change will force them to revisit this strategy and fix it, since frame parameters are effectively silently ignored in current version. --- doc/zstd_manual.html | 2 +- lib/compress/zstd_compress.c | 66 +++++++++++++++++++++---------- lib/compress/zstdmt_compress.c | 2 +- lib/zstd.h | 2 +- programs/bench.c | 2 +- programs/fileio.c | 1 + tests/fuzzer.c | 2 +- tests/zstreamtest.c | 2 +- zlibWrapper/examples/zwrapbench.c | 2 +- 9 files changed, 54 insertions(+), 27 deletions(-) diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index 0ef9b8385..9121b460f 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -400,7 +400,7 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v


ZSTD_CDict* ZSTD_createCDict_advanced(const void* dict, size_t dictSize, unsigned byReference,
-                                      ZSTD_parameters params, ZSTD_customMem customMem);
+                                      ZSTD_compressionParameters cParams, ZSTD_customMem customMem);
 

Create a ZSTD_CDict using external alloc and free, and customized compression parameters


diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index e4d82661e..5daa55542 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -20,6 +20,26 @@ #include "zstd_internal.h" /* includes zstd.h */ +/*-************************************* +* Debug +***************************************/ +#if defined(ZSTD_DEBUG) && (ZSTD_DEBUG>=1) +# include +#else +# define assert(condition) ((void)0) +#endif + +#define ZSTD_STATIC_ASSERT(c) { enum { ZSTD_static_assert = 1/(int)(!!(c)) }; } + +#if defined(ZSTD_DEBUG) && (ZSTD_DEBUG>=2) +# include + static unsigned g_debugLevel = ZSTD_DEBUG; +# define DEBUGLOG(l, ...) if (l<=g_debugLevel) { fprintf(stderr, __FILE__ ": "); fprintf(stderr, __VA_ARGS__); fprintf(stderr, " \n"); } +#else +# define DEBUGLOG(l, ...) {} /* disabled */ +#endif + + /*-************************************* * Constants ***************************************/ @@ -38,14 +58,6 @@ static size_t const entropyScratchSpace_size = HUF_WORKSPACE_SIZE; /*-************************************* * Helper functions ***************************************/ -#if defined(ZSTD_DEBUG) && (ZSTD_DEBUG==1) -# include -#else -# define assert(condition) ((void)0) -#endif - -#define ZSTD_STATIC_ASSERT(c) { enum { ZSTD_static_assert = 1/(int)(!!(c)) }; } - size_t ZSTD_compressBound(size_t srcSize) { size_t const lowLimit = 256 KB; size_t const margin = (srcSize < lowLimit) ? (lowLimit-srcSize) >> 12 : 0; /* from 64 to 0 */ @@ -407,6 +419,7 @@ size_t ZSTD_copyCCtx_internal(ZSTD_CCtx* dstCCtx, const ZSTD_CCtx* srcCCtx, memcpy(&dstCCtx->customMem, &srcCCtx->customMem, sizeof(ZSTD_customMem)); { ZSTD_parameters params = srcCCtx->params; params.fParams = fParams; + DEBUGLOG(5, "ZSTD_resetCCtx_internal : dictIDFlag : %u \n", !fParams.noDictIDFlag); ZSTD_resetCCtx_internal(dstCCtx, params, pledgedSrcSize, ZSTDcrp_noMemset); } @@ -887,7 +900,7 @@ MEM_STATIC void ZSTD_storeSeq(seqStore_t* seqStorePtr, size_t litLength, const v const U32 pos = (U32)((const BYTE*)literals - g_start); if (g_start==NULL) g_start = (const BYTE*)literals; if ((pos > 1895000) && (pos < 1895300)) - fprintf(stderr, "Cpos %6u :%5u literals & match %3u bytes at distance %6u \n", + DEBUGLOG(5, "Cpos %6u :%5u literals & match %3u bytes at distance %6u \n", pos, (U32)litLength, (U32)matchCode+MINMATCH, (U32)offsetCode); } #endif @@ -1049,7 +1062,6 @@ static size_t ZSTD_hashPtr(const void* p, U32 hBits, U32 mls) { switch(mls) { - //case 3: return ZSTD_hash3Ptr(p, hBits); default: case 4: return ZSTD_hash4Ptr(p, hBits); case 5: return ZSTD_hash5Ptr(p, hBits); @@ -2503,7 +2515,8 @@ static size_t ZSTD_compress_generic (ZSTD_CCtx* cctx, static size_t ZSTD_writeFrameHeader(void* dst, size_t dstCapacity, ZSTD_parameters params, U64 pledgedSrcSize, U32 dictID) { BYTE* const op = (BYTE*)dst; - U32 const dictIDSizeCode = (dictID>0) + (dictID>=256) + (dictID>=65536); /* 0-3 */ + U32 const dictIDSizeCodeLength = (dictID>0) + (dictID>=256) + (dictID>=65536); /* 0-3 */ + U32 const dictIDSizeCode = params.fParams.noDictIDFlag ? 0 : dictIDSizeCodeLength; /* 0-3 */ U32 const checksumFlag = params.fParams.checksumFlag>0; U32 const windowSize = 1U << params.cParams.windowLog; U32 const singleSegment = params.fParams.contentSizeFlag && (windowSize >= pledgedSrcSize); @@ -2515,6 +2528,9 @@ static size_t ZSTD_writeFrameHeader(void* dst, size_t dstCapacity, size_t pos; if (dstCapacity < ZSTD_frameHeaderSize_max) return ERROR(dstSize_tooSmall); + DEBUGLOG(5, "ZSTD_writeFrameHeader : dictIDFlag : %u \n", !params.fParams.noDictIDFlag); + DEBUGLOG(5, "ZSTD_writeFrameHeader : dictID : %u \n", dictID); + DEBUGLOG(5, "ZSTD_writeFrameHeader : dictIDSizeCode : %u \n", dictIDSizeCode); MEM_writeLE32(dst, ZSTD_MAGICNUMBER); op[4] = frameHeaderDecriptionByte; pos=5; @@ -2933,7 +2949,7 @@ size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict) } ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, unsigned byReference, - ZSTD_parameters params, ZSTD_customMem customMem) + ZSTD_compressionParameters cParams, ZSTD_customMem customMem) { if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem; if (!customMem.customAlloc || !customMem.customFree) return NULL; @@ -2958,7 +2974,11 @@ ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, u cdict->dictContent = internalBuffer; } - { size_t const errorCode = ZSTD_compressBegin_advanced(cctx, cdict->dictContent, dictSize, params, 0); + { ZSTD_frameParameters const fParams = { 0 /* contentSizeFlag */, 0 /* checksumFlag */, 0 /* noDictIDFlag */ }; /* dummy */ + ZSTD_parameters params; + params.cParams = cParams; + params.fParams = fParams; + size_t const errorCode = ZSTD_compressBegin_advanced(cctx, cdict->dictContent, dictSize, params, 0); if (ZSTD_isError(errorCode)) { ZSTD_free(cdict->dictBuffer, customMem); ZSTD_free(cdict, customMem); @@ -2975,17 +2995,15 @@ ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, u ZSTD_CDict* ZSTD_createCDict(const void* dict, size_t dictSize, int compressionLevel) { ZSTD_customMem const allocator = { NULL, NULL, NULL }; - ZSTD_parameters params = ZSTD_getParams(compressionLevel, 0, dictSize); - params.fParams.contentSizeFlag = 1; - return ZSTD_createCDict_advanced(dict, dictSize, 0, params, allocator); + ZSTD_compressionParameters cParams = ZSTD_getCParams(compressionLevel, 0, dictSize); + return ZSTD_createCDict_advanced(dict, dictSize, 0, cParams, allocator); } ZSTD_CDict* ZSTD_createCDict_byReference(const void* dict, size_t dictSize, int compressionLevel) { ZSTD_customMem const allocator = { NULL, NULL, NULL }; - ZSTD_parameters params = ZSTD_getParams(compressionLevel, 0, dictSize); - params.fParams.contentSizeFlag = 1; - return ZSTD_createCDict_advanced(dict, dictSize, 1, params, allocator); + ZSTD_compressionParameters cParams = ZSTD_getCParams(compressionLevel, 0, dictSize); + return ZSTD_createCDict_advanced(dict, dictSize, 1, cParams, allocator); } size_t ZSTD_freeCDict(ZSTD_CDict* cdict) @@ -3010,6 +3028,7 @@ size_t ZSTD_compressBegin_usingCDict_advanced( ZSTD_frameParameters const fParams, unsigned long long const pledgedSrcSize) { if (cdict==NULL) return ERROR(GENERIC); /* does not support NULL cdict */ + DEBUGLOG(5, "ZSTD_compressBegin_usingCDict_advanced : dictIDFlag == %u \n", !fParams.noDictIDFlag); if (cdict->dictContentSize) CHECK_F( ZSTD_copyCCtx_internal(cctx, cdict->refContext, fParams, pledgedSrcSize) ) else { @@ -3026,6 +3045,7 @@ size_t ZSTD_compressBegin_usingCDict_advanced( size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict) { ZSTD_frameParameters const fParams = { 0 /*content*/, 0 /*checksum*/, 0 /*noDictID*/ }; + DEBUGLOG(5, "ZSTD_compressBegin_usingCDict : dictIDFlag == %u \n", !fParams.noDictIDFlag); return ZSTD_compressBegin_usingCDict_advanced(cctx, cdict, fParams, 0); } @@ -3126,6 +3146,8 @@ static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, unsigned long long p { if (zcs->inBuffSize==0) return ERROR(stage_wrong); /* zcs has not been init at least once => can't reset */ + DEBUGLOG(5, "ZSTD_resetCStream_internal : dictIDFlag == %u \n", !zcs->params.fParams.noDictIDFlag); + if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict_advanced(zcs->cctx, zcs->cdict, zcs->params.fParams, pledgedSrcSize)) else CHECK_F(ZSTD_compressBegin_internal(zcs->cctx, NULL, 0, zcs->params, pledgedSrcSize)); @@ -3143,6 +3165,7 @@ size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize) { zcs->params.fParams.contentSizeFlag = (pledgedSrcSize > 0); + DEBUGLOG(5, "ZSTD_resetCStream : dictIDFlag == %u \n", !zcs->params.fParams.noDictIDFlag); return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } @@ -3178,6 +3201,7 @@ static size_t ZSTD_initCStream_stage2(ZSTD_CStream* zcs, zcs->checksum = params.fParams.checksumFlag > 0; zcs->params = params; + DEBUGLOG(5, "ZSTD_initCStream_stage2 : dictIDFlag == %u \n", !params.fParams.noDictIDFlag); return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } @@ -3201,11 +3225,12 @@ static size_t ZSTD_initCStream_internal(ZSTD_CStream* zcs, if (dict && dictSize >= 8) { ZSTD_freeCDict(zcs->cdictLocal); - zcs->cdictLocal = ZSTD_createCDict_advanced(dict, dictSize, 0 /* copy */, params, zcs->customMem); + zcs->cdictLocal = ZSTD_createCDict_advanced(dict, dictSize, 0 /* copy */, params.cParams, zcs->customMem); if (zcs->cdictLocal == NULL) return ERROR(memory_allocation); zcs->cdict = zcs->cdictLocal; } + DEBUGLOG(5, "ZSTD_initCStream_internal : dictIDFlag == %u \n", !params.fParams.noDictIDFlag); return ZSTD_initCStream_stage2(zcs, params, pledgedSrcSize); } @@ -3214,6 +3239,7 @@ size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, ZSTD_parameters params, unsigned long long pledgedSrcSize) { CHECK_F( ZSTD_checkCParams(params.cParams) ); + DEBUGLOG(5, "ZSTD_initCStream_advanced : dictIDFlag == %u \n", !params.fParams.noDictIDFlag); return ZSTD_initCStream_internal(zcs, dict, dictSize, params, pledgedSrcSize); } diff --git a/lib/compress/zstdmt_compress.c b/lib/compress/zstdmt_compress.c index 04a448981..fc7f52a29 100644 --- a/lib/compress/zstdmt_compress.c +++ b/lib/compress/zstdmt_compress.c @@ -521,7 +521,7 @@ static size_t ZSTDMT_initCStream_internal(ZSTDMT_CCtx* zcs, if (updateDict) { ZSTD_freeCDict(zcs->cdict); zcs->cdict = NULL; if (dict && dictSize) { - zcs->cdict = ZSTD_createCDict_advanced(dict, dictSize, 0, params, cmem); + zcs->cdict = ZSTD_createCDict_advanced(dict, dictSize, 0, params.cParams, cmem); if (zcs->cdict == NULL) return ERROR(memory_allocation); } } zcs->frameContentSize = pledgedSrcSize; diff --git a/lib/zstd.h b/lib/zstd.h index 158ff66a1..b7637f0f3 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -488,7 +488,7 @@ ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_byReference(const void* dictBuffer, siz /*! ZSTD_createCDict_advanced() : * Create a ZSTD_CDict using external alloc and free, and customized compression parameters */ ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_advanced(const void* dict, size_t dictSize, unsigned byReference, - ZSTD_parameters params, ZSTD_customMem customMem); + ZSTD_compressionParameters cParams, ZSTD_customMem customMem); /*! ZSTD_sizeof_CDict() : * Gives the amount of memory used by a given ZSTD_sizeof_CDict */ diff --git a/programs/bench.c b/programs/bench.c index c3681eb05..22b871952 100644 --- a/programs/bench.c +++ b/programs/bench.c @@ -274,7 +274,7 @@ static int BMK_benchMem(const void* srcBuffer, size_t srcSize, if (comprParams->searchLength) zparams.cParams.searchLength = comprParams->searchLength; if (comprParams->targetLength) zparams.cParams.targetLength = comprParams->targetLength; if (comprParams->strategy) zparams.cParams.strategy = (ZSTD_strategy)(comprParams->strategy - 1); - cdict = ZSTD_createCDict_advanced(dictBuffer, dictBufferSize, 1, zparams, cmem); + cdict = ZSTD_createCDict_advanced(dictBuffer, dictBufferSize, 1, zparams.cParams, cmem); if (cdict==NULL) EXM_THROW(1, "ZSTD_createCDict_advanced() allocation failure"); do { U32 blockNb; diff --git a/programs/fileio.c b/programs/fileio.c index 70d35ad67..944b11ab4 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -105,6 +105,7 @@ static clock_t g_time = 0; #undef MIN #define MIN(a,b) ((a) < (b) ? (a) : (b)) + /* ************************************************************ * Avoid fseek()'s 2GiB barrier with MSVC, MacOS, *BSD, MinGW ***************************************************************/ diff --git a/tests/fuzzer.c b/tests/fuzzer.c index 3123d1825..dd9898d37 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -407,7 +407,7 @@ static int basicUnitTests(U32 seed, double compressibility) DISPLAYLEVEL(4, "test%3i : compress with preprocessed dictionary : ", testNb++); { ZSTD_parameters params = ZSTD_getParams(1, CNBuffSize, dictSize); { ZSTD_customMem customMem = { NULL, NULL, NULL }; - ZSTD_CDict* cdict = ZSTD_createCDict_advanced(dictBuffer, dictSize, 1, params, customMem); + ZSTD_CDict* cdict = ZSTD_createCDict_advanced(dictBuffer, dictSize, 1, params.cParams, customMem); cSize = ZSTD_compress_usingCDict(cctx, compressedBuffer, ZSTD_compressBound(CNBuffSize), CNBuffer, CNBuffSize, cdict); ZSTD_freeCDict(cdict); diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index 4048a7df0..e3bb77d8b 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -453,7 +453,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo ZSTD_CDict* cdict; params.fParams.noDictIDFlag = 1; params.fParams.contentSizeFlag = 1; /* test contentSize, should be disabled with initCStream_usingCDict */ - cdict = ZSTD_createCDict_advanced(dictionary.start, dictionary.filled, 1, params, customMem); + cdict = ZSTD_createCDict_advanced(dictionary.start, dictionary.filled, 1, params.cParams, customMem); { size_t const initError = ZSTD_initCStream_usingCDict(zc, cdict); if (ZSTD_isError(initError)) goto _output_error; } cSize = 0; diff --git a/zlibWrapper/examples/zwrapbench.c b/zlibWrapper/examples/zwrapbench.c index 401daa1b5..a57ed51ec 100644 --- a/zlibWrapper/examples/zwrapbench.c +++ b/zlibWrapper/examples/zwrapbench.c @@ -236,7 +236,7 @@ static int BMK_benchMem(z_const void* srcBuffer, size_t srcSize, if (compressor == BMK_ZSTD) { ZSTD_parameters const zparams = ZSTD_getParams(cLevel, avgSize, dictBufferSize); ZSTD_customMem const cmem = { NULL, NULL, NULL }; - ZSTD_CDict* cdict = ZSTD_createCDict_advanced(dictBuffer, dictBufferSize, 1, zparams, cmem); + ZSTD_CDict* const cdict = ZSTD_createCDict_advanced(dictBuffer, dictBufferSize, 1, zparams.cParams, cmem); if (cdict==NULL) EXM_THROW(1, "ZSTD_createCDict_advanced() allocation failure"); do { From 69a54d138ac71f5fc7e935a9f373663a644dc591 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 01:10:36 -0700 Subject: [PATCH 242/305] fixed compilation warning : declaration-after-statement --- lib/common/zstd_internal.h | 5 +++-- lib/compress/zstd_compress.c | 4 +--- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/lib/common/zstd_internal.h b/lib/common/zstd_internal.h index 559d8bec6..9bcb40ffc 100644 --- a/lib/common/zstd_internal.h +++ b/lib/common/zstd_internal.h @@ -16,9 +16,10 @@ #ifdef _MSC_VER /* Visual Studio */ # define FORCE_INLINE static __forceinline # include /* For Visual 2005 */ -# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ -# pragma warning(disable : 4324) /* disable: C4324: padded structure */ # pragma warning(disable : 4100) /* disable: C4100: unreferenced formal parameter */ +# pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ +# pragma warning(disable : 4204) /* disable: C4204: non-constant aggregate initializer */ +# pragma warning(disable : 4324) /* disable: C4324: padded structure */ #else # if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ # ifdef __GNUC__ diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 5daa55542..1ed0715b4 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -2975,9 +2975,7 @@ ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, u } { ZSTD_frameParameters const fParams = { 0 /* contentSizeFlag */, 0 /* checksumFlag */, 0 /* noDictIDFlag */ }; /* dummy */ - ZSTD_parameters params; - params.cParams = cParams; - params.fParams = fParams; + ZSTD_parameters const params = { cParams, fParams }; size_t const errorCode = ZSTD_compressBegin_advanced(cctx, cdict->dictContent, dictSize, params, 0); if (ZSTD_isError(errorCode)) { ZSTD_free(cdict->dictBuffer, customMem); From 0bd5d25d020b8112993156a6501d7adcea0edaa8 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Thu, 27 Apr 2017 09:55:19 -0700 Subject: [PATCH 243/305] [pzstd] Add logging statements to tests --- contrib/pzstd/utils/test/ThreadPoolTest.cpp | 6 +++++- contrib/pzstd/utils/test/WorkQueueTest.cpp | 7 +++++++ 2 files changed, 12 insertions(+), 1 deletion(-) diff --git a/contrib/pzstd/utils/test/ThreadPoolTest.cpp b/contrib/pzstd/utils/test/ThreadPoolTest.cpp index 1d857aae8..89085afd4 100644 --- a/contrib/pzstd/utils/test/ThreadPoolTest.cpp +++ b/contrib/pzstd/utils/test/ThreadPoolTest.cpp @@ -10,6 +10,7 @@ #include #include +#include #include #include @@ -34,16 +35,19 @@ TEST(ThreadPool, AllJobsFinished) { std::atomic numFinished{0}; std::atomic start{false}; { + std::cerr << "Creating executor" << std::endl; ThreadPool executor(5); for (int i = 0; i < 10; ++i) { executor.add([ &numFinished, &start ] { while (!start.load()) { - // spin + std::this_thread::yield(); } ++numFinished; }); } + std::cerr << "Starting" << std::endl; start.store(true); + std::cerr << "Finishing" << std::endl; } EXPECT_EQ(10, numFinished.load()); } diff --git a/contrib/pzstd/utils/test/WorkQueueTest.cpp b/contrib/pzstd/utils/test/WorkQueueTest.cpp index 7f58ccb3f..8caf170d2 100644 --- a/contrib/pzstd/utils/test/WorkQueueTest.cpp +++ b/contrib/pzstd/utils/test/WorkQueueTest.cpp @@ -10,6 +10,7 @@ #include "utils/WorkQueue.h" #include +#include #include #include #include @@ -201,11 +202,13 @@ TEST(WorkQueue, BoundedSizeMPMC) { WorkQueue queue(10); std::vector results(200, -1); std::mutex mutex; + std::cerr << "Creating popperThreads" << std::endl; std::vector popperThreads; for (int i = 0; i < 4; ++i) { popperThreads.emplace_back(Popper{&queue, results.data(), &mutex}); } + std::cerr << "Creating pusherThreads" << std::endl; std::vector pusherThreads; for (int i = 0; i < 2; ++i) { auto min = i * 100; @@ -218,15 +221,19 @@ TEST(WorkQueue, BoundedSizeMPMC) { }); } + std::cerr << "Joining pusherThreads" << std::endl; for (auto& thread : pusherThreads) { thread.join(); } + std::cerr << "Finishing queue" << std::endl; queue.finish(); + std::cerr << "Joining popperThreads" << std::endl; for (auto& thread : popperThreads) { thread.join(); } + std::cerr << "Inspecting results" << std::endl; for (int i = 0; i < 200; ++i) { EXPECT_EQ(i, results[i]); } From f4bd857d8139d81c1f6f72ae15ed746b67197cfc Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 11:31:55 -0700 Subject: [PATCH 244/305] created ZSTD_compress_usingCDict_advanced() --- doc/zstd_manual.html | 19 +++++++++++++------ lib/compress/zstd_compress.c | 12 ++++++++++-- lib/zstd.h | 19 +++++++++++++------ tests/fuzzer.c | 28 ++++++++++++++-------------- 4 files changed, 50 insertions(+), 28 deletions(-) diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index 9121b460f..6023189ba 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -427,12 +427,19 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v both values are optional, select `0` if unknown.


-
size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx,
-                               void* dst, size_t dstCapacity,
-                         const void* src, size_t srcSize,
-                         const void* dict,size_t dictSize,
-                               ZSTD_parameters params);
-

Same as ZSTD_compress_usingDict(), with fine-tune control of each compression parameter +

size_t ZSTD_compress_advanced (ZSTD_CCtx* cctx,
+                      void* dst, size_t dstCapacity,
+                const void* src, size_t srcSize,
+                const void* dict,size_t dictSize,
+                      ZSTD_parameters params);
+

Same as ZSTD_compress_usingDict(), with fine-tune control over each compression parameter +


+ +
size_t ZSTD_compress_usingCDict_advanced(ZSTD_CCtx* cctx,
+                      void* dst, size_t dstCapacity,
+                const void* src, size_t srcSize,
+                const ZSTD_CDict* cdict, ZSTD_frameParameters fParams);
+

Same as ZSTD_compress_usingDict_advanced(), with fine-tune control over frame parameters


Advanced decompression functions


diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 1ed0715b4..d8d29bde8 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -3047,6 +3047,15 @@ size_t ZSTD_compressBegin_usingCDict(ZSTD_CCtx* cctx, const ZSTD_CDict* cdict)
     return ZSTD_compressBegin_usingCDict_advanced(cctx, cdict, fParams, 0);
 }
 
+size_t ZSTD_compress_usingCDict_advanced(ZSTD_CCtx* cctx,
+                                void* dst, size_t dstCapacity,
+                                const void* src, size_t srcSize,
+                                const ZSTD_CDict* cdict, ZSTD_frameParameters fParams)
+{
+    CHECK_F (ZSTD_compressBegin_usingCDict_advanced(cctx, cdict, fParams, srcSize));   /* will check if cdict != NULL */
+    return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize);
+}
+
 /*! ZSTD_compress_usingCDict() :
  *  Compression using a digested Dictionary.
  *  Faster startup than ZSTD_compress_usingDict(), recommended when same dictionary is used multiple times.
@@ -3058,8 +3067,7 @@ size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx,
                                 const ZSTD_CDict* cdict)
 {
     ZSTD_frameParameters const fParams = { 1 /*content*/, 0 /*checksum*/, 0 /*noDictID*/ };
-    CHECK_F (ZSTD_compressBegin_usingCDict_advanced(cctx, cdict, fParams, srcSize));   /* will check if cdict != NULL */
-    return ZSTD_compressEnd(cctx, dst, dstCapacity, src, srcSize);
+    return ZSTD_compress_usingCDict_advanced(cctx, dst, dstCapacity, src, srcSize, cdict, fParams);
 }
 
 
diff --git a/lib/zstd.h b/lib/zstd.h
index b7637f0f3..6a21cd65a 100644
--- a/lib/zstd.h
+++ b/lib/zstd.h
@@ -514,12 +514,19 @@ ZSTDLIB_API size_t ZSTD_checkCParams(ZSTD_compressionParameters params);
 ZSTDLIB_API ZSTD_compressionParameters ZSTD_adjustCParams(ZSTD_compressionParameters cPar, unsigned long long srcSize, size_t dictSize);
 
 /*! ZSTD_compress_advanced() :
-*   Same as ZSTD_compress_usingDict(), with fine-tune control of each compression parameter */
-ZSTDLIB_API size_t ZSTD_compress_advanced (ZSTD_CCtx* ctx,
-                                           void* dst, size_t dstCapacity,
-                                     const void* src, size_t srcSize,
-                                     const void* dict,size_t dictSize,
-                                           ZSTD_parameters params);
+*   Same as ZSTD_compress_usingDict(), with fine-tune control over each compression parameter */
+ZSTDLIB_API size_t ZSTD_compress_advanced (ZSTD_CCtx* cctx,
+                                  void* dst, size_t dstCapacity,
+                            const void* src, size_t srcSize,
+                            const void* dict,size_t dictSize,
+                                  ZSTD_parameters params);
+
+/*! ZSTD_compress_usingCDict_advanced() :
+*   Same as ZSTD_compress_usingCDict(), with fine-tune control over frame parameters */
+ZSTDLIB_API size_t ZSTD_compress_usingCDict_advanced(ZSTD_CCtx* cctx,
+                                  void* dst, size_t dstCapacity,
+                            const void* src, size_t srcSize,
+                            const ZSTD_CDict* cdict, ZSTD_frameParameters fParams);
 
 
 /*--- Advanced decompression functions ---*/
diff --git a/tests/fuzzer.c b/tests/fuzzer.c
index dd9898d37..1515e6f6d 100644
--- a/tests/fuzzer.c
+++ b/tests/fuzzer.c
@@ -405,9 +405,9 @@ static int basicUnitTests(U32 seed, double compressibility)
         DISPLAYLEVEL(4, "OK \n");
 
         DISPLAYLEVEL(4, "test%3i : compress with preprocessed dictionary : ", testNb++);
-        {   ZSTD_parameters params = ZSTD_getParams(1, CNBuffSize, dictSize);
+        {   ZSTD_compressionParameters cParams = ZSTD_getCParams(1, CNBuffSize, dictSize);
             {   ZSTD_customMem customMem = { NULL, NULL, NULL };
-                ZSTD_CDict* cdict = ZSTD_createCDict_advanced(dictBuffer, dictSize, 1, params.cParams, customMem);
+                ZSTD_CDict* cdict = ZSTD_createCDict_advanced(dictBuffer, dictSize, 1, cParams, customMem);
                 cSize = ZSTD_compress_usingCDict(cctx, compressedBuffer, ZSTD_compressBound(CNBuffSize),
                                                  CNBuffer, CNBuffSize, cdict);
                 ZSTD_freeCDict(cdict);
@@ -760,7 +760,6 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxD
     for ( ; (testNb <= nbTests) || (FUZ_clockSpan(startClock) < maxClockSpan); testNb++ ) {
         size_t sampleSize, maxTestSize, totalTestSize;
         size_t cSize, totalCSize, totalGenSize;
-        XXH64_state_t xxhState;
         U64 crcOrig;
         BYTE* sampleBuffer;
         const BYTE* dict;
@@ -917,22 +916,22 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxD
                 CHECK (ZSTD_isError(errorCode), "ZSTD_compressBegin_usingDict error : %s", ZSTD_getErrorName(errorCode));
             } else {
                 ZSTD_compressionParameters const cPar = ZSTD_getCParams(cLevel, 0, dictSize);
-                ZSTD_frameParameters const fpar = { FUZ_rand(&lseed)&1 /* contentSizeFlag */,
+                ZSTD_frameParameters const fPar = { FUZ_rand(&lseed)&1 /* contentSizeFlag */,
                                                     !(FUZ_rand(&lseed)&3) /* contentChecksumFlag*/,
                                                     0 /*NodictID*/ };   /* note : since dictionary is fake, dictIDflag has no impact */
-                ZSTD_parameters p;
-                size_t errorCode;
-                p.cParams = cPar; p.fParams = fpar;
-                errorCode = ZSTD_compressBegin_advanced(refCtx, dict, dictSize, p, 0);
+                ZSTD_parameters const p = { cPar, fPar };
+                size_t const errorCode = ZSTD_compressBegin_advanced(refCtx, dict, dictSize, p, 0);
                 CHECK (ZSTD_isError(errorCode), "ZSTD_compressBegin_advanced error : %s", ZSTD_getErrorName(errorCode));
             }
             {   size_t const errorCode = ZSTD_copyCCtx(ctx, refCtx, 0);
                 CHECK (ZSTD_isError(errorCode), "ZSTD_copyCCtx error : %s", ZSTD_getErrorName(errorCode));
         }   }
-        XXH64_reset(&xxhState, 0);
         ZSTD_setCCtxParameter(ctx, ZSTD_p_forceWindow, FUZ_rand(&lseed) & 1);
+
         {   U32 const nbChunks = (FUZ_rand(&lseed) & 127) + 2;
             U32 n;
+            XXH64_state_t xxhState;
+            XXH64_reset(&xxhState, 0);
             for (totalTestSize=0, cSize=0, n=0 ; n
Date: Thu, 27 Apr 2017 11:43:04 -0700
Subject: [PATCH 245/305] added ZSTD_initCStream_usingCDict_advanced()

---
 doc/zstd_manual.html         |  3 ++-
 lib/compress/zstd_compress.c | 16 ++++++++++++----
 lib/zstd.h                   |  1 +
 3 files changed, 15 insertions(+), 5 deletions(-)

diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html
index 6023189ba..5a9d1b5d4 100644
--- a/doc/zstd_manual.html
+++ b/doc/zstd_manual.html
@@ -439,7 +439,7 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v
                       void* dst, size_t dstCapacity,
                 const void* src, size_t srcSize,
                 const ZSTD_CDict* cdict, ZSTD_frameParameters fParams);
-

Same as ZSTD_compress_usingDict_advanced(), with fine-tune control over frame parameters +

Same as ZSTD_compress_usingCDict(), with fine-tune control over frame parameters


Advanced decompression functions


@@ -511,6 +511,7 @@ size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t di
 size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize,
                                              ZSTD_parameters params, unsigned long long pledgedSrcSize);  /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */
 size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict);  /**< note : cdict will just be referenced, and must outlive compression session */
+size_t ZSTD_initCStream_usingCDict_advanced(ZSTD_CStream* zcs, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, ZSTD_frameParameters fParams);  /**< same as ZSTD_initCStream_usingCDict(), with control over frame parameters */
 

size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize);
 

start a new compression job, using same parameters from previous job. diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index d8d29bde8..d6c5fffb7 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -3211,17 +3211,25 @@ static size_t ZSTD_initCStream_stage2(ZSTD_CStream* zcs, return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); } -/* note : cdict must outlive compression session */ -size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) +/* ZSTD_initCStream_usingCDict_advanced() : + * same as ZSTD_initCStream_usingCDict(), with control over frame parameters */ +size_t ZSTD_initCStream_usingCDict_advanced(ZSTD_CStream* zcs, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, ZSTD_frameParameters fParams) { if (!cdict) return ERROR(GENERIC); /* cannot handle NULL cdict (does not know what to do) */ { ZSTD_parameters params = ZSTD_getParamsFromCDict(cdict); - params.fParams.contentSizeFlag = 0; + params.fParams = fParams; zcs->cdict = cdict; - return ZSTD_initCStream_stage2(zcs, params, 0); + return ZSTD_initCStream_stage2(zcs, params, pledgedSrcSize); } } +/* note : cdict must outlive compression session */ +size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict) +{ + ZSTD_frameParameters const fParams = { 0 /* content */, 0 /* checksum */, 0 /* noDictID */ }; + return ZSTD_initCStream_usingCDict_advanced(zcs, cdict, 0, fParams); +} + static size_t ZSTD_initCStream_internal(ZSTD_CStream* zcs, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize) diff --git a/lib/zstd.h b/lib/zstd.h index 6a21cd65a..3852b7200 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -602,6 +602,7 @@ ZSTDLIB_API size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dic ZSTDLIB_API size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize, ZSTD_parameters params, unsigned long long pledgedSrcSize); /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */ ZSTDLIB_API size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict); /**< note : cdict will just be referenced, and must outlive compression session */ +ZSTDLIB_API size_t ZSTD_initCStream_usingCDict_advanced(ZSTD_CStream* zcs, const ZSTD_CDict* cdict, unsigned long long pledgedSrcSize, ZSTD_frameParameters fParams); /**< same as ZSTD_initCStream_usingCDict(), with control over frame parameters */ /*! ZSTD_resetCStream() : * start a new compression job, using same parameters from previous job. From 5a804456bc9b2056ce3f7d3ca29fa2440d8f5bc9 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 12:49:22 -0700 Subject: [PATCH 246/305] updated NEWS --- NEWS | 27 +++++++++++++++++++++------ 1 file changed, 21 insertions(+), 6 deletions(-) diff --git a/NEWS b/NEWS index fe0c64624..3f23d3f41 100644 --- a/NEWS +++ b/NEWS @@ -1,6 +1,21 @@ v1.1.5 -cli : fix : does not output compressed data on console, by Sean Purcell +cli : changed : Multithreading enabled by default (use target zstd-nomt or HAVE_THREAD=0 to disable) +cli : new : command -T0 means "detect and use nb of cores", by Sean Purcell +cli : new : zstdmt symlink hardwired to `zstd -T0` +cli : new : experimental target `zstd4`, with support for lz4 format, by Sean Purcell +cli : fix : does not output compressed data on console +cli : fix : ignore symbolic links unless --force specified, +API : breaking change : ZSTD_createCDict_advanced(), only use compressionParameters as argument +API : changed : added prototypes ZSTD_*_usingCDict_advanced(), for direct control over frameParameters. +API : changed : enforced consistent rules for pledgedSrcSize==0 +API : improved: ZSTDMT_compressCCtx() reduced memory usage +API : fix : ZSTDMT_compressCCtx() now provides srcSize in header (#634) +API : fix : src size stored in frame header is controlled at end of frame +API : fix : ensure error code "dstSizeTooSmall" is generated when dst size is too small build: improved cmake script, by @Majlen +build: enabled Multi-threading support for *BSD, by Baptiste Daroussin +tools: updated Paramgrill. Command -O# provides best parameters for sample and speed target. +new : contrib/linux-kernel version, by Nick Terrell v1.1.4 cli : new : can compress in *.gz format, using --format=gzip command, by Przemyslaw Skibinski @@ -9,13 +24,13 @@ cli : fix : write on sparse-enabled file systems in 32-bits mode, by @ds77 cli : fix : --rm remains silent when input is stdin cli : experimental : xzstd, with support for xz/lzma decoding, by Przemyslaw Skibinski speed : improved decompression speed in streaming mode for single shot scenarios (+5%) -memory : DDict (decompression dictionary) memory usage down from 150 KB to 20 KB -arch : 32-bits variant able to generate and decode very long matches (>32 MB), by Sean Purcell +memory: DDict (decompression dictionary) memory usage down from 150 KB to 20 KB +arch: 32-bits variant able to generate and decode very long matches (>32 MB), by Sean Purcell API : new : ZSTD_findFrameCompressedSize(), ZSTD_getFrameContentSize(), ZSTD_findDecompressedSize() API : changed : dropped support of legacy versions <= v0.3 (can be changed by modifying ZSTD_LEGACY_SUPPORT value) -build: new: meson build system in contrib/meson, by Dima Krasner -build: improved cmake script, by @Majlen -build: added -Wformat-security flag, as recommended by Padraig Brady +build : new: meson build system in contrib/meson, by Dima Krasner +build : improved cmake script, by @Majlen +build : added -Wformat-security flag, as recommended by Padraig Brady doc : new : educational decoder, by Sean Purcell v1.1.3 From 8b669535f899725bf5bcb9547ffb8427751c03d1 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 12:50:20 -0700 Subject: [PATCH 247/305] bumped version number to v1.2.0 --- NEWS | 2 +- doc/zstd_manual.html | 4 ++-- lib/zstd.h | 4 ++-- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/NEWS b/NEWS index 3f23d3f41..4f9bd2eff 100644 --- a/NEWS +++ b/NEWS @@ -1,4 +1,4 @@ -v1.1.5 +v1.2.0 cli : changed : Multithreading enabled by default (use target zstd-nomt or HAVE_THREAD=0 to disable) cli : new : command -T0 means "detect and use nb of cores", by Sean Purcell cli : new : zstdmt symlink hardwired to `zstd -T0` diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index 73e200b4e..5bfd4e291 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -1,10 +1,10 @@ -zstd 1.1.5 Manual +zstd 1.2.0 Manual -

zstd 1.1.5 Manual

+

zstd 1.2.0 Manual


Contents

    diff --git a/lib/zstd.h b/lib/zstd.h index a99e497fc..88ce6c693 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -55,8 +55,8 @@ extern "C" { /*------ Version ------*/ #define ZSTD_VERSION_MAJOR 1 -#define ZSTD_VERSION_MINOR 1 -#define ZSTD_VERSION_RELEASE 5 +#define ZSTD_VERSION_MINOR 2 +#define ZSTD_VERSION_RELEASE 0 #define ZSTD_LIB_VERSION ZSTD_VERSION_MAJOR.ZSTD_VERSION_MINOR.ZSTD_VERSION_RELEASE #define ZSTD_QUOTE(str) #str From 1c3ab0c77f7924ee2a9f2206ceb5416ea6108ea5 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 12:57:11 -0700 Subject: [PATCH 248/305] fixed init error on Visual 2008 --- lib/compress/zstd_compress.c | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index d6c5fffb7..518358711 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -2948,6 +2948,14 @@ size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict) return ZSTD_sizeof_CCtx(cdict->refContext) + (cdict->dictBuffer ? cdict->dictContentSize : 0) + sizeof(*cdict); } +static ZSTD_parameters ZSTD_makeParams(ZSTD_compressionParameters cParams, ZSTD_frameParameters fParams) +{ + ZSTD_parameters params; + params.cParams = cParams; + params.fParams = fParams; + return params; +} + ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, unsigned byReference, ZSTD_compressionParameters cParams, ZSTD_customMem customMem) { @@ -2975,7 +2983,7 @@ ZSTD_CDict* ZSTD_createCDict_advanced(const void* dictBuffer, size_t dictSize, u } { ZSTD_frameParameters const fParams = { 0 /* contentSizeFlag */, 0 /* checksumFlag */, 0 /* noDictIDFlag */ }; /* dummy */ - ZSTD_parameters const params = { cParams, fParams }; + ZSTD_parameters const params = ZSTD_makeParams(cParams, fParams); size_t const errorCode = ZSTD_compressBegin_advanced(cctx, cdict->dictContent, dictSize, params, 0); if (ZSTD_isError(errorCode)) { ZSTD_free(cdict->dictBuffer, customMem); From 7321345fd2d70ae25187aaa0c437ec1b01827fb2 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 14:19:34 -0700 Subject: [PATCH 249/305] fixed another VS2008 init error --- tests/fuzzer.c | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/tests/fuzzer.c b/tests/fuzzer.c index 1515e6f6d..dadf42c9f 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -697,6 +697,14 @@ static size_t findDiff(const void* buf1, const void* buf2, size_t max) } +static ZSTD_parameters FUZ_makeParams(ZSTD_compressionParameters cParams, ZSTD_frameParameters fParams) +{ + ZSTD_parameters params; + params.cParams = cParams; + params.fParams = fParams; + return params; +} + static size_t FUZ_rLogLength(U32* seed, U32 logLength) { size_t const lengthMask = ((size_t)1 << logLength) - 1; @@ -919,7 +927,7 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxD ZSTD_frameParameters const fPar = { FUZ_rand(&lseed)&1 /* contentSizeFlag */, !(FUZ_rand(&lseed)&3) /* contentChecksumFlag*/, 0 /*NodictID*/ }; /* note : since dictionary is fake, dictIDflag has no impact */ - ZSTD_parameters const p = { cPar, fPar }; + ZSTD_parameters const p = FUZ_makeParams(cPar, fPar); size_t const errorCode = ZSTD_compressBegin_advanced(refCtx, dict, dictSize, p, 0); CHECK (ZSTD_isError(errorCode), "ZSTD_compressBegin_advanced error : %s", ZSTD_getErrorName(errorCode)); } From d3694e6c70702502d7704ec4af5426b488836528 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 14:29:35 -0700 Subject: [PATCH 250/305] removed C4204 --- lib/common/zstd_internal.h | 1 - 1 file changed, 1 deletion(-) diff --git a/lib/common/zstd_internal.h b/lib/common/zstd_internal.h index 9bcb40ffc..2533333ba 100644 --- a/lib/common/zstd_internal.h +++ b/lib/common/zstd_internal.h @@ -18,7 +18,6 @@ # include /* For Visual 2005 */ # pragma warning(disable : 4100) /* disable: C4100: unreferenced formal parameter */ # pragma warning(disable : 4127) /* disable: C4127: conditional expression is constant */ -# pragma warning(disable : 4204) /* disable: C4204: non-constant aggregate initializer */ # pragma warning(disable : 4324) /* disable: C4324: padded structure */ #else # if defined (__cplusplus) || defined (__STDC_VERSION__) && __STDC_VERSION__ >= 199901L /* C99 */ From 2f73427d35863b83a057dda3c2cb2794cf446f90 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 14:39:39 -0700 Subject: [PATCH 251/305] added test for ZSTD_compress_usingCDict_advanced() --- tests/fuzzer.c | 41 +++++++++++++++++++++++++++++++++-------- 1 file changed, 33 insertions(+), 8 deletions(-) diff --git a/tests/fuzzer.c b/tests/fuzzer.c index dadf42c9f..687e7a371 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -405,14 +405,13 @@ static int basicUnitTests(U32 seed, double compressibility) DISPLAYLEVEL(4, "OK \n"); DISPLAYLEVEL(4, "test%3i : compress with preprocessed dictionary : ", testNb++); - { ZSTD_compressionParameters cParams = ZSTD_getCParams(1, CNBuffSize, dictSize); - { ZSTD_customMem customMem = { NULL, NULL, NULL }; - ZSTD_CDict* cdict = ZSTD_createCDict_advanced(dictBuffer, dictSize, 1, cParams, customMem); - cSize = ZSTD_compress_usingCDict(cctx, compressedBuffer, ZSTD_compressBound(CNBuffSize), + { ZSTD_compressionParameters const cParams = ZSTD_getCParams(1, CNBuffSize, dictSize); + ZSTD_customMem customMem = { NULL, NULL, NULL }; + ZSTD_CDict* cdict = ZSTD_createCDict_advanced(dictBuffer, dictSize, 1, cParams, customMem); + cSize = ZSTD_compress_usingCDict(cctx, compressedBuffer, ZSTD_compressBound(CNBuffSize), CNBuffer, CNBuffSize, cdict); - ZSTD_freeCDict(cdict); - if (ZSTD_isError(cSize)) goto _output_error; - } + ZSTD_freeCDict(cdict); + if (ZSTD_isError(cSize)) goto _output_error; } DISPLAYLEVEL(4, "OK (%u bytes : %.2f%%)\n", (U32)cSize, (double)cSize/CNBuffSize*100); @@ -430,7 +429,33 @@ static int basicUnitTests(U32 seed, double compressibility) if (r != CNBuffSize) goto _output_error); DISPLAYLEVEL(4, "OK \n"); - DISPLAYLEVEL(4, "test%3i : compress without dictID : ", testNb++); + DISPLAYLEVEL(4, "test%3i : ZSTD_compress_usingCDict_advanced, no contentSize, no dictID : ", testNb++); + { ZSTD_frameParameters const fParams = { 0 /* frameSize */, 1 /* checksum */, 1 /* noDictID*/ }; + ZSTD_compressionParameters const cParams = ZSTD_getCParams(1, CNBuffSize, dictSize); + ZSTD_customMem const customMem = { NULL, NULL, NULL }; + ZSTD_CDict* const cdict = ZSTD_createCDict_advanced(dictBuffer, dictSize, 1, cParams, customMem); + cSize = ZSTD_compress_usingCDict_advanced(cctx, compressedBuffer, ZSTD_compressBound(CNBuffSize), + CNBuffer, CNBuffSize, cdict, fParams); + ZSTD_freeCDict(cdict); + if (ZSTD_isError(cSize)) goto _output_error; + } + DISPLAYLEVEL(4, "OK (%u bytes : %.2f%%)\n", (U32)cSize, (double)cSize/CNBuffSize*100); + + DISPLAYLEVEL(4, "test%3i : try retrieving contentSize from frame : ", testNb++); + { U64 const contentSize = ZSTD_getFrameContentSize(compressedBuffer, cSize); + if (contentSize != ZSTD_CONTENTSIZE_UNKNOWN) goto _output_error; + } + DISPLAYLEVEL(4, "OK (unknown)\n"); + + DISPLAYLEVEL(4, "test%3i : frame built without dictID should be decompressible : ", testNb++); + CHECKPLUS(r, ZSTD_decompress_usingDict(dctx, + decodedBuffer, CNBuffSize, + compressedBuffer, cSize, + dictBuffer, dictSize), + if (r != CNBuffSize) goto _output_error); + DISPLAYLEVEL(4, "OK \n"); + + DISPLAYLEVEL(4, "test%3i : ZSTD_compress_advanced, no dictID : ", testNb++); { ZSTD_parameters p = ZSTD_getParams(3, CNBuffSize, dictSize); p.fParams.noDictIDFlag = 1; cSize = ZSTD_compress_advanced(cctx, compressedBuffer, ZSTD_compressBound(CNBuffSize), From 7d283cdfa230c5884c56c0d59563a0ebb6684a39 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 14:48:34 -0700 Subject: [PATCH 252/305] added test for ZSTD_initCStream_usingCDict_advanced() --- tests/zstreamtest.c | 14 ++++++-------- 1 file changed, 6 insertions(+), 8 deletions(-) diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index e3bb77d8b..6d9694e45 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -448,14 +448,12 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo if (!ZSTD_isError(r)) goto _output_error; /* must fail : frame requires > 100 bytes */ DISPLAYLEVEL(3, "OK (%s)\n", ZSTD_getErrorName(r)); } - DISPLAYLEVEL(3, "test%3i : dictionary compression with masked dictID : ", testNb++); - { ZSTD_parameters params = ZSTD_getParams(1, CNBufferSize, dictionary.filled); - ZSTD_CDict* cdict; - params.fParams.noDictIDFlag = 1; - params.fParams.contentSizeFlag = 1; /* test contentSize, should be disabled with initCStream_usingCDict */ - cdict = ZSTD_createCDict_advanced(dictionary.start, dictionary.filled, 1, params.cParams, customMem); - { size_t const initError = ZSTD_initCStream_usingCDict(zc, cdict); - if (ZSTD_isError(initError)) goto _output_error; } + DISPLAYLEVEL(3, "test%3i : ZSTD_initCStream_usingCDict_advanced with masked dictID : ", testNb++); + { ZSTD_compressionParameters const cParams = ZSTD_getCParams(1, CNBufferSize, dictionary.filled); + ZSTD_frameParameters const fParams = { 1 /* contentSize */, 1 /* checksum */, 1 /* noDictID */}; + ZSTD_CDict* const cdict = ZSTD_createCDict_advanced(dictionary.start, dictionary.filled, 1, cParams, customMem); + size_t const initError = ZSTD_initCStream_usingCDict_advanced(zc, cdict, CNBufferSize, fParams); + if (ZSTD_isError(initError)) goto _output_error; cSize = 0; outBuff.dst = compressedBuffer; outBuff.size = compressedBufferSize; From 32c658a2898e37f3e78ba64efff5fdc061389a36 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 27 Apr 2017 15:04:31 -0700 Subject: [PATCH 253/305] SquashFS linux kernel patch --- .../linux-kernel/fs/squashfs/zstd_wrapper.c | 146 +++++++++++ contrib/linux-kernel/squashfs.diff | 242 ++++++++++++++++++ 2 files changed, 388 insertions(+) create mode 100644 contrib/linux-kernel/fs/squashfs/zstd_wrapper.c create mode 100644 contrib/linux-kernel/squashfs.diff diff --git a/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c b/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c new file mode 100644 index 000000000..81f4b5a4b --- /dev/null +++ b/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c @@ -0,0 +1,146 @@ +/* + * Squashfs - a compressed read only filesystem for Linux + * + * Copyright (c) 2017 Facebook + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2, + * or (at your option) any later version. + * + * 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 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, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + * + * zstd_wrapper.c + */ + +#include +#include +#include +#include +#include + +#include "squashfs_fs.h" +#include "squashfs_fs_sb.h" +#include "squashfs.h" +#include "decompressor.h" +#include "page_actor.h" + +struct workspace { + void *mem; + size_t mem_size; +}; + +static void *zstd_init(struct squashfs_sb_info *msblk, void *buff) +{ + struct workspace *wksp = kmalloc(sizeof(*wksp), GFP_KERNEL); + if (wksp == NULL) + goto failed; + wksp->mem_size = ZSTD_DStreamWorkspaceBound(max_t(size_t, + msblk->block_size, SQUASHFS_METADATA_SIZE)); + wksp->mem = vmalloc(wksp->mem_size); + if (wksp->mem == NULL) + goto failed; + + return wksp; + +failed: + ERROR("Failed to allocate zstd workspace\n"); + kfree(wksp); + return ERR_PTR(-ENOMEM); +} + + +static void zstd_free(void *strm) +{ + struct workspace *wksp = strm; + + if (wksp) + vfree(wksp->mem); + kfree(wksp); +} + + +static int zstd_uncompress(struct squashfs_sb_info *msblk, void *strm, + struct buffer_head **bh, int b, int offset, int length, + struct squashfs_page_actor *output) +{ + struct workspace *wksp = strm; + ZSTD_DStream *stream; + size_t total_out = 0; + size_t zstd_err; + int k = 0; + ZSTD_inBuffer in_buf = { NULL, 0, 0 }; + ZSTD_outBuffer out_buf = { NULL, 0, 0 }; + + stream = ZSTD_initDStream(wksp->mem_size, wksp->mem, wksp->mem_size); + + if (!stream) { + ERROR("Failed to initialize zstd decompressor\n"); + goto out; + } + + out_buf.size = PAGE_SIZE; + out_buf.dst = squashfs_first_page(output); + + do { + if (in_buf.pos == in_buf.size && k < b) { + int avail = min(length, msblk->devblksize - offset); + length -= avail; + in_buf.src = bh[k]->b_data + offset; + in_buf.size = avail; + in_buf.pos = 0; + offset = 0; + } + + if (out_buf.pos == out_buf.size) { + out_buf.dst = squashfs_next_page(output); + out_buf.pos = 0; + if (out_buf.dst != NULL) { + out_buf.size = PAGE_SIZE; + } else { + out_buf.size = 0; + } + } + + total_out -= out_buf.pos; + zstd_err = ZSTD_decompressStream(stream, &out_buf, &in_buf); + total_out += out_buf.pos; /* add the additional data produced */ + + if (in_buf.pos == in_buf.size && k < b) + put_bh(bh[k++]); + } while (zstd_err != 0 && !ZSTD_isError(zstd_err)); + + if (ZSTD_isError(zstd_err)) { + ERROR("zstd decompression error: %d\n", + (int)ZSTD_getErrorCode(zstd_err)); + } + + squashfs_finish_page(output); + + if (k < b) + goto out; + + return total_out; + +out: + for (; k < b; k++) + put_bh(bh[k]); + + return -EIO; +} + +const struct squashfs_decompressor squashfs_zstd_comp_ops = { + .init = zstd_init, + .free = zstd_free, + .decompress = zstd_uncompress, + .id = ZSTD_COMPRESSION, + .name = "zstd", + .supported = 1 +}; diff --git a/contrib/linux-kernel/squashfs.diff b/contrib/linux-kernel/squashfs.diff new file mode 100644 index 000000000..e1e1235f0 --- /dev/null +++ b/contrib/linux-kernel/squashfs.diff @@ -0,0 +1,242 @@ +commit 7289653483a0579c3b63a06abf008210c8cc6c8b +Author: Sean Purcell +Date: Thu Apr 27 14:56:25 2017 -0700 + + Add zstd support to kernel squashfs + +diff --git a/fs/squashfs/Kconfig b/fs/squashfs/Kconfig +index ffb093e..1adb334 100644 +--- a/fs/squashfs/Kconfig ++++ b/fs/squashfs/Kconfig +@@ -165,6 +165,20 @@ config SQUASHFS_XZ + + If unsure, say N. + ++config SQUASHFS_ZSTD ++ bool "Include support for ZSTD compressed file systems" ++ depends on SQUASHFS ++ select ZSTD_DECOMPRESS ++ help ++ Saying Y here includes support for reading Squashfs file systems ++ compressed with ZSTD compression. ZSTD gives better compression than ++ the default ZLIB compression, while using less CPU. ++ ++ ZSTD is not the standard compression used in Squashfs and so most ++ file systems will be readable without selecting this option. ++ ++ If unsure, say N. ++ + config SQUASHFS_4K_DEVBLK_SIZE + bool "Use 4K device block size?" + depends on SQUASHFS +diff --git a/fs/squashfs/Makefile b/fs/squashfs/Makefile +index 246a6f3..6655631 100644 +--- a/fs/squashfs/Makefile ++++ b/fs/squashfs/Makefile +@@ -15,3 +15,4 @@ squashfs-$(CONFIG_SQUASHFS_LZ4) += lz4_wrapper.o + squashfs-$(CONFIG_SQUASHFS_LZO) += lzo_wrapper.o + squashfs-$(CONFIG_SQUASHFS_XZ) += xz_wrapper.o + squashfs-$(CONFIG_SQUASHFS_ZLIB) += zlib_wrapper.o ++squashfs-$(CONFIG_SQUASHFS_ZSTD) += zstd_wrapper.o +diff --git a/fs/squashfs/decompressor.c b/fs/squashfs/decompressor.c +index d2bc136..8366398 100644 +--- a/fs/squashfs/decompressor.c ++++ b/fs/squashfs/decompressor.c +@@ -65,6 +65,12 @@ static const struct squashfs_decompressor squashfs_zlib_comp_ops = { + }; + #endif + ++#ifndef CONFIG_SQUASHFS_ZSTD ++static const struct squashfs_decompressor squashfs_zstd_comp_ops = { ++ NULL, NULL, NULL, NULL, ZSTD_COMPRESSION, "zstd", 0 ++}; ++#endif ++ + static const struct squashfs_decompressor squashfs_unknown_comp_ops = { + NULL, NULL, NULL, NULL, 0, "unknown", 0 + }; +@@ -75,6 +81,7 @@ static const struct squashfs_decompressor *decompressor[] = { + &squashfs_lzo_comp_ops, + &squashfs_xz_comp_ops, + &squashfs_lzma_unsupported_comp_ops, ++ &squashfs_zstd_comp_ops, + &squashfs_unknown_comp_ops + }; + +diff --git a/fs/squashfs/decompressor.h b/fs/squashfs/decompressor.h +index a25713c..0f5a8e4 100644 +--- a/fs/squashfs/decompressor.h ++++ b/fs/squashfs/decompressor.h +@@ -58,4 +58,8 @@ extern const struct squashfs_decompressor squashfs_lzo_comp_ops; + extern const struct squashfs_decompressor squashfs_zlib_comp_ops; + #endif + ++#ifdef CONFIG_SQUASHFS_ZSTD ++extern const struct squashfs_decompressor squashfs_zstd_comp_ops; ++#endif ++ + #endif +diff --git a/fs/squashfs/squashfs_fs.h b/fs/squashfs/squashfs_fs.h +index 506f4ba..24d12fd 100644 +--- a/fs/squashfs/squashfs_fs.h ++++ b/fs/squashfs/squashfs_fs.h +@@ -241,6 +241,7 @@ struct meta_index { + #define LZO_COMPRESSION 3 + #define XZ_COMPRESSION 4 + #define LZ4_COMPRESSION 5 ++#define ZSTD_COMPRESSION 6 + + struct squashfs_super_block { + __le32 s_magic; +diff --git a/fs/squashfs/zstd_wrapper.c b/fs/squashfs/zstd_wrapper.c +new file mode 100644 +index 0000000..81f4b5a +--- /dev/null ++++ b/fs/squashfs/zstd_wrapper.c +@@ -0,0 +1,146 @@ ++/* ++ * Squashfs - a compressed read only filesystem for Linux ++ * ++ * Copyright (c) 2017 Facebook ++ * ++ * This program is free software; you can redistribute it and/or ++ * modify it under the terms of the GNU General Public License ++ * as published by the Free Software Foundation; either version 2, ++ * or (at your option) any later version. ++ * ++ * 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 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, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. ++ * ++ * zstd_wrapper.c ++ */ ++ ++#include ++#include ++#include ++#include ++#include ++ ++#include "squashfs_fs.h" ++#include "squashfs_fs_sb.h" ++#include "squashfs.h" ++#include "decompressor.h" ++#include "page_actor.h" ++ ++struct workspace { ++ void *mem; ++ size_t mem_size; ++}; ++ ++static void *zstd_init(struct squashfs_sb_info *msblk, void *buff) ++{ ++ struct workspace *wksp = kmalloc(sizeof(*wksp), GFP_KERNEL); ++ if (wksp == NULL) ++ goto failed; ++ wksp->mem_size = ZSTD_DStreamWorkspaceBound(max_t(size_t, ++ msblk->block_size, SQUASHFS_METADATA_SIZE)); ++ wksp->mem = vmalloc(wksp->mem_size); ++ if (wksp->mem == NULL) ++ goto failed; ++ ++ return wksp; ++ ++failed: ++ ERROR("Failed to allocate zstd workspace\n"); ++ kfree(wksp); ++ return ERR_PTR(-ENOMEM); ++} ++ ++ ++static void zstd_free(void *strm) ++{ ++ struct workspace *wksp = strm; ++ ++ if (wksp) ++ vfree(wksp->mem); ++ kfree(wksp); ++} ++ ++ ++static int zstd_uncompress(struct squashfs_sb_info *msblk, void *strm, ++ struct buffer_head **bh, int b, int offset, int length, ++ struct squashfs_page_actor *output) ++{ ++ struct workspace *wksp = strm; ++ ZSTD_DStream *stream; ++ size_t total_out = 0; ++ size_t zstd_err; ++ int k = 0; ++ ZSTD_inBuffer in_buf = { NULL, 0, 0 }; ++ ZSTD_outBuffer out_buf = { NULL, 0, 0 }; ++ ++ stream = ZSTD_initDStream(wksp->mem_size, wksp->mem, wksp->mem_size); ++ ++ if (!stream) { ++ ERROR("Failed to initialize zstd decompressor\n"); ++ goto out; ++ } ++ ++ out_buf.size = PAGE_SIZE; ++ out_buf.dst = squashfs_first_page(output); ++ ++ do { ++ if (in_buf.pos == in_buf.size && k < b) { ++ int avail = min(length, msblk->devblksize - offset); ++ length -= avail; ++ in_buf.src = bh[k]->b_data + offset; ++ in_buf.size = avail; ++ in_buf.pos = 0; ++ offset = 0; ++ } ++ ++ if (out_buf.pos == out_buf.size) { ++ out_buf.dst = squashfs_next_page(output); ++ out_buf.pos = 0; ++ if (out_buf.dst != NULL) { ++ out_buf.size = PAGE_SIZE; ++ } else { ++ out_buf.size = 0; ++ } ++ } ++ ++ total_out -= out_buf.pos; ++ zstd_err = ZSTD_decompressStream(stream, &out_buf, &in_buf); ++ total_out += out_buf.pos; /* add the additional data produced */ ++ ++ if (in_buf.pos == in_buf.size && k < b) ++ put_bh(bh[k++]); ++ } while (zstd_err != 0 && !ZSTD_isError(zstd_err)); ++ ++ if (ZSTD_isError(zstd_err)) { ++ ERROR("zstd decompression error: %d\n", ++ (int)ZSTD_getErrorCode(zstd_err)); ++ } ++ ++ squashfs_finish_page(output); ++ ++ if (k < b) ++ goto out; ++ ++ return total_out; ++ ++out: ++ for (; k < b; k++) ++ put_bh(bh[k]); ++ ++ return -EIO; ++} ++ ++const struct squashfs_decompressor squashfs_zstd_comp_ops = { ++ .init = zstd_init, ++ .free = zstd_free, ++ .decompress = zstd_uncompress, ++ .id = ZSTD_COMPRESSION, ++ .name = "zstd", ++ .supported = 1 ++}; From a92cbb70045313ece955d33e7dfe1e2c815b7fab Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 15:08:56 -0700 Subject: [PATCH 254/305] Added a secondary test, checking dictID presence after setting noDictIdFLag=1 --- lib/zstd.h | 2 +- tests/zstreamtest.c | 8 +++++++- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/lib/zstd.h b/lib/zstd.h index 3852b7200..9a50f23cd 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -586,7 +586,7 @@ ZSTDLIB_API unsigned ZSTD_getDictID_fromDDict(const ZSTD_DDict* ddict); * Note : this use case also happens when using a non-conformant dictionary. * - `srcSize` is too small, and as a result, the frame header could not be decoded (only possible if `srcSize < ZSTD_FRAMEHEADERSIZE_MAX`). * - This is not a Zstandard frame. - * When identifying the exact failure cause, it's possible to used ZSTD_getFrameParams(), which will provide a more precise error code. */ + * When identifying the exact failure cause, it's possible to use ZSTD_getFrameParams(), which will provide a more precise error code. */ ZSTDLIB_API unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize); diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index 6d9694e45..0e09e1856 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -451,7 +451,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo DISPLAYLEVEL(3, "test%3i : ZSTD_initCStream_usingCDict_advanced with masked dictID : ", testNb++); { ZSTD_compressionParameters const cParams = ZSTD_getCParams(1, CNBufferSize, dictionary.filled); ZSTD_frameParameters const fParams = { 1 /* contentSize */, 1 /* checksum */, 1 /* noDictID */}; - ZSTD_CDict* const cdict = ZSTD_createCDict_advanced(dictionary.start, dictionary.filled, 1, cParams, customMem); + ZSTD_CDict* const cdict = ZSTD_createCDict_advanced(dictionary.start, dictionary.filled, 1 /* byReference */, cParams, customMem); size_t const initError = ZSTD_initCStream_usingCDict_advanced(zc, cdict, CNBufferSize, fParams); if (ZSTD_isError(initError)) goto _output_error; cSize = 0; @@ -471,6 +471,12 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo DISPLAYLEVEL(3, "OK (%u bytes : %.2f%%)\n", (U32)cSize, (double)cSize/CNBufferSize*100); } + DISPLAYLEVEL(3, "test%3i : try retrieving dictID from frame : ", testNb++); + { U32 const did = ZSTD_getDictID_fromFrame(compressedBuffer, cSize); + if (did != 0) goto _output_error; + } + DISPLAYLEVEL(3, "OK (not detected) \n"); + DISPLAYLEVEL(3, "test%3i : decompress without dictionary : ", testNb++); { size_t const r = ZSTD_decompress(decodedBuffer, CNBufferSize, compressedBuffer, cSize); if (!ZSTD_isError(r)) goto _output_error; /* must fail : dictionary not used */ From c6915429f234f03d7f75633e3a9f4f08843bf765 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 16:24:53 -0700 Subject: [PATCH 255/305] shortened Appveyor release tests fuzzer supports time suffix `s` for "seconds" --- appveyor.yml | 2 +- tests/fuzzer.c | 81 +++++++++++++++++++++++++------------------------- 2 files changed, 41 insertions(+), 42 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index 4aad71469..67c4f72d7 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -166,7 +166,7 @@ make -C contrib\pzstd check && make -C contrib\pzstd clean ) - - SET "FUZZERTEST=-T1mn" + - SET "FUZZERTEST=-T30s" - if [%HOST%]==[visual] if [%CONFIGURATION%]==[Release] ( CD tests && SET ZSTD=./zstd.exe && diff --git a/tests/fuzzer.c b/tests/fuzzer.c index 3123d1825..a4423f244 100644 --- a/tests/fuzzer.c +++ b/tests/fuzzer.c @@ -1003,7 +1003,7 @@ _output_error: /*_******************************************************* * Command line *********************************************************/ -int FUZ_usage(const char* programName) +static int FUZ_usage(const char* programName) { DISPLAY( "Usage :\n"); DISPLAY( " %s [args]\n", programName); @@ -1019,20 +1019,39 @@ int FUZ_usage(const char* programName) return 0; } +/*! readU32FromChar() : + @return : unsigned integer value read from input in `char` format + allows and interprets K, KB, KiB, M, MB and MiB suffix. + Will also modify `*stringPtr`, advancing it to position where it stopped reading. + Note : function result can overflow if digit string > MAX_UINT */ +static unsigned readU32FromChar(const char** stringPtr) +{ + unsigned result = 0; + while ((**stringPtr >='0') && (**stringPtr <='9')) + result *= 10, result += **stringPtr - '0', (*stringPtr)++ ; + if ((**stringPtr=='K') || (**stringPtr=='M')) { + result <<= 10; + if (**stringPtr=='M') result <<= 10; + (*stringPtr)++ ; + if (**stringPtr=='i') (*stringPtr)++; + if (**stringPtr=='B') (*stringPtr)++; + } + return result; +} int main(int argc, const char** argv) { - U32 seed=0; - int seedset=0; + U32 seed = 0; + int seedset = 0; int argNb; int nbTests = nbTestsDefault; int testNb = 0; U32 proba = FUZ_compressibility_default; - int result=0; + int result = 0; U32 mainPause = 0; U32 maxDuration = 0; int bigTests = 1; - const char* programName = argv[0]; + const char* const programName = argv[0]; /* Check command line */ for (argNb=1; argNb='0') && (*argument<='9')) { - nbTests *= 10; - nbTests += *argument - '0'; - argument++; - } + argument++; maxDuration = 0; + nbTests = readU32FromChar(&argument); break; case 'T': argument++; - nbTests=0; maxDuration=0; - while ((*argument>='0') && (*argument<='9')) { - maxDuration *= 10; - maxDuration += *argument - '0'; - argument++; - } - if (*argument=='m') maxDuration *=60, argument++; + nbTests = 0; + maxDuration = readU32FromChar(&argument); + if (*argument=='s') argument++; /* seconds */ + if (*argument=='m') maxDuration *= 60, argument++; /* minutes */ if (*argument=='n') argument++; break; case 's': argument++; - seed=0; - seedset=1; - while ((*argument>='0') && (*argument<='9')) { - seed *= 10; - seed += *argument - '0'; - argument++; - } + seedset = 1; + seed = readU32FromChar(&argument); break; case 't': argument++; - testNb=0; - while ((*argument>='0') && (*argument<='9')) { - testNb *= 10; - testNb += *argument - '0'; - argument++; - } + testNb = readU32FromChar(&argument); break; case 'P': /* compressibility % */ argument++; - proba=0; - while ((*argument>='0') && (*argument<='9')) { - proba *= 10; - proba += *argument - '0'; - argument++; - } - if (proba>100) proba=100; + proba = readU32FromChar(&argument); + if (proba>100) proba = 100; break; default: - return FUZ_usage(programName); + return (FUZ_usage(programName), 1); } } } } /* for (argNb=1; argNb Date: Thu, 27 Apr 2017 16:54:05 -0700 Subject: [PATCH 256/305] Fix case where pages run out before end of stream --- .../linux-kernel/fs/squashfs/zstd_wrapper.c | 13 ++++++------ contrib/linux-kernel/squashfs.diff | 21 ++++++++++--------- 2 files changed, 18 insertions(+), 16 deletions(-) diff --git a/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c b/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c index 81f4b5a4b..af72c7b22 100644 --- a/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c +++ b/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c @@ -101,12 +101,13 @@ static int zstd_uncompress(struct squashfs_sb_info *msblk, void *strm, if (out_buf.pos == out_buf.size) { out_buf.dst = squashfs_next_page(output); - out_buf.pos = 0; - if (out_buf.dst != NULL) { - out_buf.size = PAGE_SIZE; - } else { - out_buf.size = 0; + if (out_buf.dst == NULL) { + /* shouldn't run out of pages before stream is + * done */ + goto out; } + out_buf.pos = 0; + out_buf.size = PAGE_SIZE; } total_out -= out_buf.pos; @@ -127,7 +128,7 @@ static int zstd_uncompress(struct squashfs_sb_info *msblk, void *strm, if (k < b) goto out; - return total_out; + return (int)total_out; out: for (; k < b; k++) diff --git a/contrib/linux-kernel/squashfs.diff b/contrib/linux-kernel/squashfs.diff index e1e1235f0..e3b9e8552 100644 --- a/contrib/linux-kernel/squashfs.diff +++ b/contrib/linux-kernel/squashfs.diff @@ -1,6 +1,6 @@ -commit 7289653483a0579c3b63a06abf008210c8cc6c8b +commit 47ba72b36506f91c3774c0bb1fa3c7a5dcfe3ea1 Author: Sean Purcell -Date: Thu Apr 27 14:56:25 2017 -0700 +Date: Thu Apr 27 16:50:53 2017 -0700 Add zstd support to kernel squashfs @@ -90,10 +90,10 @@ index 506f4ba..24d12fd 100644 __le32 s_magic; diff --git a/fs/squashfs/zstd_wrapper.c b/fs/squashfs/zstd_wrapper.c new file mode 100644 -index 0000000..81f4b5a +index 0000000..af72c7b --- /dev/null +++ b/fs/squashfs/zstd_wrapper.c -@@ -0,0 +1,146 @@ +@@ -0,0 +1,147 @@ +/* + * Squashfs - a compressed read only filesystem for Linux + * @@ -197,12 +197,13 @@ index 0000000..81f4b5a + + if (out_buf.pos == out_buf.size) { + out_buf.dst = squashfs_next_page(output); -+ out_buf.pos = 0; -+ if (out_buf.dst != NULL) { -+ out_buf.size = PAGE_SIZE; -+ } else { -+ out_buf.size = 0; ++ if (out_buf.dst == NULL) { ++ /* shouldn't run out of pages before stream is ++ * done */ ++ goto out; + } ++ out_buf.pos = 0; ++ out_buf.size = PAGE_SIZE; + } + + total_out -= out_buf.pos; @@ -223,7 +224,7 @@ index 0000000..81f4b5a + if (k < b) + goto out; + -+ return total_out; ++ return (int)total_out; + +out: + for (; k < b; k++) From c7e107197a80577057adfa3c2bd1cbc152f72285 Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Thu, 27 Apr 2017 17:19:20 -0700 Subject: [PATCH 257/305] Fix missing squashfs_finish_page's --- contrib/linux-kernel/fs/squashfs/zstd_wrapper.c | 6 ++++-- contrib/linux-kernel/squashfs.diff | 16 +++++++++------- 2 files changed, 13 insertions(+), 9 deletions(-) diff --git a/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c b/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c index af72c7b22..7cc93030b 100644 --- a/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c +++ b/contrib/linux-kernel/fs/squashfs/zstd_wrapper.c @@ -104,6 +104,7 @@ static int zstd_uncompress(struct squashfs_sb_info *msblk, void *strm, if (out_buf.dst == NULL) { /* shouldn't run out of pages before stream is * done */ + squashfs_finish_page(output); goto out; } out_buf.pos = 0; @@ -118,13 +119,14 @@ static int zstd_uncompress(struct squashfs_sb_info *msblk, void *strm, put_bh(bh[k++]); } while (zstd_err != 0 && !ZSTD_isError(zstd_err)); + squashfs_finish_page(output); + if (ZSTD_isError(zstd_err)) { ERROR("zstd decompression error: %d\n", (int)ZSTD_getErrorCode(zstd_err)); + goto out; } - squashfs_finish_page(output); - if (k < b) goto out; diff --git a/contrib/linux-kernel/squashfs.diff b/contrib/linux-kernel/squashfs.diff index e3b9e8552..ddf7b3578 100644 --- a/contrib/linux-kernel/squashfs.diff +++ b/contrib/linux-kernel/squashfs.diff @@ -1,8 +1,8 @@ -commit 47ba72b36506f91c3774c0bb1fa3c7a5dcfe3ea1 +commit 16bb6b9fd684eadba41a36223d67805d7ea741e7 Author: Sean Purcell -Date: Thu Apr 27 16:50:53 2017 -0700 +Date: Thu Apr 27 17:17:58 2017 -0700 - Add zstd support to kernel squashfs + Add zstd support to squashfs diff --git a/fs/squashfs/Kconfig b/fs/squashfs/Kconfig index ffb093e..1adb334 100644 @@ -90,10 +90,10 @@ index 506f4ba..24d12fd 100644 __le32 s_magic; diff --git a/fs/squashfs/zstd_wrapper.c b/fs/squashfs/zstd_wrapper.c new file mode 100644 -index 0000000..af72c7b +index 0000000..7cc9303 --- /dev/null +++ b/fs/squashfs/zstd_wrapper.c -@@ -0,0 +1,147 @@ +@@ -0,0 +1,149 @@ +/* + * Squashfs - a compressed read only filesystem for Linux + * @@ -200,6 +200,7 @@ index 0000000..af72c7b + if (out_buf.dst == NULL) { + /* shouldn't run out of pages before stream is + * done */ ++ squashfs_finish_page(output); + goto out; + } + out_buf.pos = 0; @@ -214,13 +215,14 @@ index 0000000..af72c7b + put_bh(bh[k++]); + } while (zstd_err != 0 && !ZSTD_isError(zstd_err)); + ++ squashfs_finish_page(output); ++ + if (ZSTD_isError(zstd_err)) { + ERROR("zstd decompression error: %d\n", + (int)ZSTD_getErrorCode(zstd_err)); ++ goto out; + } + -+ squashfs_finish_page(output); -+ + if (k < b) + goto out; + From 36e79a1a1f2126061782ae4719bc888b8f4e112b Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 17:28:32 -0700 Subject: [PATCH 258/305] tests : -g is part of DEBUGFLAGS --- tests/Makefile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tests/Makefile b/tests/Makefile index f5260e50e..ea58c0fe5 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -26,12 +26,12 @@ PYTHON ?= python3 TESTARTEFACT := versionsTest namespaceTest -DEBUGFLAG= -DZSTD_DEBUG=1 +DEBUGFLAGS=-g -DZSTD_DEBUG=1 CPPFLAGS+= -I$(ZSTDDIR) -I$(ZSTDDIR)/common -I$(ZSTDDIR)/compress \ -I$(ZSTDDIR)/dictBuilder -I$(ZSTDDIR)/deprecated -I$(PRGDIR) \ $(DEBUGFLAG) CFLAGS ?= -O3 -CFLAGS += -g -Wall -Wextra -Wcast-qual -Wcast-align -Wshadow \ +CFLAGS += -Wall -Wextra -Wcast-qual -Wcast-align -Wshadow \ -Wstrict-aliasing=1 -Wswitch-enum -Wdeclaration-after-statement \ -Wstrict-prototypes -Wundef -Wformat-security CFLAGS += $(MOREFLAGS) From 29297c6751289087d3d52688012dc776eccf1d2b Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 27 Apr 2017 17:44:01 -0700 Subject: [PATCH 259/305] Changed default level 18 (large input) Previous -18 : 4.7 MB/s, R:3.833 New -18 : 5.1 MB/s. R:3.825 It's a better fit within -17 (6.8 MB/s) and -19 (4.0 MB/s) The new level 18 also uses significantly less memory. And, it makes a good transition between level 17 (mml5) and level 19 (mml3). Up to now, there was no level with mml4. (note : minmatch setting can have a large impact on some (specific) datasets) --- lib/compress/zstd_compress.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index 9a2d194ad..74ae66cb7 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -3432,7 +3432,7 @@ static const ZSTD_compressionParameters ZSTD_defaultCParameters[4][ZSTD_MAX_CLEV { 22, 21, 21, 5, 5, 16, ZSTD_btlazy2 }, /* level 15 */ { 23, 22, 22, 5, 5, 16, ZSTD_btlazy2 }, /* level 16 */ { 23, 21, 22, 4, 5, 24, ZSTD_btopt }, /* level 17 */ - { 23, 23, 22, 6, 5, 32, ZSTD_btopt }, /* level 18 */ + { 23, 22, 22, 5, 4, 32, ZSTD_btopt }, /* level 18 */ { 23, 23, 22, 6, 3, 48, ZSTD_btopt }, /* level 19 */ { 25, 25, 23, 7, 3, 64, ZSTD_btopt2 }, /* level 20 */ { 26, 26, 23, 7, 3,256, ZSTD_btopt2 }, /* level 21 */ From 7bd68e194d84c09ff2b86f9068a26d0247fee821 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 28 Apr 2017 08:54:13 -0700 Subject: [PATCH 260/305] minor release note update --- NEWS | 1 + 1 file changed, 1 insertion(+) diff --git a/NEWS b/NEWS index 4f9bd2eff..2a9921e8e 100644 --- a/NEWS +++ b/NEWS @@ -2,6 +2,7 @@ v1.2.0 cli : changed : Multithreading enabled by default (use target zstd-nomt or HAVE_THREAD=0 to disable) cli : new : command -T0 means "detect and use nb of cores", by Sean Purcell cli : new : zstdmt symlink hardwired to `zstd -T0` +cli : new : command --threads=# (#671) cli : new : experimental target `zstd4`, with support for lz4 format, by Sean Purcell cli : fix : does not output compressed data on console cli : fix : ignore symbolic links unless --force specified, From 470993c9b11dff9fcf1f457b297edbbeef00565f Mon Sep 17 00:00:00 2001 From: Sean Purcell Date: Fri, 28 Apr 2017 12:17:09 -0700 Subject: [PATCH 261/305] Add raw seek table construction API and parallel compression example --- contrib/seekable_format/examples/.gitignore | 1 + contrib/seekable_format/examples/Makefile | 8 +- .../examples/parallel_compression.c | 214 ++++++++++++++++++ .../examples/parallel_processing.c | 2 - contrib/seekable_format/zstd_seekable.h | 23 ++ contrib/seekable_format/zstdseek_compress.c | 185 +++++++++------ 6 files changed, 358 insertions(+), 75 deletions(-) create mode 100644 contrib/seekable_format/examples/parallel_compression.c diff --git a/contrib/seekable_format/examples/.gitignore b/contrib/seekable_format/examples/.gitignore index 1e1661d80..df2f9ab07 100644 --- a/contrib/seekable_format/examples/.gitignore +++ b/contrib/seekable_format/examples/.gitignore @@ -1,3 +1,4 @@ seekable_compression seekable_decompression parallel_processing +parallel_compression diff --git a/contrib/seekable_format/examples/Makefile b/contrib/seekable_format/examples/Makefile index a77143889..625e1fcc8 100644 --- a/contrib/seekable_format/examples/Makefile +++ b/contrib/seekable_format/examples/Makefile @@ -9,7 +9,7 @@ # This Makefile presumes libzstd is built, using `make` in / or /lib/ -LDFLAGS += -L../../../lib/ -lzstd +LDFLAGS += ../../../lib/libzstd.a CPPFLAGS += -I../ -I../../../lib -I../../../lib/common CFLAGS ?= -O3 @@ -32,7 +32,11 @@ seekable_decompression : seekable_decompression.c $(SEEKABLE_OBJS) parallel_processing : parallel_processing.c $(SEEKABLE_OBJS) $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ -pthread +parallel_compression : parallel_compression.c $(SEEKABLE_OBJS) + $(CC) $(CPPFLAGS) $(CFLAGS) $^ $(LDFLAGS) -o $@ -pthread + clean: @rm -f core *.o tmp* result* *.zst \ - seekable_compression seekable_decompression parallel_processing + seekable_compression seekable_decompression \ + parallel_processing parallel_compression @echo Cleaning completed diff --git a/contrib/seekable_format/examples/parallel_compression.c b/contrib/seekable_format/examples/parallel_compression.c new file mode 100644 index 000000000..89a13185f --- /dev/null +++ b/contrib/seekable_format/examples/parallel_compression.c @@ -0,0 +1,214 @@ +/** + * Copyright 2017-present, Facebook, Inc. + * All rights reserved. + * + * This source code is licensed under the license found in the + * LICENSE-examples file in the root directory of this source tree. + */ + +#include // malloc, free, exit, atoi +#include // fprintf, perror, feof, fopen, etc. +#include // strlen, memset, strcat +#define ZSTD_STATIC_LINKING_ONLY +#include // presumes zstd library is installed +#include +#if defined(WIN32) || defined(_WIN32) +# include +# define SLEEP(x) Sleep(x) +#else +# include +# define SLEEP(x) usleep(x * 1000) +#endif + +#define XXH_NAMESPACE ZSTD_ +#include "xxhash.h" + +#include "pool.h" // use zstd thread pool for demo + +#include "zstd_seekable.h" + +static void* malloc_orDie(size_t size) +{ + void* const buff = malloc(size); + if (buff) return buff; + /* error */ + perror("malloc:"); + exit(1); +} + +static FILE* fopen_orDie(const char *filename, const char *instruction) +{ + FILE* const inFile = fopen(filename, instruction); + if (inFile) return inFile; + /* error */ + perror(filename); + exit(3); +} + +static size_t fread_orDie(void* buffer, size_t sizeToRead, FILE* file) +{ + size_t const readSize = fread(buffer, 1, sizeToRead, file); + if (readSize == sizeToRead) return readSize; /* good */ + if (feof(file)) return readSize; /* good, reached end of file */ + /* error */ + perror("fread"); + exit(4); +} + +static size_t fwrite_orDie(const void* buffer, size_t sizeToWrite, FILE* file) +{ + size_t const writtenSize = fwrite(buffer, 1, sizeToWrite, file); + if (writtenSize == sizeToWrite) return sizeToWrite; /* good */ + /* error */ + perror("fwrite"); + exit(5); +} + +static size_t fclose_orDie(FILE* file) +{ + if (!fclose(file)) return 0; + /* error */ + perror("fclose"); + exit(6); +} + +static void fseek_orDie(FILE* file, long int offset, int origin) +{ + if (!fseek(file, offset, origin)) { + if (!fflush(file)) return; + } + /* error */ + perror("fseek"); + exit(7); +} + +static long int ftell_orDie(FILE* file) +{ + long int off = ftell(file); + if (off != -1) return off; + /* error */ + perror("ftell"); + exit(8); +} + +struct job { + const void* src; + size_t srcSize; + void* dst; + size_t dstSize; + + unsigned checksum; + + int compressionLevel; + int done; +}; + +static void compressFrame(void* opaque) +{ + struct job* job = opaque; + + job->checksum = XXH64(job->src, job->srcSize, 0); + + size_t ret = ZSTD_compress(job->dst, job->dstSize, job->src, job->srcSize, job->compressionLevel); + if (ZSTD_isError(ret)) { + fprintf(stderr, "ZSTD_compress() error : %s \n", ZSTD_getErrorName(ret)); + exit(20); + } + + job->dstSize = ret; + job->done = 1; +} + +static void compressFile_orDie(const char* fname, const char* outName, int cLevel, unsigned frameSize, int nbThreads) +{ + POOL_ctx* pool = POOL_create(nbThreads, nbThreads); + if (pool == NULL) { fprintf(stderr, "POOL_create() error \n"); exit(9); } + + FILE* const fin = fopen_orDie(fname, "rb"); + FILE* const fout = fopen_orDie(outName, "wb"); + + if (ZSTD_compressBound(frameSize) > 0xFFFFFFFFU) { fprintf(stderr, "Frame size too large \n"); exit(10); } + unsigned dstSize = ZSTD_compressBound(frameSize); + + + fseek_orDie(fin, 0, SEEK_END); + long int length = ftell_orDie(fin); + fseek_orDie(fin, 0, SEEK_SET); + + size_t numFrames = (length + frameSize - 1) / frameSize; + + struct job* jobs = malloc_orDie(sizeof(struct job) * numFrames); + + size_t i; + for(i = 0; i < numFrames; i++) { + void* in = malloc_orDie(frameSize); + void* out = malloc_orDie(dstSize); + + size_t inSize = fread_orDie(in, frameSize, fin); + + jobs[i].src = in; + jobs[i].srcSize = inSize; + jobs[i].dst = out; + jobs[i].dstSize = dstSize; + jobs[i].compressionLevel = cLevel; + jobs[i].done = 0; + POOL_add(pool, compressFrame, &jobs[i]); + } + + ZSTD_frameLog* fl = ZSTD_seekable_createFrameLog(1); + if (fl == NULL) { fprintf(stderr, "ZSTD_seekable_createFrameLog() failed \n"); exit(11); } + for (i = 0; i < numFrames; i++) { + while (!jobs[i].done) SLEEP(5); /* wake up every 5 milliseconds to check */ + fwrite_orDie(jobs[i].dst, jobs[i].dstSize, fout); + free((void*)jobs[i].src); + free(jobs[i].dst); + + size_t ret = ZSTD_seekable_logFrame(fl, jobs[i].dstSize, jobs[i].srcSize, jobs[i].checksum); + if (ZSTD_isError(ret)) { fprintf(stderr, "ZSTD_seekable_logFrame() error : %s \n", ZSTD_getErrorName(ret)); } + } + + { unsigned char seekTableBuff[1024]; + ZSTD_outBuffer out = {seekTableBuff, 1024, 0}; + while (ZSTD_seekable_writeSeekTable(fl, &out) != 0) { + fwrite_orDie(seekTableBuff, out.pos, fout); + out.pos = 0; + } + fwrite_orDie(seekTableBuff, out.pos, fout); + } + + ZSTD_seekable_freeFrameLog(fl); + free(jobs); + fclose_orDie(fout); + fclose_orDie(fin); +} + +static const char* createOutFilename_orDie(const char* filename) +{ + size_t const inL = strlen(filename); + size_t const outL = inL + 5; + void* outSpace = malloc_orDie(outL); + memset(outSpace, 0, outL); + strcat(outSpace, filename); + strcat(outSpace, ".zst"); + return (const char*)outSpace; +} + +int main(int argc, const char** argv) { + const char* const exeName = argv[0]; + if (argc!=4) { + printf("wrong arguments\n"); + printf("usage:\n"); + printf("%s FILE FRAME_SIZE NB_THREADS\n", exeName); + return 1; + } + + { const char* const inFileName = argv[1]; + unsigned const frameSize = (unsigned)atoi(argv[2]); + int const nbThreads = atoi(argv[3]); + + const char* const outFileName = createOutFilename_orDie(inFileName); + compressFile_orDie(inFileName, outFileName, 5, frameSize, nbThreads); + } + + return 0; +} diff --git a/contrib/seekable_format/examples/parallel_processing.c b/contrib/seekable_format/examples/parallel_processing.c index cf2d0d2af..cea4d5364 100644 --- a/contrib/seekable_format/examples/parallel_processing.c +++ b/contrib/seekable_format/examples/parallel_processing.c @@ -95,8 +95,6 @@ static void fseek_orDie(FILE* file, long int offset, int origin) { exit(7); } -static const char* filename; - struct sum_job { const char* fname; unsigned long long sum; diff --git a/contrib/seekable_format/zstd_seekable.h b/contrib/seekable_format/zstd_seekable.h index 71b4c0be7..438ac2014 100644 --- a/contrib/seekable_format/zstd_seekable.h +++ b/contrib/seekable_format/zstd_seekable.h @@ -82,6 +82,29 @@ ZSTDLIB_API size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD ZSTDLIB_API size_t ZSTD_seekable_endFrame(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); ZSTDLIB_API size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output); +/*= Raw seek table API + * These functions allow for the seek table to be constructed directly. + * This table can then be appended to a file of concatenated frames. + * This allows the frames to be compressed independently, even in parallel, + * and compiled together afterward into a seekable archive. + * + * Use ZSTD_seekable_createFrameLog() to allocate and initialize a tracking + * structure. + * + * Call ZSTD_seekable_logFrame() once for each frame in the archive. + * checksum is optional, and will not be used if checksumFlag was 0 when the + * frame log was created. If present, it should be the least significant 32 + * bits of the XXH64 hash of the uncompressed data. + * + * Call ZSTD_seekable_writeSeekTable to serialize the data into a seek table. + * If the entire table was written, the return value will be 0. Otherwise, + * it will be equal to the number of bytes left to write. */ +typedef struct ZSTD_frameLog_s ZSTD_frameLog; +ZSTDLIB_API ZSTD_frameLog* ZSTD_seekable_createFrameLog(int checksumFlag); +ZSTDLIB_API size_t ZSTD_seekable_freeFrameLog(ZSTD_frameLog* fl); +ZSTDLIB_API size_t ZSTD_seekable_logFrame(ZSTD_frameLog* fl, unsigned compressedSize, unsigned decompressedSize, unsigned checksum); +ZSTDLIB_API size_t ZSTD_seekable_writeSeekTable(ZSTD_frameLog* fl, ZSTD_outBuffer* output); + /*-**************************************************************************** * Seekable decompression - HowTo * A ZSTD_seekable object is required to tracking the seekTable. diff --git a/contrib/seekable_format/zstdseek_compress.c b/contrib/seekable_format/zstdseek_compress.c index 2d1698bee..5fe26ed28 100644 --- a/contrib/seekable_format/zstdseek_compress.c +++ b/contrib/seekable_format/zstdseek_compress.c @@ -35,15 +35,21 @@ typedef struct { U32 checksum; } framelogEntry_t; -typedef struct { +struct ZSTD_frameLog_s { framelogEntry_t* entries; U32 size; U32 capacity; + + int checksumFlag; + + /* for use when streaming out the seek table */ + U32 seekTablePos; + U32 seekTableIndex; } framelog_t; struct ZSTD_seekable_CStream_s { ZSTD_CStream* cstream; - framelog_t framelog; + ZSTD_frameLog framelog; U32 frameCSize; U32 frameDSize; @@ -52,13 +58,52 @@ struct ZSTD_seekable_CStream_s { U32 maxFrameSize; - int checksumFlag; - int writingSeekTable; - U32 seekTablePos; - U32 seekTableIndex; }; +size_t ZSTD_seekable_frameLog_allocVec(ZSTD_frameLog* fl) +{ + /* allocate some initial space */ + size_t const FRAMELOG_STARTING_CAPACITY = 16; + fl->entries = (framelogEntry_t*)malloc( + sizeof(framelogEntry_t) * FRAMELOG_STARTING_CAPACITY); + if (fl->entries == NULL) return ERROR(memory_allocation); + fl->capacity = FRAMELOG_STARTING_CAPACITY; + + return 0; +} + +size_t ZSTD_seekable_frameLog_freeVec(ZSTD_frameLog* fl) +{ + if (fl != NULL) free(fl->entries); + return 0; +} + +ZSTD_frameLog* ZSTD_seekable_createFrameLog(int checksumFlag) +{ + ZSTD_frameLog* fl = malloc(sizeof(ZSTD_frameLog)); + if (fl == NULL) return NULL; + + if (ZSTD_isError(ZSTD_seekable_frameLog_allocVec(fl))) { + free(fl); + return NULL; + } + + fl->checksumFlag = checksumFlag; + fl->seekTablePos = 0; + fl->seekTableIndex = 0; + fl->size = 0; + + return fl; +} + +size_t ZSTD_seekable_freeFrameLog(ZSTD_frameLog* fl) +{ + ZSTD_seekable_frameLog_freeVec(fl); + free(fl); + return 0; +} + ZSTD_seekable_CStream* ZSTD_seekable_createCStream() { ZSTD_seekable_CStream* zcs = malloc(sizeof(ZSTD_seekable_CStream)); @@ -70,13 +115,7 @@ ZSTD_seekable_CStream* ZSTD_seekable_createCStream() zcs->cstream = ZSTD_createCStream(); if (zcs->cstream == NULL) goto failed1; - /* allocate some initial space */ - { size_t const FRAMELOG_STARTING_CAPACITY = 16; - zcs->framelog.entries = (framelogEntry_t*)malloc( - sizeof(framelogEntry_t) * FRAMELOG_STARTING_CAPACITY); - if (zcs->framelog.entries == NULL) goto failed2; - zcs->framelog.capacity = FRAMELOG_STARTING_CAPACITY; - } + if (ZSTD_isError(ZSTD_seekable_frameLog_allocVec(&zcs->framelog))) goto failed2; return zcs; @@ -91,7 +130,7 @@ size_t ZSTD_seekable_freeCStream(ZSTD_seekable_CStream* zcs) { if (zcs == NULL) return 0; /* support free on null */ ZSTD_freeCStream(zcs->cstream); - free(zcs->framelog.entries); + ZSTD_seekable_frameLog_freeVec(&zcs->framelog); free(zcs); return 0; @@ -115,44 +154,43 @@ size_t ZSTD_seekable_initCStream(ZSTD_seekable_CStream* zcs, ? maxFrameSize : ZSTD_SEEKABLE_MAX_FRAME_DECOMPRESSED_SIZE; - zcs->checksumFlag = checksumFlag; - if (zcs->checksumFlag) { + zcs->framelog.checksumFlag = checksumFlag; + if (zcs->framelog.checksumFlag) { XXH64_reset(&zcs->xxhState, 0); } - zcs->seekTablePos = 0; - zcs->seekTableIndex = 0; + zcs->framelog.seekTablePos = 0; + zcs->framelog.seekTableIndex = 0; zcs->writingSeekTable = 0; return ZSTD_initCStream(zcs->cstream, compressionLevel); } -static size_t ZSTD_seekable_logFrame(ZSTD_seekable_CStream* zcs) +size_t ZSTD_seekable_logFrame(ZSTD_frameLog* fl, + unsigned compressedSize, + unsigned decompressedSize, + unsigned checksum) { - if (zcs->framelog.size == ZSTD_SEEKABLE_MAXFRAMES) + if (fl->size == ZSTD_SEEKABLE_MAXFRAMES) return ERROR(frameIndex_tooLarge); /* grow the buffer if required */ - if (zcs->framelog.size == zcs->framelog.capacity) { + if (fl->size == fl->capacity) { /* exponential size increase for constant amortized runtime */ - size_t const newCapacity = zcs->framelog.capacity * 2; - framelogEntry_t* const newEntries = realloc(zcs->framelog.entries, + size_t const newCapacity = fl->capacity * 2; + framelogEntry_t* const newEntries = realloc(fl->entries, sizeof(framelogEntry_t) * newCapacity); if (newEntries == NULL) return ERROR(memory_allocation); - zcs->framelog.entries = newEntries; - zcs->framelog.capacity = newCapacity; + fl->entries = newEntries; + fl->capacity = newCapacity; } - zcs->framelog.entries[zcs->framelog.size] = (framelogEntry_t){ - zcs->frameCSize, zcs->frameDSize, + fl->entries[fl->size] = (framelogEntry_t){ + compressedSize, decompressedSize, checksum }; - if (zcs->checksumFlag) - zcs->framelog.entries[zcs->framelog.size].checksum = - XXH64_digest(&zcs->xxhState) & 0xFFFFFFFFU; /* take lower 32 bits of digest */ - - zcs->framelog.size++; + fl->size++; return 0; } @@ -171,7 +209,11 @@ size_t ZSTD_seekable_endFrame(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output /* frame done */ /* store the frame data for later */ - ret = ZSTD_seekable_logFrame(zcs); + ret = ZSTD_seekable_logFrame( + &zcs->framelog, zcs->frameCSize, zcs->frameDSize, + zcs->framelog.checksumFlag + ? XXH64_digest(&zcs->xxhState) & 0xFFFFFFFFU + : 0); if (ret) return ret; /* reset for the next frame */ @@ -179,7 +221,7 @@ size_t ZSTD_seekable_endFrame(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output zcs->frameDSize = 0; ZSTD_resetCStream(zcs->cstream, 0); - if (zcs->checksumFlag) + if (zcs->framelog.checksumFlag) XXH64_reset(&zcs->xxhState, 0); return 0; @@ -199,7 +241,7 @@ size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* size_t const ret = ZSTD_compressStream(zcs->cstream, output, &inTmp); - if (zcs->checksumFlag) { + if (zcs->framelog.checksumFlag) { XXH64_update(&zcs->xxhState, inBase, inTmp.pos); } @@ -223,36 +265,36 @@ size_t ZSTD_seekable_compressStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* return (size_t)(zcs->maxFrameSize - zcs->frameDSize); } -static inline size_t ZSTD_seekable_seekTableSize(ZSTD_seekable_CStream const* zcs) +static inline size_t ZSTD_seekable_seekTableSize(const ZSTD_frameLog* fl) { - size_t const sizePerFrame = 8 + (zcs->checksumFlag?4:0); + size_t const sizePerFrame = 8 + (fl->checksumFlag?4:0); size_t const seekTableLen = ZSTD_skippableHeaderSize + - sizePerFrame * zcs->framelog.size + + sizePerFrame * fl->size + ZSTD_seekTableFooterSize; return seekTableLen; } -static inline size_t ZSTD_stwrite32(ZSTD_seekable_CStream* zcs, +static inline size_t ZSTD_stwrite32(ZSTD_frameLog* fl, ZSTD_outBuffer* output, U32 const value, U32 const offset) { - if (zcs->seekTablePos < offset + 4) { + if (fl->seekTablePos < offset + 4) { BYTE tmp[4]; /* so that we can work with buffers too small to write a whole word to */ size_t const lenWrite = - MIN(output->size - output->pos, offset + 4 - zcs->seekTablePos); + MIN(output->size - output->pos, offset + 4 - fl->seekTablePos); MEM_writeLE32(tmp, value); memcpy((BYTE*)output->dst + output->pos, - tmp + (zcs->seekTablePos - offset), lenWrite); + tmp + (fl->seekTablePos - offset), lenWrite); output->pos += lenWrite; - zcs->seekTablePos += lenWrite; + fl->seekTablePos += lenWrite; - if (lenWrite < 4) return ZSTD_seekable_seekTableSize(zcs) - zcs->seekTablePos; + if (lenWrite < 4) return ZSTD_seekable_seekTableSize(fl) - fl->seekTablePos; } return 0; } -static size_t ZSTD_seekable_writeSeekTable(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* output) +size_t ZSTD_seekable_writeSeekTable(ZSTD_frameLog* fl, ZSTD_outBuffer* output) { /* seekTableIndex: the current index in the table and * seekTableSize: the amount of the table written so far @@ -261,50 +303,51 @@ static size_t ZSTD_seekable_writeSeekTable(ZSTD_seekable_CStream* zcs, ZSTD_outB * because of a small buffer, it can keep going where it left off. */ - size_t const sizePerFrame = 8 + (zcs->checksumFlag?4:0); - size_t const seekTableLen = ZSTD_seekable_seekTableSize(zcs); + size_t const sizePerFrame = 8 + (fl->checksumFlag?4:0); + size_t const seekTableLen = ZSTD_seekable_seekTableSize(fl); - CHECK_Z(ZSTD_stwrite32(zcs, output, ZSTD_MAGIC_SKIPPABLE_START | 0xE, 0)); - CHECK_Z(ZSTD_stwrite32(zcs, output, seekTableLen - ZSTD_skippableHeaderSize, + CHECK_Z(ZSTD_stwrite32(fl, output, ZSTD_MAGIC_SKIPPABLE_START | 0xE, 0)); + CHECK_Z(ZSTD_stwrite32(fl, output, seekTableLen - ZSTD_skippableHeaderSize, 4)); - while (zcs->seekTableIndex < zcs->framelog.size) { - CHECK_Z(ZSTD_stwrite32( - zcs, output, zcs->framelog.entries[zcs->seekTableIndex].cSize, - ZSTD_skippableHeaderSize + sizePerFrame * zcs->seekTableIndex)); + while (fl->seekTableIndex < fl->size) { + CHECK_Z(ZSTD_stwrite32(fl, output, + fl->entries[fl->seekTableIndex].cSize, + ZSTD_skippableHeaderSize + + sizePerFrame * fl->seekTableIndex + 0)); - CHECK_Z(ZSTD_stwrite32( - zcs, output, zcs->framelog.entries[zcs->seekTableIndex].dSize, - ZSTD_skippableHeaderSize + sizePerFrame * zcs->seekTableIndex + 4)); + CHECK_Z(ZSTD_stwrite32(fl, output, + fl->entries[fl->seekTableIndex].dSize, + ZSTD_skippableHeaderSize + + sizePerFrame * fl->seekTableIndex + 4)); - if (zcs->checksumFlag) { + if (fl->checksumFlag) { CHECK_Z(ZSTD_stwrite32( - zcs, output, - zcs->framelog.entries[zcs->seekTableIndex].checksum, - ZSTD_skippableHeaderSize + sizePerFrame * zcs->seekTableIndex + - 8)); + fl, output, fl->entries[fl->seekTableIndex].checksum, + ZSTD_skippableHeaderSize + + sizePerFrame * fl->seekTableIndex + 8)); } - zcs->seekTableIndex++; + fl->seekTableIndex++; } - CHECK_Z(ZSTD_stwrite32(zcs, output, zcs->framelog.size, + CHECK_Z(ZSTD_stwrite32(fl, output, fl->size, seekTableLen - ZSTD_seekTableFooterSize)); - if (output->size - output->pos < 1) return seekTableLen - zcs->seekTablePos; - if (zcs->seekTablePos < seekTableLen - 4) { + if (output->size - output->pos < 1) return seekTableLen - fl->seekTablePos; + if (fl->seekTablePos < seekTableLen - 4) { BYTE sfd = 0; - sfd |= (zcs->checksumFlag) << 7; + sfd |= (fl->checksumFlag) << 7; ((BYTE*)output->dst)[output->pos] = sfd; output->pos++; - zcs->seekTablePos++; + fl->seekTablePos++; } - CHECK_Z(ZSTD_stwrite32(zcs, output, ZSTD_SEEKABLE_MAGICNUMBER, + CHECK_Z(ZSTD_stwrite32(fl, output, ZSTD_SEEKABLE_MAGICNUMBER, seekTableLen - 4)); - if (zcs->seekTablePos != seekTableLen) return ERROR(GENERIC); + if (fl->seekTablePos != seekTableLen) return ERROR(GENERIC); return 0; } @@ -314,10 +357,10 @@ size_t ZSTD_seekable_endStream(ZSTD_seekable_CStream* zcs, ZSTD_outBuffer* outpu const size_t endFrame = ZSTD_seekable_endFrame(zcs, output); if (ZSTD_isError(endFrame)) return endFrame; /* return an accurate size hint */ - if (endFrame) return endFrame + ZSTD_seekable_seekTableSize(zcs); + if (endFrame) return endFrame + ZSTD_seekable_seekTableSize(&zcs->framelog); } zcs->writingSeekTable = 1; - return ZSTD_seekable_writeSeekTable(zcs, output); + return ZSTD_seekable_writeSeekTable(&zcs->framelog, output); } From 68a7d3d49a776954cb0d1b6f6c5ae86b35f0e029 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 28 Apr 2017 12:46:48 -0700 Subject: [PATCH 262/305] added HUF_PUBLIC_API macro to huf.h to make it possible to control symbol visibility. Also : better separation and comments between "public" and "static" sections --- NEWS | 4 ++-- lib/common/huf.h | 57 +++++++++++++++++++++++++++++++++--------------- 2 files changed, 41 insertions(+), 20 deletions(-) diff --git a/NEWS b/NEWS index 2a9921e8e..b910e2993 100644 --- a/NEWS +++ b/NEWS @@ -7,11 +7,11 @@ cli : new : experimental target `zstd4`, with support for lz4 format, by Sean Pu cli : fix : does not output compressed data on console cli : fix : ignore symbolic links unless --force specified, API : breaking change : ZSTD_createCDict_advanced(), only use compressionParameters as argument -API : changed : added prototypes ZSTD_*_usingCDict_advanced(), for direct control over frameParameters. -API : changed : enforced consistent rules for pledgedSrcSize==0 +API : added : prototypes ZSTD_*_usingCDict_advanced(), for direct control over frameParameters. API : improved: ZSTDMT_compressCCtx() reduced memory usage API : fix : ZSTDMT_compressCCtx() now provides srcSize in header (#634) API : fix : src size stored in frame header is controlled at end of frame +API : fix : enforced consistent rules for pledgedSrcSize==0 API : fix : ensure error code "dstSizeTooSmall" is generated when dst size is too small build: improved cmake script, by @Majlen build: enabled Multi-threading support for *BSD, by Baptiste Daroussin diff --git a/lib/common/huf.h b/lib/common/huf.h index 32313cf74..7873ca3d4 100644 --- a/lib/common/huf.h +++ b/lib/common/huf.h @@ -43,6 +43,21 @@ extern "C" { #include /* size_t */ +/* *** library symbols visibility *** */ +/* Note : when linking with -fvisibility=hidden on gcc, or by default on Visual, + * HUF symbols remain "private" (internal symbols for library only). + * Set macro FSE_DLL_EXPORT to 1 if you want HUF symbols visible on DLL interface */ +#if defined(FSE_DLL_EXPORT) && (FSE_DLL_EXPORT==1) && defined(__GNUC__) && (__GNUC__ >= 4) +# define HUF_PUBLIC_API __attribute__ ((visibility ("default"))) +#elif defined(FSE_DLL_EXPORT) && (FSE_DLL_EXPORT==1) /* Visual expected */ +# define HUF_PUBLIC_API __declspec(dllexport) +#elif defined(FSE_DLL_IMPORT) && (FSE_DLL_IMPORT==1) +# define HUF_PUBLIC_API __declspec(dllimport) /* not required, just to generate faster code (saves a function pointer load from IAT and an indirect jump) */ +#else +# define HUF_PUBLIC_API +#endif + + /* *** simple functions *** */ /** HUF_compress() : @@ -55,8 +70,8 @@ HUF_compress() : if return == 1, srcData is a single repeated byte symbol (RLE compression). if HUF_isError(return), compression failed (more details using HUF_getErrorName()) */ -size_t HUF_compress(void* dst, size_t dstCapacity, - const void* src, size_t srcSize); +HUF_PUBLIC_API size_t HUF_compress(void* dst, size_t dstCapacity, + const void* src, size_t srcSize); /** HUF_decompress() : @@ -69,32 +84,42 @@ HUF_decompress() : @return : size of regenerated data (== originalSize), or an error code, which can be tested using HUF_isError() */ -size_t HUF_decompress(void* dst, size_t originalSize, - const void* cSrc, size_t cSrcSize); +HUF_PUBLIC_API size_t HUF_decompress(void* dst, size_t originalSize, + const void* cSrc, size_t cSrcSize); /* *** Tool functions *** */ -#define HUF_BLOCKSIZE_MAX (128 * 1024) /**< maximum input size for a single block compressed with HUF_compress */ -size_t HUF_compressBound(size_t size); /**< maximum compressed size (worst case) */ +#define HUF_BLOCKSIZE_MAX (128 * 1024) /**< maximum input size for a single block compressed with HUF_compress */ +HUF_PUBLIC_API size_t HUF_compressBound(size_t size); /**< maximum compressed size (worst case) */ /* Error Management */ -unsigned HUF_isError(size_t code); /**< tells if a return value is an error code */ -const char* HUF_getErrorName(size_t code); /**< provides error code string (useful for debugging) */ +HUF_PUBLIC_API unsigned HUF_isError(size_t code); /**< tells if a return value is an error code */ +HUF_PUBLIC_API const char* HUF_getErrorName(size_t code); /**< provides error code string (useful for debugging) */ /* *** Advanced function *** */ /** HUF_compress2() : - * Same as HUF_compress(), but offers direct control over `maxSymbolValue` and `tableLog` . - * `tableLog` must be `<= HUF_TABLELOG_MAX` . */ -size_t HUF_compress2 (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog); + * Same as HUF_compress(), but offers direct control over `maxSymbolValue` and `tableLog`. + * `tableLog` must be `<= HUF_TABLELOG_MAX` . */ +HUF_PUBLIC_API size_t HUF_compress2 (void* dst, size_t dstCapacity, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog); /** HUF_compress4X_wksp() : -* Same as HUF_compress2(), but uses externally allocated `workSpace`, which must be a table of >= 1024 unsigned */ -size_t HUF_compress4X_wksp (void* dst, size_t dstSize, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize); /**< `workSpace` must be a table of at least HUF_WORKSPACE_SIZE_U32 unsigned */ + * Same as HUF_compress2(), but uses externally allocated `workSpace`. + * `workspace` must have minimum alignment of 4, and be at least as large as following macro */ +#define HUF_WORKSPACE_SIZE (6 << 10) +#define HUF_WORKSPACE_SIZE_U32 (HUF_WORKSPACE_SIZE / sizeof(U32)) +HUF_PUBLIC_API size_t HUF_compress4X_wksp (void* dst, size_t dstCapacity, const void* src, size_t srcSize, unsigned maxSymbolValue, unsigned tableLog, void* workSpace, size_t wkspSize); +/* ****************************************************************** + * WARNING !! + * The following section contains advanced and experimental definitions + * which shall never be used in the context of dll + * because they are not guaranteed to remain stable in the future. + * Only consider them in association with static linking. + *******************************************************************/ #ifdef HUF_STATIC_LINKING_ONLY /* *** Dependencies *** */ @@ -117,7 +142,7 @@ size_t HUF_compress4X_wksp (void* dst, size_t dstSize, const void* src, size_t s ******************************************/ /* HUF buffer bounds */ #define HUF_CTABLEBOUND 129 -#define HUF_BLOCKBOUND(size) (size + (size>>8) + 8) /* only true if incompressible pre-filtered with fast heuristic */ +#define HUF_BLOCKBOUND(size) (size + (size>>8) + 8) /* only true when incompressible is pre-filtered with fast heuristic */ #define HUF_COMPRESSBOUND(size) (HUF_CTABLEBOUND + HUF_BLOCKBOUND(size)) /* Macro version, useful for static allocation */ /* static allocation of HUF's Compression Table */ @@ -136,10 +161,6 @@ typedef U32 HUF_DTable; #define HUF_CREATE_STATIC_DTABLEX4(DTable, maxTableLog) \ HUF_DTable DTable[HUF_DTABLE_SIZE(maxTableLog)] = { ((U32)(maxTableLog) * 0x01000001) } -/* The workspace must have alignment at least 4 and be at least this large */ -#define HUF_WORKSPACE_SIZE (6 << 10) -#define HUF_WORKSPACE_SIZE_U32 (HUF_WORKSPACE_SIZE / sizeof(U32)) - /* **************************************** * Advanced decompression functions From 89f50deec7b37ff2c7c01cfa85ef487189ae7701 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 28 Apr 2017 16:52:36 -0700 Subject: [PATCH 263/305] minor code refactoring clearer tables --- lib/decompress/zstd_decompress.c | 252 ++++++++----------------------- 1 file changed, 62 insertions(+), 190 deletions(-) diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c index 7499b8ae1..634e2152e 100644 --- a/lib/decompress/zstd_decompress.c +++ b/lib/decompress/zstd_decompress.c @@ -177,30 +177,6 @@ void ZSTD_copyDCtx(ZSTD_DCtx* dstDCtx, const ZSTD_DCtx* srcDCtx) memcpy(dstDCtx, srcDCtx, sizeof(ZSTD_DCtx) - workSpaceSize); /* no need to copy workspace */ } -#if 0 -/* deprecated */ -static void ZSTD_refDCtx(ZSTD_DCtx* dstDCtx, const ZSTD_DCtx* srcDCtx) -{ - ZSTD_decompressBegin(dstDCtx); /* init */ - if (srcDCtx) { /* support refDCtx on NULL */ - dstDCtx->dictEnd = srcDCtx->dictEnd; - dstDCtx->vBase = srcDCtx->vBase; - dstDCtx->base = srcDCtx->base; - dstDCtx->previousDstEnd = srcDCtx->previousDstEnd; - dstDCtx->dictID = srcDCtx->dictID; - dstDCtx->litEntropy = srcDCtx->litEntropy; - dstDCtx->fseEntropy = srcDCtx->fseEntropy; - dstDCtx->LLTptr = srcDCtx->entropy.LLTable; - dstDCtx->MLTptr = srcDCtx->entropy.MLTable; - dstDCtx->OFTptr = srcDCtx->entropy.OFTable; - dstDCtx->HUFptr = srcDCtx->entropy.hufTable; - dstDCtx->entropy.rep[0] = srcDCtx->entropy.rep[0]; - dstDCtx->entropy.rep[1] = srcDCtx->entropy.rep[1]; - dstDCtx->entropy.rep[2] = srcDCtx->entropy.rep[2]; - } -} -#endif - static void ZSTD_refDDict(ZSTD_DCtx* dstDCtx, const ZSTD_DDict* ddict); @@ -431,7 +407,8 @@ typedef struct /*! ZSTD_getcBlockSize() : * Provides the size of compressed block from block header `src` */ -size_t ZSTD_getcBlockSize(const void* src, size_t srcSize, blockProperties_t* bpPtr) +size_t ZSTD_getcBlockSize(const void* src, size_t srcSize, + blockProperties_t* bpPtr) { if (srcSize < ZSTD_blockHeaderSize) return ERROR(srcSize_wrong); { U32 const cBlockHeader = MEM_readLE24(src); @@ -446,7 +423,8 @@ size_t ZSTD_getcBlockSize(const void* src, size_t srcSize, blockProperties_t* bp } -static size_t ZSTD_copyRawBlock(void* dst, size_t dstCapacity, const void* src, size_t srcSize) +static size_t ZSTD_copyRawBlock(void* dst, size_t dstCapacity, + const void* src, size_t srcSize) { if (srcSize > dstCapacity) return ERROR(dstSize_tooSmall); memcpy(dst, src, srcSize); @@ -454,7 +432,9 @@ static size_t ZSTD_copyRawBlock(void* dst, size_t dstCapacity, const void* src, } -static size_t ZSTD_setRleBlock(void* dst, size_t dstCapacity, const void* src, size_t srcSize, size_t regenSize) +static size_t ZSTD_setRleBlock(void* dst, size_t dstCapacity, + const void* src, size_t srcSize, + size_t regenSize) { if (srcSize != 1) return ERROR(srcSize_wrong); if (regenSize > dstCapacity) return ERROR(dstSize_tooSmall); @@ -595,176 +575,70 @@ typedef union { U32 alignedBy4; } FSE_decode_t4; +/* Default FSE distribution table for Literal Lengths */ static const FSE_decode_t4 LL_defaultDTable[(1< (size_t)(oLitEnd - base)) { - /* offset beyond prefix */ + /* offset beyond prefix -> go into extDict */ if (sequence.offset > (size_t)(oLitEnd - vBase)) return ERROR(corruption_detected); match = dictEnd + (match - base); if (match + sequence.matchLength <= dictEnd) { From 202082f285ed5a9d971b29a42f52ae5bf9995584 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 28 Apr 2017 16:56:39 -0700 Subject: [PATCH 264/305] sync bitstream from FSE project add assert into unsafe *_fast() variants --- doc/zstd_manual.html | 2 +- lib/common/bitstream.h | 17 +++++++++++++++-- 2 files changed, 16 insertions(+), 3 deletions(-) diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index 4755396e3..2e77e7742 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -499,7 +499,7 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v Note : this use case also happens when using a non-conformant dictionary. - `srcSize` is too small, and as a result, the frame header could not be decoded (only possible if `srcSize < ZSTD_FRAMEHEADERSIZE_MAX`). - This is not a Zstandard frame. - When identifying the exact failure cause, it's possible to used ZSTD_getFrameParams(), which will provide a more precise error code. + When identifying the exact failure cause, it's possible to use ZSTD_getFrameParams(), which will provide a more precise error code.


Advanced streaming functions


diff --git a/lib/common/bitstream.h b/lib/common/bitstream.h
index 0e3d2fc55..c7a764465 100644
--- a/lib/common/bitstream.h
+++ b/lib/common/bitstream.h
@@ -2,7 +2,7 @@
    bitstream
    Part of FSE library
    header file (to include)
-   Copyright (C) 2013-2016, Yann Collet.
+   Copyright (C) 2013-2017, Yann Collet.
 
    BSD 2-Clause License (http://www.opensource.org/licenses/bsd-license.php)
 
@@ -53,6 +53,16 @@ extern "C" {
 #include "error_private.h"  /* error codes and messages */
 
 
+/*-*************************************
+*  Debug
+***************************************/
+#if defined(BIT_DEBUG) && (BIT_DEBUG>=1)
+#  include 
+#else
+#  define assert(condition) ((void)0)
+#endif
+
+
 /*=========================================
 *  Target specific
 =========================================*/
@@ -209,6 +219,7 @@ MEM_STATIC void BIT_addBits(BIT_CStream_t* bitC, size_t value, unsigned nbBits)
  *  works only if `value` is _clean_, meaning all high bits above nbBits are 0 */
 MEM_STATIC void BIT_addBitsFast(BIT_CStream_t* bitC, size_t value, unsigned nbBits)
 {
+    assert((value>>nbBits) == 0);
     bitC->bitContainer |= value << bitC->bitPos;
     bitC->bitPos += nbBits;
 }
@@ -336,10 +347,11 @@ MEM_STATIC size_t BIT_getLowerBits(size_t bitContainer, U32 const nbBits)
 }
 
 /*! BIT_lookBitsFast() :
-*   unsafe version; only works only if nbBits >= 1 */
+ *  unsafe version; only works if nbBits >= 1 */
 MEM_STATIC size_t BIT_lookBitsFast(const BIT_DStream_t* bitD, U32 nbBits)
 {
     U32 const bitMask = sizeof(bitD->bitContainer)*8 - 1;
+    assert(nbBits >= 1);
     return (bitD->bitContainer << (bitD->bitsConsumed & bitMask)) >> (((bitMask+1)-nbBits) & bitMask);
 }
 
@@ -365,6 +377,7 @@ MEM_STATIC size_t BIT_readBits(BIT_DStream_t* bitD, U32 nbBits)
 MEM_STATIC size_t BIT_readBitsFast(BIT_DStream_t* bitD, U32 nbBits)
 {
     size_t const value = BIT_lookBitsFast(bitD, nbBits);
+    assert(nbBits >= 1);
     BIT_skipBits(bitD, nbBits);
     return value;
 }

From f39a6731ec0f2e83964ed3f732f5492fb7fe7c78 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Mon, 1 May 2017 09:56:03 -0700
Subject: [PATCH 265/305] sync bitstream.h from fse library

---
 lib/common/bitstream.h | 55 +++++++++++++++++++++++++++---------------
 1 file changed, 36 insertions(+), 19 deletions(-)

diff --git a/lib/common/bitstream.h b/lib/common/bitstream.h
index c7a764465..ad07b2aa9 100644
--- a/lib/common/bitstream.h
+++ b/lib/common/bitstream.h
@@ -84,7 +84,7 @@ extern "C" {
 typedef struct
 {
     size_t bitContainer;
-    int    bitPos;
+    unsigned bitPos;
     char*  startPtr;
     char*  ptr;
     char*  endPtr;
@@ -122,6 +122,7 @@ typedef struct
     unsigned bitsConsumed;
     const char* ptr;
     const char* start;
+    const char* limitPtr;
 } BIT_DStream_t;
 
 typedef enum { BIT_DStream_unfinished = 0,
@@ -173,7 +174,10 @@ MEM_STATIC unsigned BIT_highbit32 (register U32 val)
 #   elif defined(__GNUC__) && (__GNUC__ >= 3)   /* Use GCC Intrinsic */
     return 31 - __builtin_clz (val);
 #   else   /* Software version */
-    static const unsigned DeBruijnClz[32] = { 0, 9, 1, 10, 13, 21, 2, 29, 11, 14, 16, 18, 22, 25, 3, 30, 8, 12, 20, 28, 15, 17, 24, 7, 19, 27, 23, 6, 26, 5, 4, 31 };
+    static const unsigned DeBruijnClz[32] = { 0,  9,  1, 10, 13, 21,  2, 29,
+                                             11, 14, 16, 18, 22, 25,  3, 30,
+                                              8, 12, 20, 28, 15, 17, 24,  7,
+                                             19, 27, 23,  6, 26,  5,  4, 31 };
     U32 v = val;
     v |= v >> 1;
     v |= v >> 2;
@@ -185,31 +189,36 @@ MEM_STATIC unsigned BIT_highbit32 (register U32 val)
 }
 
 /*=====    Local Constants   =====*/
-static const unsigned BIT_mask[] = { 0, 1, 3, 7, 0xF, 0x1F, 0x3F, 0x7F, 0xFF, 0x1FF, 0x3FF, 0x7FF, 0xFFF, 0x1FFF, 0x3FFF, 0x7FFF, 0xFFFF, 0x1FFFF, 0x3FFFF, 0x7FFFF, 0xFFFFF, 0x1FFFFF, 0x3FFFFF, 0x7FFFFF,  0xFFFFFF, 0x1FFFFFF, 0x3FFFFFF };   /* up to 26 bits */
+static const unsigned BIT_mask[] = { 0, 1, 3, 7, 0xF, 0x1F, 0x3F, 0x7F,
+                                    0xFF, 0x1FF, 0x3FF, 0x7FF, 0xFFF, 0x1FFF, 0x3FFF, 0x7FFF,
+                                    0xFFFF, 0x1FFFF, 0x3FFFF, 0x7FFFF, 0xFFFFF, 0x1FFFFF, 0x3FFFFF, 0x7FFFFF,
+                                    0xFFFFFF, 0x1FFFFFF, 0x3FFFFFF };   /* up to 26 bits */
 
 
 /*-**************************************************************
 *  bitStream encoding
 ****************************************************************/
 /*! BIT_initCStream() :
- *  `dstCapacity` must be > sizeof(void*)
+ *  `dstCapacity` must be > sizeof(size_t)
  *  @return : 0 if success,
               otherwise an error code (can be tested using ERR_isError() ) */
-MEM_STATIC size_t BIT_initCStream(BIT_CStream_t* bitC, void* startPtr, size_t dstCapacity)
+MEM_STATIC size_t BIT_initCStream(BIT_CStream_t* bitC,
+                                  void* const startPtr, size_t const dstCapacity)
 {
     bitC->bitContainer = 0;
     bitC->bitPos = 0;
     bitC->startPtr = (char*)startPtr;
     bitC->ptr = bitC->startPtr;
-    bitC->endPtr = bitC->startPtr + dstCapacity - sizeof(bitC->ptr);
-    if (dstCapacity <= sizeof(bitC->ptr)) return ERROR(dstSize_tooSmall);
+    bitC->endPtr = bitC->startPtr + dstCapacity - sizeof(bitC->bitContainer);
+    if (dstCapacity <= sizeof(bitC->bitContainer)) return ERROR(dstSize_tooSmall);
     return 0;
 }
 
 /*! BIT_addBits() :
     can add up to 26 bits into `bitC`.
     Does not check for register overflow ! */
-MEM_STATIC void BIT_addBits(BIT_CStream_t* bitC, size_t value, unsigned nbBits)
+MEM_STATIC void BIT_addBits(BIT_CStream_t* bitC,
+                            size_t const value, unsigned const nbBits)
 {
     bitC->bitContainer |= (value & BIT_mask[nbBits]) << bitC->bitPos;
     bitC->bitPos += nbBits;
@@ -217,7 +226,8 @@ MEM_STATIC void BIT_addBits(BIT_CStream_t* bitC, size_t value, unsigned nbBits)
 
 /*! BIT_addBitsFast() :
  *  works only if `value` is _clean_, meaning all high bits above nbBits are 0 */
-MEM_STATIC void BIT_addBitsFast(BIT_CStream_t* bitC, size_t value, unsigned nbBits)
+MEM_STATIC void BIT_addBitsFast(BIT_CStream_t* bitC,
+                                size_t const value, unsigned const nbBits)
 {
     assert((value>>nbBits) == 0);
     bitC->bitContainer |= value << bitC->bitPos;
@@ -225,22 +235,27 @@ MEM_STATIC void BIT_addBitsFast(BIT_CStream_t* bitC, size_t value, unsigned nbBi
 }
 
 /*! BIT_flushBitsFast() :
+ *  assumption : bitContainer has not overflowed
  *  unsafe version; does not check buffer overflow */
 MEM_STATIC void BIT_flushBitsFast(BIT_CStream_t* bitC)
 {
     size_t const nbBytes = bitC->bitPos >> 3;
+    assert( bitC->bitPos <= (sizeof(bitC->bitContainer)*8) );
     MEM_writeLEST(bitC->ptr, bitC->bitContainer);
     bitC->ptr += nbBytes;
+    assert(bitC->ptr <= bitC->endPtr);
     bitC->bitPos &= 7;
-    bitC->bitContainer >>= nbBytes*8;   /* if bitPos >= sizeof(bitContainer)*8 --> undefined behavior */
+    bitC->bitContainer >>= nbBytes*8;
 }
 
 /*! BIT_flushBits() :
+ *  assumption : bitContainer has not overflowed
  *  safe version; check for buffer overflow, and prevents it.
  *  note : does not signal buffer overflow. This will be revealed later on using BIT_closeCStream() */
 MEM_STATIC void BIT_flushBits(BIT_CStream_t* bitC)
 {
     size_t const nbBytes = bitC->bitPos >> 3;
+    assert( bitC->bitPos <= (sizeof(bitC->bitContainer)*8) );
     MEM_writeLEST(bitC->ptr, bitC->bitContainer);
     bitC->ptr += nbBytes;
     if (bitC->ptr > bitC->endPtr) bitC->ptr = bitC->endPtr;
@@ -275,15 +290,16 @@ MEM_STATIC size_t BIT_initDStream(BIT_DStream_t* bitD, const void* srcBuffer, si
 {
     if (srcSize < 1) { memset(bitD, 0, sizeof(*bitD)); return ERROR(srcSize_wrong); }
 
+    bitD->start = (const char*)srcBuffer;
+    bitD->limitPtr = bitD->start + sizeof(bitD->bitContainer);
+
     if (srcSize >=  sizeof(bitD->bitContainer)) {  /* normal case */
-        bitD->start = (const char*)srcBuffer;
         bitD->ptr   = (const char*)srcBuffer + srcSize - sizeof(bitD->bitContainer);
         bitD->bitContainer = MEM_readLEST(bitD->ptr);
         { BYTE const lastByte = ((const BYTE*)srcBuffer)[srcSize-1];
           bitD->bitsConsumed = lastByte ? 8 - BIT_highbit32(lastByte) : 0;  /* ensures bitsConsumed is always set */
           if (lastByte == 0) return ERROR(GENERIC); /* endMark not present */ }
     } else {
-        bitD->start = (const char*)srcBuffer;
         bitD->ptr   = bitD->start;
         bitD->bitContainer = *(const BYTE*)(bitD->start);
         switch(srcSize)
@@ -341,8 +357,8 @@ MEM_STATIC size_t BIT_getLowerBits(size_t bitContainer, U32 const nbBits)
 #if defined(__BMI__) && defined(__GNUC__)   /* experimental; fails if bitD->bitsConsumed + nbBits > sizeof(bitD->bitContainer)*8 */
     return BIT_getMiddleBits(bitD->bitContainer, (sizeof(bitD->bitContainer)*8) - bitD->bitsConsumed - nbBits, nbBits);
 #else
-    U32 const bitMask = sizeof(bitD->bitContainer)*8 - 1;
-    return ((bitD->bitContainer << (bitD->bitsConsumed & bitMask)) >> 1) >> ((bitMask-nbBits) & bitMask);
+    U32 const regMask = sizeof(bitD->bitContainer)*8 - 1;
+    return ((bitD->bitContainer << (bitD->bitsConsumed & regMask)) >> 1) >> ((regMask-nbBits) & regMask);
 #endif
 }
 
@@ -350,9 +366,9 @@ MEM_STATIC size_t BIT_getLowerBits(size_t bitContainer, U32 const nbBits)
  *  unsafe version; only works if nbBits >= 1 */
 MEM_STATIC size_t BIT_lookBitsFast(const BIT_DStream_t* bitD, U32 nbBits)
 {
-    U32 const bitMask = sizeof(bitD->bitContainer)*8 - 1;
+    U32 const regMask = sizeof(bitD->bitContainer)*8 - 1;
     assert(nbBits >= 1);
-    return (bitD->bitContainer << (bitD->bitsConsumed & bitMask)) >> (((bitMask+1)-nbBits) & bitMask);
+    return (bitD->bitContainer << (bitD->bitsConsumed & regMask)) >> (((regMask+1)-nbBits) & regMask);
 }
 
 MEM_STATIC void BIT_skipBits(BIT_DStream_t* bitD, U32 nbBits)
@@ -389,10 +405,10 @@ MEM_STATIC size_t BIT_readBitsFast(BIT_DStream_t* bitD, U32 nbBits)
               if status == BIT_DStream_unfinished, internal register is filled with >= (sizeof(bitD->bitContainer)*8 - 7) bits */
 MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD)
 {
-    if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8))  /* should not happen => corruption detected */
+    if (bitD->bitsConsumed > (sizeof(bitD->bitContainer)*8))  /* overflow detected, like end of stream */
         return BIT_DStream_overflow;
 
-    if (bitD->ptr >= bitD->start + sizeof(bitD->bitContainer)) {
+    if (bitD->ptr >= bitD->limitPtr) {
         bitD->ptr -= bitD->bitsConsumed >> 3;
         bitD->bitsConsumed &= 7;
         bitD->bitContainer = MEM_readLEST(bitD->ptr);
@@ -402,6 +418,7 @@ MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD)
         if (bitD->bitsConsumed < sizeof(bitD->bitContainer)*8) return BIT_DStream_endOfBuffer;
         return BIT_DStream_completed;
     }
+    /* start < ptr < limitPtr */
     {   U32 nbBytes = bitD->bitsConsumed >> 3;
         BIT_DStream_status result = BIT_DStream_unfinished;
         if (bitD->ptr - nbBytes < bitD->start) {
@@ -410,7 +427,7 @@ MEM_STATIC BIT_DStream_status BIT_reloadDStream(BIT_DStream_t* bitD)
         }
         bitD->ptr -= nbBytes;
         bitD->bitsConsumed -= nbBytes*8;
-        bitD->bitContainer = MEM_readLEST(bitD->ptr);   /* reminder : srcSize > sizeof(bitD) */
+        bitD->bitContainer = MEM_readLEST(bitD->ptr);   /* reminder : srcSize > sizeof(bitD->bitContainer), otherwise bitD->ptr == bitD->start */
         return result;
     }
 }

From 33c38b092538717de110fb03cf313a5095537a8f Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Mon, 1 May 2017 11:12:30 -0700
Subject: [PATCH 266/305] fixed const in prototype, that Visual doesn't accept

---
 lib/common/bitstream.h | 15 +++++++--------
 1 file changed, 7 insertions(+), 8 deletions(-)

diff --git a/lib/common/bitstream.h b/lib/common/bitstream.h
index ad07b2aa9..ca42850df 100644
--- a/lib/common/bitstream.h
+++ b/lib/common/bitstream.h
@@ -203,7 +203,7 @@ static const unsigned BIT_mask[] = { 0, 1, 3, 7, 0xF, 0x1F, 0x3F, 0x7F,
  *  @return : 0 if success,
               otherwise an error code (can be tested using ERR_isError() ) */
 MEM_STATIC size_t BIT_initCStream(BIT_CStream_t* bitC,
-                                  void* const startPtr, size_t const dstCapacity)
+                                  void* startPtr, size_t dstCapacity)
 {
     bitC->bitContainer = 0;
     bitC->bitPos = 0;
@@ -218,7 +218,7 @@ MEM_STATIC size_t BIT_initCStream(BIT_CStream_t* bitC,
     can add up to 26 bits into `bitC`.
     Does not check for register overflow ! */
 MEM_STATIC void BIT_addBits(BIT_CStream_t* bitC,
-                            size_t const value, unsigned const nbBits)
+                            size_t value, unsigned nbBits)
 {
     bitC->bitContainer |= (value & BIT_mask[nbBits]) << bitC->bitPos;
     bitC->bitPos += nbBits;
@@ -227,7 +227,7 @@ MEM_STATIC void BIT_addBits(BIT_CStream_t* bitC,
 /*! BIT_addBitsFast() :
  *  works only if `value` is _clean_, meaning all high bits above nbBits are 0 */
 MEM_STATIC void BIT_addBitsFast(BIT_CStream_t* bitC,
-                                size_t const value, unsigned const nbBits)
+                                size_t value, unsigned nbBits)
 {
     assert((value>>nbBits) == 0);
     bitC->bitContainer |= value << bitC->bitPos;
@@ -251,7 +251,8 @@ MEM_STATIC void BIT_flushBitsFast(BIT_CStream_t* bitC)
 /*! BIT_flushBits() :
  *  assumption : bitContainer has not overflowed
  *  safe version; check for buffer overflow, and prevents it.
- *  note : does not signal buffer overflow. This will be revealed later on using BIT_closeCStream() */
+ *  note : does not signal buffer overflow.
+ *  overflow will be revealed later on using BIT_closeCStream() */
 MEM_STATIC void BIT_flushBits(BIT_CStream_t* bitC)
 {
     size_t const nbBytes = bitC->bitPos >> 3;
@@ -260,7 +261,7 @@ MEM_STATIC void BIT_flushBits(BIT_CStream_t* bitC)
     bitC->ptr += nbBytes;
     if (bitC->ptr > bitC->endPtr) bitC->ptr = bitC->endPtr;
     bitC->bitPos &= 7;
-    bitC->bitContainer >>= nbBytes*8;   /* if bitPos >= sizeof(bitContainer)*8 --> undefined behavior */
+    bitC->bitContainer >>= nbBytes*8;
 }
 
 /*! BIT_closeCStream() :
@@ -270,9 +271,7 @@ MEM_STATIC size_t BIT_closeCStream(BIT_CStream_t* bitC)
 {
     BIT_addBitsFast(bitC, 1, 1);   /* endMark */
     BIT_flushBits(bitC);
-
-    if (bitC->ptr >= bitC->endPtr) return 0; /* doesn't fit within authorized budget : cancel */
-
+    if (bitC->ptr >= bitC->endPtr) return 0; /* overflow detected */
     return (bitC->ptr - bitC->startPtr) + (bitC->bitPos > 0);
 }
 

From b184589c4c8d33ce6960eb5a1114f133e957cfbc Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Mon, 1 May 2017 11:22:24 -0700
Subject: [PATCH 267/305] minor code refactoring for clarity

---
 lib/decompress/zstd_decompress.c | 70 ++++++++++++++++++++------------
 1 file changed, 44 insertions(+), 26 deletions(-)

diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c
index 634e2152e..910f9ab78 100644
--- a/lib/decompress/zstd_decompress.c
+++ b/lib/decompress/zstd_decompress.c
@@ -815,21 +815,26 @@ static seq_t ZSTD_decodeSequence(seqState_t* seqState)
     U32 const totalBits = llBits+mlBits+ofBits;
 
     static const U32 LL_base[MaxLL+1] = {
-                             0,  1,  2,  3,  4,  5,  6,  7,  8,  9,   10,    11,    12,    13,    14,     15,
-                            16, 18, 20, 22, 24, 28, 32, 40, 48, 64, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000,
+                             0,    1,    2,     3,     4,     5,     6,      7,
+                             8,    9,   10,    11,    12,    13,    14,     15,
+                            16,   18,   20,    22,    24,    28,    32,     40,
+                            48,   64, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000,
                             0x2000, 0x4000, 0x8000, 0x10000 };
 
     static const U32 ML_base[MaxML+1] = {
-                             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, 37, 39, 41, 43, 47, 51, 59, 67, 83, 99, 0x83, 0x103, 0x203, 0x403, 0x803,
+                             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, 37, 39,   41,    43,    47,    51,    59,
+                            67, 83, 99, 0x83, 0x103, 0x203, 0x403, 0x803,
                             0x1003, 0x2003, 0x4003, 0x8003, 0x10003 };
 
     static const U32 OF_base[MaxOff+1] = {
-                             0,        1,       1,       5,     0xD,     0x1D,     0x3D,     0x7D,
-                             0xFD,   0x1FD,   0x3FD,   0x7FD,   0xFFD,   0x1FFD,   0x3FFD,   0x7FFD,
-                             0xFFFD, 0x1FFFD, 0x3FFFD, 0x7FFFD, 0xFFFFD, 0x1FFFFD, 0x3FFFFD, 0x7FFFFD,
-                             0xFFFFFD, 0x1FFFFFD, 0x3FFFFFD, 0x7FFFFFD, 0xFFFFFFD };
+                     0,        1,       1,       5,     0xD,     0x1D,     0x3D,     0x7D,
+                     0xFD,   0x1FD,   0x3FD,   0x7FD,   0xFFD,   0x1FFD,   0x3FFD,   0x7FFD,
+                     0xFFFD, 0x1FFFD, 0x3FFFD, 0x7FFFD, 0xFFFFD, 0x1FFFFD, 0x3FFFFD, 0x7FFFFD,
+                     0xFFFFFD, 0x1FFFFFD, 0x3FFFFFD, 0x7FFFFFD, 0xFFFFFFD };
 
     /* sequence */
     {   size_t offset;
@@ -1028,21 +1033,26 @@ FORCE_INLINE seq_t ZSTD_decodeSequenceLong_generic(seqState_t* seqState, int con
     U32 const totalBits = llBits+mlBits+ofBits;
 
     static const U32 LL_base[MaxLL+1] = {
-                             0,  1,  2,  3,  4,  5,  6,  7,  8,  9,   10,    11,    12,    13,    14,     15,
-                            16, 18, 20, 22, 24, 28, 32, 40, 48, 64, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000,
+                             0,  1,    2,     3,     4,     5,     6,      7,
+                             8,  9,   10,    11,    12,    13,    14,     15,
+                            16, 18,   20,    22,    24,    28,    32,     40,
+                            48, 64, 0x80, 0x100, 0x200, 0x400, 0x800, 0x1000,
                             0x2000, 0x4000, 0x8000, 0x10000 };
 
     static const U32 ML_base[MaxML+1] = {
-                             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, 37, 39, 41, 43, 47, 51, 59, 67, 83, 99, 0x83, 0x103, 0x203, 0x403, 0x803,
+                             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, 37, 39,   41,    43,    47,    51,    59,
+                            67, 83, 99, 0x83, 0x103, 0x203, 0x403, 0x803,
                             0x1003, 0x2003, 0x4003, 0x8003, 0x10003 };
 
     static const U32 OF_base[MaxOff+1] = {
-                             0,        1,       1,       5,     0xD,     0x1D,     0x3D,     0x7D,
-                             0xFD,   0x1FD,   0x3FD,   0x7FD,   0xFFD,   0x1FFD,   0x3FFD,   0x7FFD,
-                             0xFFFD, 0x1FFFD, 0x3FFFD, 0x7FFFD, 0xFFFFD, 0x1FFFFD, 0x3FFFFD, 0x7FFFFD,
-                             0xFFFFFD, 0x1FFFFFD, 0x3FFFFFD, 0x7FFFFFD, 0xFFFFFFD };
+                     0,        1,       1,       5,     0xD,     0x1D,     0x3D,     0x7D,
+                     0xFD,   0x1FD,   0x3FD,   0x7FD,   0xFFD,   0x1FFD,   0x3FFD,   0x7FFD,
+                     0xFFFD, 0x1FFFD, 0x3FFFD, 0x7FFFD, 0xFFFFD, 0x1FFFFD, 0x3FFFFD, 0x7FFFFD,
+                     0xFFFFFD, 0x1FFFFFD, 0x3FFFFFD, 0x7FFFFFD, 0xFFFFFFD };
 
     /* sequence */
     {   size_t offset;
@@ -1987,15 +1997,18 @@ unsigned ZSTD_getDictID_fromDDict(const ZSTD_DDict* ddict)
 }
 
 /*! ZSTD_getDictID_fromFrame() :
- *  Provides the dictID required to decompressed the frame stored within `src`.
+ *  Provides the dictID required to decompresse frame stored within `src`.
  *  If @return == 0, the dictID could not be decoded.
  *  This could for one of the following reasons :
- *  - The frame does not require a dictionary to be decoded (most common case).
- *  - The frame was built with dictID intentionally removed. Whatever dictionary is necessary is a hidden information.
+ *  - The frame does not require a dictionary (most common case).
+ *  - The frame was built with dictID intentionally removed.
+ *    Needed dictionary is a hidden information.
  *    Note : this use case also happens when using a non-conformant dictionary.
- *  - `srcSize` is too small, and as a result, the frame header could not be decoded (only possible if `srcSize < ZSTD_FRAMEHEADERSIZE_MAX`).
+ *  - `srcSize` is too small, and as a result, frame header could not be decoded.
+ *    Note : possible if `srcSize < ZSTD_FRAMEHEADERSIZE_MAX`.
  *  - This is not a Zstandard frame.
- *  When identifying the exact failure cause, it's possible to used ZSTD_getFrameParams(), which will provide a more precise error code. */
+ *  When identifying the exact failure cause, it's possible to use
+ *  ZSTD_getFrameParams(), which will provide a more precise error code. */
 unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize)
 {
     ZSTD_frameParams zfp = { 0 , 0 , 0 , 0 };
@@ -2123,7 +2136,9 @@ size_t ZSTD_initDStream(ZSTD_DStream* zds)
     return ZSTD_initDStream_usingDict(zds, NULL, 0);
 }
 
-size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, const ZSTD_DDict* ddict)  /**< note : ddict will just be referenced, and must outlive decompression session */
+/* ZSTD_initDStream_usingDDict() :
+ * ddict will just be referenced, and must outlive decompression session */
+size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, const ZSTD_DDict* ddict)
 {
     size_t const initResult = ZSTD_initDStream(zds);
     zds->ddict = ddict;
@@ -2153,8 +2168,11 @@ size_t ZSTD_setDStreamParameter(ZSTD_DStream* zds,
 
 size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds)
 {
-    if (zds==NULL) return 0;   /* support sizeof on NULL */
-    return sizeof(*zds) + ZSTD_sizeof_DCtx(zds->dctx) + ZSTD_sizeof_DDict(zds->ddictLocal) + zds->inBuffSize + zds->outBuffSize;
+    if (zds==NULL) return 0;   /* support sizeof NULL */
+    return sizeof(*zds)
+           + ZSTD_sizeof_DCtx(zds->dctx)
+           + ZSTD_sizeof_DDict(zds->ddictLocal)
+           + zds->inBuffSize + zds->outBuffSize;
 }
 
 

From feba969a6932d35b97701e4581ae042dcc32ab00 Mon Sep 17 00:00:00 2001
From: Nick Terrell 
Date: Mon, 1 May 2017 10:13:56 -0700
Subject: [PATCH 268/305] Fix LZ4_MSG in xzstd

---
 programs/Makefile | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/programs/Makefile b/programs/Makefile
index 2619614c5..2ab29e1ef 100644
--- a/programs/Makefile
+++ b/programs/Makefile
@@ -145,7 +145,7 @@ zstd-nogz : ZLIB_MSG := $(NO_ZLIB_MSG)
 zstd-nogz : LZMA_MSG := $(NO_LZMA_MSG)
 xzstd : CPPFLAGS += $(ZLIBCPP) $(LZMACPP)
 xzstd : LDFLAGS += $(ZLIBLD) $(LZMALD)
-xzstd : LZ4_MSG := $(NO_LZMA_MSG)
+xzstd : LZ4_MSG := $(NO_LZ4_MSG)
 zstd4 : CPPFLAGS += $(ZLIBCPP) $(LZ4CPP)
 zstd4 : LDFLAGS += $(ZLIBLD) $(LZ4LD)
 zstd4 : LZMA_MSG := $(NO_LZMA_MSG)

From 865918dd041ec7bdfb730ecc2c346659e469216e Mon Sep 17 00:00:00 2001
From: Nick Terrell 
Date: Mon, 1 May 2017 10:14:15 -0700
Subject: [PATCH 269/305] Fix typo in zdict.h

---
 lib/dictBuilder/zdict.h | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/lib/dictBuilder/zdict.h b/lib/dictBuilder/zdict.h
index 669b78d08..9b53de346 100644
--- a/lib/dictBuilder/zdict.h
+++ b/lib/dictBuilder/zdict.h
@@ -88,7 +88,7 @@ ZDICTLIB_API size_t ZDICT_trainFromBuffer_advanced(void* dictBuffer, size_t dict
 
 /*! COVER_params_t :
     For all values 0 means default.
-    kMin and d are the only required parameters.
+    k and d are the only required parameters.
 */
 typedef struct {
     unsigned k;                  /* Segment size : constraint: 0 < k : Reasonable range [16, 2048+] */

From f2d9ef1dc0d491b75877a9200876a7c16dd9694d Mon Sep 17 00:00:00 2001
From: Nick Terrell 
Date: Mon, 1 May 2017 10:25:49 -0700
Subject: [PATCH 270/305] [cover] Optimize case where d <= 8

---
 lib/dictBuilder/cover.c | 37 ++++++++++++++++++++++++++++++-------
 programs/zstd.1         |  4 ++--
 programs/zstd.1.md      |  3 ++-
 3 files changed, 34 insertions(+), 10 deletions(-)

diff --git a/lib/dictBuilder/cover.c b/lib/dictBuilder/cover.c
index 1db42f95b..4235f112b 100644
--- a/lib/dictBuilder/cover.c
+++ b/lib/dictBuilder/cover.c
@@ -234,10 +234,22 @@ static size_t COVER_sum(const size_t *samplesSizes, unsigned nbSamples) {
  * Returns 1 if the dmer at lp is greater than the dmer at rp.
  */
 static int COVER_cmp(COVER_ctx_t *ctx, const void *lp, const void *rp) {
-  const U32 lhs = *(const U32 *)lp;
-  const U32 rhs = *(const U32 *)rp;
+  U32 const lhs = *(U32 const *)lp;
+  U32 const rhs = *(U32 const *)rp;
   return memcmp(ctx->samples + lhs, ctx->samples + rhs, ctx->d);
 }
+/**
+ * Faster version for d <= 8.
+ */
+static int COVER_cmp8(COVER_ctx_t *ctx, const void *lp, const void *rp) {
+  U64 const mask = (ctx->d == 8) ? (U64)-1 : (((U64)1 << (8 * ctx->d)) - 1);
+  U64 const lhs = MEM_readLE64(ctx->samples + *(U32 const *)lp) & mask;
+  U64 const rhs = MEM_readLE64(ctx->samples + *(U32 const *)rp) & mask;
+  if (lhs < rhs) {
+    return -1;
+  }
+  return (lhs > rhs);
+}
 
 /**
  * Same as COVER_cmp() except ties are broken by pointer value
@@ -251,6 +263,16 @@ static int COVER_strict_cmp(const void *lp, const void *rp) {
   }
   return result;
 }
+/**
+ * Faster version for d <= 8.
+ */
+static int COVER_strict_cmp8(const void *lp, const void *rp) {
+  int result = COVER_cmp8(g_ctx, lp, rp);
+  if (result == 0) {
+    result = lp < rp ? -1 : 1;
+  }
+  return result;
+}
 
 /**
  * Returns the first pointer in [first, last) whose element does not compare
@@ -506,7 +528,7 @@ static int COVER_ctx_init(COVER_ctx_t *ctx, const void *samplesBuffer,
   const BYTE *const samples = (const BYTE *)samplesBuffer;
   const size_t totalSamplesSize = COVER_sum(samplesSizes, nbSamples);
   /* Checks */
-  if (totalSamplesSize < d ||
+  if (totalSamplesSize < MAX(d, sizeof(U64)) ||
       totalSamplesSize >= (size_t)COVER_MAX_SAMPLES_SIZE) {
     DISPLAYLEVEL(1, "Total samples size is too large, maximum size is %u MB\n",
                  (COVER_MAX_SAMPLES_SIZE >> 20));
@@ -520,7 +542,7 @@ static int COVER_ctx_init(COVER_ctx_t *ctx, const void *samplesBuffer,
   ctx->samplesSizes = samplesSizes;
   ctx->nbSamples = nbSamples;
   /* Partial suffix array */
-  ctx->suffixSize = totalSamplesSize - d + 1;
+  ctx->suffixSize = totalSamplesSize - MAX(d, sizeof(U64)) + 1;
   ctx->suffix = (U32 *)malloc(ctx->suffixSize * sizeof(U32));
   /* Maps index to the dmerID */
   ctx->dmerAt = (U32 *)malloc(ctx->suffixSize * sizeof(U32));
@@ -554,7 +576,8 @@ static int COVER_ctx_init(COVER_ctx_t *ctx, const void *samplesBuffer,
     }
     /* qsort doesn't take an opaque pointer, so pass as a global */
     g_ctx = ctx;
-    qsort(ctx->suffix, ctx->suffixSize, sizeof(U32), &COVER_strict_cmp);
+    qsort(ctx->suffix, ctx->suffixSize, sizeof(U32),
+          (ctx->d <= 8 ? &COVER_strict_cmp8 : &COVER_strict_cmp));
   }
   DISPLAYLEVEL(2, "Computing frequencies\n");
   /* For each dmer group (group of positions with the same first d bytes):
@@ -564,8 +587,8 @@ static int COVER_ctx_init(COVER_ctx_t *ctx, const void *samplesBuffer,
    * 2. We calculate how many samples the dmer occurs in and save it in
    *    freqs[dmerId].
    */
-  COVER_groupBy(ctx->suffix, ctx->suffixSize, sizeof(U32), ctx, &COVER_cmp,
-                &COVER_group);
+  COVER_groupBy(ctx->suffix, ctx->suffixSize, sizeof(U32), ctx,
+                (ctx->d <= 8 ? &COVER_cmp8 : &COVER_cmp), &COVER_group);
   ctx->freqs = ctx->suffix;
   ctx->suffix = NULL;
   return 1;
diff --git a/programs/zstd.1 b/programs/zstd.1
index 999dc8169..5bd966a84 100644
--- a/programs/zstd.1
+++ b/programs/zstd.1
@@ -1,5 +1,5 @@
 .
-.TH "ZSTD" "1" "April 2017" "zstd 1.1.5" "User Commands"
+.TH "ZSTD" "1" "May 2017" "zstd 1.2.0" "User Commands"
 .
 .SH "NAME"
 \fBzstd\fR \- zstd, zstdmt, unzstd, zstdcat \- Compress or decompress \.zst files
@@ -188,7 +188,7 @@ dictionary selectivity level (default: 9) the smaller the value, the denser the
 .
 .TP
 \fB\-\-cover=k#,d=#\fR
-Use alternate dictionary builder algorithm named cover with parameters \fIk\fR and \fId\fR with \fId\fR <= \fIk\fR\. Selects segments of size \fIk\fR with the highest score to put in the dictionary\. The score of a segment is computed by the sum of the frequencies of all the subsegments of of size \fId\fR\. Generally \fId\fR should be in the range [6, 24]\. Good values for \fIk\fR vary widely based on the input data, but a safe range is [32, 2048]\.
+Use alternate dictionary builder algorithm named cover with parameters \fIk\fR and \fId\fR with \fId\fR <= \fIk\fR\. Selects segments of size \fIk\fR with the highest score to put in the dictionary\. The score of a segment is computed by the sum of the frequencies of all the subsegments of of size \fId\fR\. Generally \fId\fR should be in the range [6, 8], but no more than 24\. When \fId\fR <= 8, the dictionary builder will run significantly faster\. Good values for \fIk\fR vary widely based on the input data, but a safe range is [32, 2048]\.
 .
 .br
 Example: \fB\-\-train \-\-cover=k=64,d=8 FILEs\fR\.
diff --git a/programs/zstd.1.md b/programs/zstd.1.md
index f2d04d16f..0919da702 100644
--- a/programs/zstd.1.md
+++ b/programs/zstd.1.md
@@ -186,7 +186,8 @@ Typical gains range from 10% (at 64KB) to x5 better (at <1KB).
     Selects segments of size _k_ with the highest score to put in the dictionary.
     The score of a segment is computed by the sum of the frequencies of all the
     subsegments of of size _d_.
-    Generally _d_ should be in the range [6, 24].
+    Generally _d_ should be in the range [6, 8], but no more than 24.
+    When _d_ <= 8, the dictionary builder will run significantly faster.
     Good values for _k_ vary widely based on the input data,
     but a safe range is [32, 2048].
Example: `--train --cover=k=64,d=8 FILEs`. From 020b960e13499cb2cd524cc6b7ddb0ccbd76c361 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Mon, 1 May 2017 21:26:33 -0700 Subject: [PATCH 271/305] [cover] Make optimization faster --- lib/dictBuilder/cover.c | 4 ++-- programs/zstd.1 | 6 +++--- programs/zstd.1.md | 9 +++------ 3 files changed, 8 insertions(+), 11 deletions(-) diff --git a/lib/dictBuilder/cover.c b/lib/dictBuilder/cover.c index 4235f112b..0e156bd58 100644 --- a/lib/dictBuilder/cover.c +++ b/lib/dictBuilder/cover.c @@ -939,8 +939,8 @@ ZDICTLIB_API size_t COVER_optimizeTrainFromBuffer(void *dictBuffer, /* constants */ const unsigned nbThreads = parameters->nbThreads; const unsigned kMinD = parameters->d == 0 ? 6 : parameters->d; - const unsigned kMaxD = parameters->d == 0 ? 16 : parameters->d; - const unsigned kMinK = parameters->k == 0 ? kMaxD : parameters->k; + const unsigned kMaxD = parameters->d == 0 ? 8 : parameters->d; + const unsigned kMinK = parameters->k == 0 ? 40 + kMaxD : parameters->k; const unsigned kMaxK = parameters->k == 0 ? 2048 : parameters->k; const unsigned kSteps = parameters->steps == 0 ? 32 : parameters->steps; const unsigned kStepSize = MAX((kMaxK - kMinK) / kSteps, 1); diff --git a/programs/zstd.1 b/programs/zstd.1 index 5bd966a84..648c3365c 100644 --- a/programs/zstd.1 +++ b/programs/zstd.1 @@ -195,13 +195,13 @@ Example: \fB\-\-train \-\-cover=k=64,d=8 FILEs\fR\. . .TP \fB\-\-optimize\-cover[=steps=#,k=#,d=#]\fR -If \fIsteps\fR is not specified, the default value of 32 is used\. If \fIk\fR is not specified, the \fIk\fR values in [16, 2048] are checked for each value of \fId\fR\. If \fId\fR is not specified, the values checked are [6, 8, \.\.\., 16]\. +If \fIsteps\fR is not specified, the default value of 32 is used\. If \fIk\fR is not specified, the \fIk\fR values in [48, 2048] are checked for each value of \fId\fR\. If \fId\fR is not specified, the values checked are [6, 8]\. . .IP Runs the cover dictionary builder for each parameter set and saves the optimal parameters and dictionary\. Prints optimal parameters and writes optimal dictionary into output file\. Supports multithreading if \fBzstd\fR is compiled with threading support\. . .IP -The parameter \fIk\fR is more sensitive than \fId\fR, and is faster to optimize over\. Suggested use is to run with a \fIsteps\fR <= 32 with neither \fIk\fR nor \fId\fR set\. Once it completes, use the value of \fId\fR it selects with a higher \fIsteps\fR (in the range [256, 1024])\. +The parameter \fIk\fR is more sensitive than \fId\fR, and is faster to optimize over\. . .IP Examples : @@ -210,7 +210,7 @@ Examples : \fBzstd \-\-train \-\-optimize\-cover FILEs\fR . .IP -\fBzstd \-\-train \-\-optimize\-cover=d=d,steps=512 FILEs\fR +\fBzstd \-\-train \-\-optimize\-cover=d=8,steps=512 FILEs\fR . .SH "BENCHMARK" . diff --git a/programs/zstd.1.md b/programs/zstd.1.md index 0919da702..ea346cfb7 100644 --- a/programs/zstd.1.md +++ b/programs/zstd.1.md @@ -194,9 +194,9 @@ Typical gains range from 10% (at 64KB) to x5 better (at <1KB). * `--optimize-cover[=steps=#,k=#,d=#]`: If _steps_ is not specified, the default value of 32 is used. - If _k_ is not specified, the _k_ values in [16, 2048] are checked for each + If _k_ is not specified, the _k_ values in [48, 2048] are checked for each value of _d_. - If _d_ is not specified, the values checked are [6, 8, ..., 16]. + If _d_ is not specified, the values checked are [6, 8]. Runs the cover dictionary builder for each parameter set and saves the optimal parameters and dictionary. @@ -204,15 +204,12 @@ Typical gains range from 10% (at 64KB) to x5 better (at <1KB). Supports multithreading if `zstd` is compiled with threading support. The parameter _k_ is more sensitive than _d_, and is faster to optimize over. - Suggested use is to run with a _steps_ <= 32 with neither _k_ nor _d_ set. - Once it completes, use the value of _d_ it selects with a higher _steps_ - (in the range [256, 1024]). Examples : `zstd --train --optimize-cover FILEs` - `zstd --train --optimize-cover=d=d,steps=512 FILEs` + `zstd --train --optimize-cover=d=8,steps=512 FILEs` BENCHMARK From f376d47c11e083c08a35ec3de6e4a66b7da60b14 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Mon, 1 May 2017 23:40:20 -0700 Subject: [PATCH 272/305] [CLI] Switch dictionary builder on CLI to cover --- lib/dictBuilder/cover.c | 6 ++-- programs/zstd.1 | 54 ++++++++++++++++++------------- programs/zstd.1.md | 71 +++++++++++++++++++++++------------------ programs/zstdcli.c | 62 ++++++++++++++++++++++++++--------- tests/playTests.sh | 32 +++++++++++++++---- 5 files changed, 147 insertions(+), 78 deletions(-) diff --git a/lib/dictBuilder/cover.c b/lib/dictBuilder/cover.c index 0e156bd58..1863c8f34 100644 --- a/lib/dictBuilder/cover.c +++ b/lib/dictBuilder/cover.c @@ -940,9 +940,9 @@ ZDICTLIB_API size_t COVER_optimizeTrainFromBuffer(void *dictBuffer, const unsigned nbThreads = parameters->nbThreads; const unsigned kMinD = parameters->d == 0 ? 6 : parameters->d; const unsigned kMaxD = parameters->d == 0 ? 8 : parameters->d; - const unsigned kMinK = parameters->k == 0 ? 40 + kMaxD : parameters->k; - const unsigned kMaxK = parameters->k == 0 ? 2048 : parameters->k; - const unsigned kSteps = parameters->steps == 0 ? 32 : parameters->steps; + const unsigned kMinK = parameters->k == 0 ? 50 : parameters->k; + const unsigned kMaxK = parameters->k == 0 ? 2000 : parameters->k; + const unsigned kSteps = parameters->steps == 0 ? 40 : parameters->steps; const unsigned kStepSize = MAX((kMaxK - kMinK) / kSteps, 1); const unsigned kIterations = (1 + (kMaxD - kMinD) / 2) * (1 + (kMaxK - kMinK) / kStepSize); diff --git a/programs/zstd.1 b/programs/zstd.1 index 648c3365c..6cc5f7e9d 100644 --- a/programs/zstd.1 +++ b/programs/zstd.1 @@ -168,49 +168,57 @@ All arguments after \fB\-\-\fR are treated as files . .TP \fB\-\-train FILEs\fR -use FILEs as training set to create a dictionary\. The training set should contain a lot of small files (> 100), and weight typically 100x the target dictionary size (for example, 10 MB for a 100 KB dictionary)\. +Use FILEs as training set to create a dictionary\. The training set should contain a lot of small files (> 100), and weight typically 100x the target dictionary size (for example, 10 MB for a 100 KB dictionary)\. +. +.IP +Supports multithreading if \fBzstd\fR is compiled with threading support\. Additional parameters can be specified with \fB\-\-train\-cover\fR\. The legacy dictionary builder can be accessed with \fB\-\-train\-legacy\fR\. Equivalent to \fB\-\-train\-cover=d=8,steps=4\fR\. . .TP \fB\-o file\fR -dictionary saved into \fBfile\fR (default name: dictionary) +Dictionary saved into \fBfile\fR (default name: dictionary)\. . .TP \fB\-\-maxdict=#\fR -limit dictionary to specified size (default : (112640) +Limit dictionary to specified size (default: 112640)\. . .TP \fB\-\-dictID=#\fR A dictionary ID is a locally unique ID that a decoder can use to verify it is using the right dictionary\. By default, zstd will create a 4\-bytes random number ID\. It\'s possible to give a precise number instead\. Short numbers have an advantage : an ID < 256 will only need 1 byte in the compressed frame header, and an ID < 65536 will only need 2 bytes\. This compares favorably to 4 bytes default\. However, it\'s up to the dictionary manager to not assign twice the same ID to 2 different dictionaries\. . .TP -\fB\-s#\fR -dictionary selectivity level (default: 9) the smaller the value, the denser the dictionary, improving its efficiency but reducing its possible maximum size\. +\fB\-\-train\-cover[=k#,d=#,steps=#]\fR +Select parameters for the default dictionary builder algorithm named cover\. If \fId\fR is not specified, then it tries \fId\fR = 6 and \fId\fR = 8\. If \fIk\fR is not specified, then it tries \fIsteps\fR values in the range [50, 2000]\. If \fIsteps\fR is not specified, then the default value of 40 is used\. Requires that \fId\fR <= \fIk\fR\. +. +.IP +Selects segments of size \fIk\fR with highest score to put in the dictionary\. The score of a segment is computed by the sum of the frequencies of all the subsegments of size \fId\fR\. Generally \fId\fR should be in the range [6, 8], occasionally up to 16, but the algorithm will run faster with d <= \fI8\fR\. Good values for \fIk\fR vary widely based on the input data, but a safe range is [2 * \fId\fR, 2000]\. Supports multithreading if \fBzstd\fR is compiled with threading support\. +. +.IP +Examples: +. +.IP +\fBzstd \-\-train\-cover FILEs\fR +. +.IP +\fBzstd \-\-train\-cover=k=50,d=8 FILEs\fR +. +.IP +\fBzstd \-\-train\-cover=d=8,steps=500 FILEs\fR +. +.IP +\fBzstd \-\-train\-cover=k=50 FILEs\fR . .TP -\fB\-\-cover=k#,d=#\fR -Use alternate dictionary builder algorithm named cover with parameters \fIk\fR and \fId\fR with \fId\fR <= \fIk\fR\. Selects segments of size \fIk\fR with the highest score to put in the dictionary\. The score of a segment is computed by the sum of the frequencies of all the subsegments of of size \fId\fR\. Generally \fId\fR should be in the range [6, 8], but no more than 24\. When \fId\fR <= 8, the dictionary builder will run significantly faster\. Good values for \fIk\fR vary widely based on the input data, but a safe range is [32, 2048]\. -. -.br -Example: \fB\-\-train \-\-cover=k=64,d=8 FILEs\fR\. -. -.TP -\fB\-\-optimize\-cover[=steps=#,k=#,d=#]\fR -If \fIsteps\fR is not specified, the default value of 32 is used\. If \fIk\fR is not specified, the \fIk\fR values in [48, 2048] are checked for each value of \fId\fR\. If \fId\fR is not specified, the values checked are [6, 8]\. +\fB\-\-train\-legacy[=selectivity=#]\fR +Use legacy dictionary builder algorithm with the given dictionary \fIselectivity\fR (default: 9)\. The smaller the \fIselectivity\fR value, the denser the dictionary, improving its efficiency but reducing its possible maximum size\. \fB\-\-train\-legacy=s=#\fR is also accepted\. . .IP -Runs the cover dictionary builder for each parameter set and saves the optimal parameters and dictionary\. Prints optimal parameters and writes optimal dictionary into output file\. Supports multithreading if \fBzstd\fR is compiled with threading support\. +Examples: . .IP -The parameter \fIk\fR is more sensitive than \fId\fR, and is faster to optimize over\. +\fBzstd \-\-train\-legacy FILEs\fR . .IP -Examples : -. -.IP -\fBzstd \-\-train \-\-optimize\-cover FILEs\fR -. -.IP -\fBzstd \-\-train \-\-optimize\-cover=d=8,steps=512 FILEs\fR +\fBzstd \-\-train\-legacy=selectivity=8 FILEs\fR . .SH "BENCHMARK" . diff --git a/programs/zstd.1.md b/programs/zstd.1.md index ea346cfb7..118c9f2f8 100644 --- a/programs/zstd.1.md +++ b/programs/zstd.1.md @@ -158,14 +158,19 @@ It will improve compression ratio of small files. Typical gains range from 10% (at 64KB) to x5 better (at <1KB). * `--train FILEs`: - use FILEs as training set to create a dictionary. + Use FILEs as training set to create a dictionary. The training set should contain a lot of small files (> 100), and weight typically 100x the target dictionary size (for example, 10 MB for a 100 KB dictionary). + + Supports multithreading if `zstd` is compiled with threading support. + Additional parameters can be specified with `--train-cover`. + The legacy dictionary builder can be accessed with `--train-legacy`. + Equivalent to `--train-cover=d=8,steps=4`. * `-o file`: - dictionary saved into `file` (default name: dictionary) + Dictionary saved into `file` (default name: dictionary). * `--maxdict=#`: - limit dictionary to specified size (default : (112640) + Limit dictionary to specified size (default: 112640). * `--dictID=#`: A dictionary ID is a locally unique ID that a decoder can use to verify it is using the right dictionary. @@ -176,40 +181,44 @@ Typical gains range from 10% (at 64KB) to x5 better (at <1KB). This compares favorably to 4 bytes default. However, it's up to the dictionary manager to not assign twice the same ID to 2 different dictionaries. -* `-s#`: - dictionary selectivity level (default: 9) - the smaller the value, the denser the dictionary, - improving its efficiency but reducing its possible maximum size. -* `--cover=k#,d=#`: - Use alternate dictionary builder algorithm named cover with parameters - _k_ and _d_ with _d_ <= _k_. - Selects segments of size _k_ with the highest score to put in the dictionary. +* `--train-cover[=k#,d=#,steps=#]`: + Select parameters for the default dictionary builder algorithm named cover. + If _d_ is not specified, then it tries _d_ = 6 and _d_ = 8. + If _k_ is not specified, then it tries _steps_ values in the range [50, 2000]. + If _steps_ is not specified, then the default value of 40 is used. + Requires that _d_ <= _k_. + + Selects segments of size _k_ with highest score to put in the dictionary. The score of a segment is computed by the sum of the frequencies of all the - subsegments of of size _d_. - Generally _d_ should be in the range [6, 8], but no more than 24. - When _d_ <= 8, the dictionary builder will run significantly faster. - Good values for _k_ vary widely based on the input data, - but a safe range is [32, 2048].
- Example: `--train --cover=k=64,d=8 FILEs`. - -* `--optimize-cover[=steps=#,k=#,d=#]`: - If _steps_ is not specified, the default value of 32 is used. - If _k_ is not specified, the _k_ values in [48, 2048] are checked for each - value of _d_. - If _d_ is not specified, the values checked are [6, 8]. - - Runs the cover dictionary builder for each parameter set - and saves the optimal parameters and dictionary. - Prints optimal parameters and writes optimal dictionary into output file. + subsegments of size _d_. + Generally _d_ should be in the range [6, 8], occasionally up to 16, but the + algorithm will run faster with d <= _8_. + Good values for _k_ vary widely based on the input data, but a safe range is + [2 * _d_, 2000]. Supports multithreading if `zstd` is compiled with threading support. - The parameter _k_ is more sensitive than _d_, and is faster to optimize over. + Examples: - Examples : + `zstd --train-cover FILEs` - `zstd --train --optimize-cover FILEs` + `zstd --train-cover=k=50,d=8 FILEs` - `zstd --train --optimize-cover=d=8,steps=512 FILEs` + `zstd --train-cover=d=8,steps=500 FILEs` + + `zstd --train-cover=k=50 FILEs` + +* `--train-legacy[=selectivity=#]`: + Use legacy dictionary builder algorithm with the given dictionary + _selectivity_ (default: 9). + The smaller the _selectivity_ value, the denser the dictionary, + improving its efficiency but reducing its possible maximum size. + `--train-legacy=s=#` is also accepted. + + Examples: + + `zstd --train-legacy FILEs` + + `zstd --train-legacy=selectivity=8 FILEs` BENCHMARK diff --git a/programs/zstdcli.c b/programs/zstdcli.c index 79bc84877..f6ce2e1a6 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -153,11 +153,10 @@ static int usage_advanced(const char* programName) DISPLAY( "\n"); DISPLAY( "Dictionary builder :\n"); DISPLAY( "--train ## : create a dictionary from a training set of files \n"); - DISPLAY( "--cover=k=#,d=# : use the cover algorithm with parameters k and d \n"); - DISPLAY( "--optimize-cover[=steps=#,k=#,d=#] : optimize cover parameters with optional parameters\n"); + DISPLAY( "--train-cover[=k=#,d=#,steps=#] : use the cover algorithm with optional args\n"); + DISPLAY( "--train-legacy[=s=#] : use the legacy algorithm with selectivity (default: %u)\n", g_defaultSelectivityLevel); DISPLAY( " -o file : `file` is dictionary name (default: %s) \n", g_defaultDictName); DISPLAY( "--maxdict=# : limit dictionary to specified size (default : %u) \n", g_defaultMaxDictSize); - DISPLAY( " -s# : dictionary selectivity level (default: %u)\n", g_defaultSelectivityLevel); DISPLAY( "--dictID=# : force dictionary ID to specified value (default: random)\n"); #endif #ifndef ZSTD_NOBENCH @@ -241,11 +240,11 @@ static unsigned longCommandWArg(const char** stringPtr, const char* longCommand) #ifndef ZSTD_NODICT /** * parseCoverParameters() : - * reads cover parameters from *stringPtr (e.g. "--cover=smoothing=100,kmin=48,kstep=4,kmax=64,d=8") into *params + * reads cover parameters from *stringPtr (e.g. "--train-cover=k=48,d=8,steps=32") into *params * @return 1 means that cover parameters were correct * @return 0 in case of malformed parameters */ -static unsigned parseCoverParameters(const char* stringPtr, COVER_params_t *params) +static unsigned parseCoverParameters(const char* stringPtr, COVER_params_t* params) { memset(params, 0, sizeof(*params)); for (; ;) { @@ -255,9 +254,33 @@ static unsigned parseCoverParameters(const char* stringPtr, COVER_params_t *para return 0; } if (stringPtr[0] != 0) return 0; - DISPLAYLEVEL(4, "k=%u\nd=%u\nsteps=%u\n", params->k, params->d, params->steps); + DISPLAYLEVEL(4, "cover: k=%u\nd=%u\nsteps=%u\n", params->k, params->d, params->steps); return 1; } + +/** + * parseLegacyParameters() : + * reads legacy dictioanry builter parameters from *stringPtr (e.g. "--train-legacy=selectivity=8") into *selectivity + * @return 1 means that legacy dictionary builder parameters were correct + * @return 0 in case of malformed parameters + */ +static unsigned parseLegacyParameters(const char* stringPtr, unsigned* selectivity) +{ + if (!longCommandWArg(&stringPtr, "s=") && !longCommandWArg(&stringPtr, "selectivity=")) { return 0; } + *selectivity = readU32FromChar(&stringPtr); + if (stringPtr[0] != 0) return 0; + DISPLAYLEVEL(4, "legacy: selectivity=%u\n", *selectivity); + return 1; +} + +static COVER_params_t defaultCoverParams(void) +{ + COVER_params_t params; + memset(¶ms, 0, sizeof(params)); + params.d = 8; + params.steps = 4; + return params; +} #endif @@ -331,8 +354,8 @@ int main(int argCount, const char* argv[]) unsigned fileNamesNb; #endif #ifndef ZSTD_NODICT - COVER_params_t coverParams; - int cover = 0; + COVER_params_t coverParams = defaultCoverParams(); + int cover = 1; #endif /* init */ @@ -413,18 +436,26 @@ int main(int argCount, const char* argv[]) /* long commands with arguments */ #ifndef ZSTD_NODICT - if (longCommandWArg(&argument, "--cover=")) { - cover=1; if (!parseCoverParameters(argument, &coverParams)) CLEAN_RETURN(badusage(programName)); - continue; - } - if (longCommandWArg(&argument, "--optimize-cover")) { - cover=2; + if (longCommandWArg(&argument, "--train-cover")) { + operation = zom_train; + outFileName = g_defaultDictName; + cover = 1; /* Allow optional arguments following an = */ if (*argument == 0) { memset(&coverParams, 0, sizeof(coverParams)); } else if (*argument++ != '=') { CLEAN_RETURN(badusage(programName)); } else if (!parseCoverParameters(argument, &coverParams)) { CLEAN_RETURN(badusage(programName)); } continue; } + if (longCommandWArg(&argument, "--train-legacy")) { + operation = zom_train; + outFileName = g_defaultDictName; + cover = 0; + /* Allow optional arguments following an = */ + if (*argument == 0) { continue; } + else if (*argument++ != '=') { CLEAN_RETURN(badusage(programName)); } + else if (!parseLegacyParameters(argument, &dictSelect)) { CLEAN_RETURN(badusage(programName)); } + continue; + } #endif if (longCommandWArg(&argument, "--threads=")) { nbThreads = readU32FromChar(&argument); continue; } if (longCommandWArg(&argument, "--memlimit=")) { memLimit = readU32FromChar(&argument); continue; } @@ -659,11 +690,12 @@ int main(int argCount, const char* argv[]) if (operation==zom_train) { #ifndef ZSTD_NODICT if (cover) { + int const optimize = !coverParams.k || !coverParams.d; coverParams.nbThreads = nbThreads; coverParams.compressionLevel = dictCLevel; coverParams.notificationLevel = g_displayLevel; coverParams.dictID = dictID; - operationResult = DiB_trainFromFiles(outFileName, maxDictSize, filenameTable, filenameIdx, NULL, &coverParams, cover - 1); + operationResult = DiB_trainFromFiles(outFileName, maxDictSize, filenameTable, filenameIdx, NULL, &coverParams, optimize); } else { ZDICT_params_t dictParams; memset(&dictParams, 0, sizeof(dictParams)); diff --git a/tests/playTests.sh b/tests/playTests.sh index 369506c2e..021fd59fe 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -314,9 +314,9 @@ esac rm -rf dirTestDict $ECHO "- dictionary builder on bogus input" $ECHO "Hello World" > tmp -$ZSTD --train -q tmp && die "Dictionary training should fail : not enough input source" +$ZSTD --train-legacy -q tmp && die "Dictionary training should fail : not enough input source" ./datagen -P0 -g10M > tmp -$ZSTD --train -q tmp && die "Dictionary training should fail : source is pure noise" +$ZSTD --train-legacy -q tmp && die "Dictionary training should fail : source is pure noise" rm tmp* @@ -325,19 +325,39 @@ $ECHO "\n**** cover dictionary tests **** " TESTFILE=../programs/zstdcli.c ./datagen > tmpDict $ECHO "- Create first dictionary" -$ZSTD --train --cover=k=46,d=8 *.c ../programs/*.c -o tmpDict +$ZSTD --train-cover=k=46,d=8 *.c ../programs/*.c -o tmpDict cp $TESTFILE tmp $ZSTD -f tmp -D tmpDict $ZSTD -d tmp.zst -D tmpDict -fo result $DIFF $TESTFILE result $ECHO "- Create second (different) dictionary" -$ZSTD --train --cover=k=56,d=8 *.c ../programs/*.c ../programs/*.h -o tmpDictC +$ZSTD --train-cover=k=56,d=8 *.c ../programs/*.c ../programs/*.h -o tmpDictC $ZSTD -d tmp.zst -D tmpDictC -fo result && die "wrong dictionary not detected!" $ECHO "- Create dictionary with short dictID" -$ZSTD --train --cover=k=46,d=8 *.c ../programs/*.c --dictID=1 -o tmpDict1 +$ZSTD --train-cover=k=46,d=8 *.c ../programs/*.c --dictID=1 -o tmpDict1 cmp tmpDict tmpDict1 && die "dictionaries should have different ID !" $ECHO "- Create dictionary with size limit" -$ZSTD --train --optimize-cover=steps=8 *.c ../programs/*.c -o tmpDict2 --maxdict=4K +$ZSTD --train-cover=steps=8 *.c ../programs/*.c -o tmpDict2 --maxdict=4K +rm tmp* + +$ECHO "\n**** legacy dictionary tests **** " + +TESTFILE=../programs/zstdcli.c +./datagen > tmpDict +$ECHO "- Create first dictionary" +$ZSTD --train-legacy=selectivity=8 *.c ../programs/*.c -o tmpDict +cp $TESTFILE tmp +$ZSTD -f tmp -D tmpDict +$ZSTD -d tmp.zst -D tmpDict -fo result +$DIFF $TESTFILE result +$ECHO "- Create second (different) dictionary" +$ZSTD --train-legacy=s=5 *.c ../programs/*.c ../programs/*.h -o tmpDictC +$ZSTD -d tmp.zst -D tmpDictC -fo result && die "wrong dictionary not detected!" +$ECHO "- Create dictionary with short dictID" +$ZSTD --train-legacy -s5 *.c ../programs/*.c --dictID=1 -o tmpDict1 +cmp tmpDict tmpDict1 && die "dictionaries should have different ID !" +$ECHO "- Create dictionary with size limit" +$ZSTD --train-legacy -s9 *.c ../programs/*.c -o tmpDict2 --maxdict=4K rm tmp* From 7aada3ca44831d6d9b714427107b6c38390dcc31 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 2 May 2017 12:14:12 -0700 Subject: [PATCH 273/305] updated NEWS for v1.2.0 --- NEWS | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/NEWS b/NEWS index b910e2993..407f3b8ac 100644 --- a/NEWS +++ b/NEWS @@ -3,6 +3,8 @@ cli : changed : Multithreading enabled by default (use target zstd-nomt or HAVE_ cli : new : command -T0 means "detect and use nb of cores", by Sean Purcell cli : new : zstdmt symlink hardwired to `zstd -T0` cli : new : command --threads=# (#671) +cli : changed : cover dictionary builder by default, for improved quality, by Nick Terrell +cli : new : commands --train-cover and --train-legacy, to select dictionary algorithm and parameters cli : new : experimental target `zstd4`, with support for lz4 format, by Sean Purcell cli : fix : does not output compressed data on console cli : fix : ignore symbolic links unless --force specified, @@ -11,8 +13,8 @@ API : added : prototypes ZSTD_*_usingCDict_advanced(), for direct control over f API : improved: ZSTDMT_compressCCtx() reduced memory usage API : fix : ZSTDMT_compressCCtx() now provides srcSize in header (#634) API : fix : src size stored in frame header is controlled at end of frame -API : fix : enforced consistent rules for pledgedSrcSize==0 -API : fix : ensure error code "dstSizeTooSmall" is generated when dst size is too small +API : fix : enforced consistent rules for pledgedSrcSize==0 (#641) +API : fix : error code "GENERIC" replaced by "dstSizeTooSmall" when appropriate build: improved cmake script, by @Majlen build: enabled Multi-threading support for *BSD, by Baptiste Daroussin tools: updated Paramgrill. Command -O# provides best parameters for sample and speed target. From 79a85e13719eba7dbd47877d706f941d624be286 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 2 May 2017 13:41:57 -0700 Subject: [PATCH 274/305] [btrfs] Update patch to use ZSTD_init{C,D}Stream() --- contrib/linux-kernel/btrfs.diff | 14 +++++++------- contrib/linux-kernel/fs/btrfs/zstd.c | 12 ++++++------ 2 files changed, 13 insertions(+), 13 deletions(-) diff --git a/contrib/linux-kernel/btrfs.diff b/contrib/linux-kernel/btrfs.diff index b0f8b924b..51fc52fad 100644 --- a/contrib/linux-kernel/btrfs.diff +++ b/contrib/linux-kernel/btrfs.diff @@ -193,7 +193,7 @@ index 1f157fb..b0dec90 100644 BTRFS_FEAT_ATTR_PTR(raid56), diff --git a/fs/btrfs/zstd.c b/fs/btrfs/zstd.c new file mode 100644 -index 0000000..b7f319e +index 0000000..010548c --- /dev/null +++ b/fs/btrfs/zstd.c @@ -0,0 +1,415 @@ @@ -291,10 +291,10 @@ index 0000000..b7f319e + *total_in = 0; + + /* Initialize the stream */ -+ stream = ZSTD_createCStream(params, len, workspace->mem, ++ stream = ZSTD_initCStream(params, len, workspace->mem, + workspace->size); + if (!stream) { -+ pr_warn("BTRFS: ZSTD_createStream failed\n"); ++ pr_warn("BTRFS: ZSTD_initStream failed\n"); + ret = -EIO; + goto out; + } @@ -458,10 +458,10 @@ index 0000000..b7f319e + ZSTD_inBuffer in_buf = { NULL, 0, 0 }; + ZSTD_outBuffer out_buf = { NULL, 0, 0 }; + -+ stream = ZSTD_createDStream( ++ stream = ZSTD_initDStream( + ZSTD_BTRFS_MAX_INPUT, workspace->mem, workspace->size); + if (!stream) { -+ pr_debug("BTRFS: ZSTD_createDStream failed\n"); ++ pr_debug("BTRFS: ZSTD_initDStream failed\n"); + ret = -EIO; + goto done; + } @@ -536,10 +536,10 @@ index 0000000..b7f319e + unsigned long pg_offset = 0; + char *kaddr; + -+ stream = ZSTD_createDStream( ++ stream = ZSTD_initDStream( + ZSTD_BTRFS_MAX_INPUT, workspace->mem, workspace->size); + if (!stream) { -+ pr_warn("BTRFS: ZSTD_createDStream failed\n"); ++ pr_warn("BTRFS: ZSTD_initDStream failed\n"); + ret = -EIO; + goto finish; + } diff --git a/contrib/linux-kernel/fs/btrfs/zstd.c b/contrib/linux-kernel/fs/btrfs/zstd.c index b7f319e7a..010548cea 100644 --- a/contrib/linux-kernel/fs/btrfs/zstd.c +++ b/contrib/linux-kernel/fs/btrfs/zstd.c @@ -92,10 +92,10 @@ static int zstd_compress_pages(struct list_head *ws, *total_in = 0; /* Initialize the stream */ - stream = ZSTD_createCStream(params, len, workspace->mem, + stream = ZSTD_initCStream(params, len, workspace->mem, workspace->size); if (!stream) { - pr_warn("BTRFS: ZSTD_createStream failed\n"); + pr_warn("BTRFS: ZSTD_initStream failed\n"); ret = -EIO; goto out; } @@ -259,10 +259,10 @@ static int zstd_decompress_bio(struct list_head *ws, struct page **pages_in, ZSTD_inBuffer in_buf = { NULL, 0, 0 }; ZSTD_outBuffer out_buf = { NULL, 0, 0 }; - stream = ZSTD_createDStream( + stream = ZSTD_initDStream( ZSTD_BTRFS_MAX_INPUT, workspace->mem, workspace->size); if (!stream) { - pr_debug("BTRFS: ZSTD_createDStream failed\n"); + pr_debug("BTRFS: ZSTD_initDStream failed\n"); ret = -EIO; goto done; } @@ -337,10 +337,10 @@ static int zstd_decompress(struct list_head *ws, unsigned char *data_in, unsigned long pg_offset = 0; char *kaddr; - stream = ZSTD_createDStream( + stream = ZSTD_initDStream( ZSTD_BTRFS_MAX_INPUT, workspace->mem, workspace->size); if (!stream) { - pr_warn("BTRFS: ZSTD_createDStream failed\n"); + pr_warn("BTRFS: ZSTD_initDStream failed\n"); ret = -EIO; goto finish; } From 99972fa9a8e7c42b83c428bc3bc1e86a2b7b3bf9 Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Tue, 2 May 2017 13:57:40 -0700 Subject: [PATCH 275/305] [btrfs] Fix typo in pr_warn() message --- contrib/linux-kernel/btrfs.diff | 2 +- contrib/linux-kernel/fs/btrfs/zstd.c | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/contrib/linux-kernel/btrfs.diff b/contrib/linux-kernel/btrfs.diff index 51fc52fad..92a6e2057 100644 --- a/contrib/linux-kernel/btrfs.diff +++ b/contrib/linux-kernel/btrfs.diff @@ -294,7 +294,7 @@ index 0000000..010548c + stream = ZSTD_initCStream(params, len, workspace->mem, + workspace->size); + if (!stream) { -+ pr_warn("BTRFS: ZSTD_initStream failed\n"); ++ pr_warn("BTRFS: ZSTD_initCStream failed\n"); + ret = -EIO; + goto out; + } diff --git a/contrib/linux-kernel/fs/btrfs/zstd.c b/contrib/linux-kernel/fs/btrfs/zstd.c index 010548cea..706fa66ef 100644 --- a/contrib/linux-kernel/fs/btrfs/zstd.c +++ b/contrib/linux-kernel/fs/btrfs/zstd.c @@ -95,7 +95,7 @@ static int zstd_compress_pages(struct list_head *ws, stream = ZSTD_initCStream(params, len, workspace->mem, workspace->size); if (!stream) { - pr_warn("BTRFS: ZSTD_initStream failed\n"); + pr_warn("BTRFS: ZSTD_initCStream failed\n"); ret = -EIO; goto out; } From 01a71739b0c38ab0deb00c10d5e84683abead2d9 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 2 May 2017 14:39:03 -0700 Subject: [PATCH 276/305] updated DSpeed chart to remove 3D effect (#589) --- doc/images/Dspeed4.png | Bin 29425 -> 24692 bytes 1 file changed, 0 insertions(+), 0 deletions(-) diff --git a/doc/images/Dspeed4.png b/doc/images/Dspeed4.png index a0c7f65c25312279b4b25294cbcdb6ecfa51ae4e..b7baef1ff3f7adc356d69bbabf2290bb38a1c994 100644 GIT binary patch literal 24692 zcmeHwcT^PXx1}N~%9S8d1VK@dEGjt~00jXN$w|pMNDkdCCM2Us4vmtts34Q0Z)Vo4HEU+xd$asQ?-IJYs=oT_oPGA*=Unf{3Nq);(w#ka?AW=7 z5AHuXc8ov+{J|2P0zY{|-unvt=eYe7nY+ibTIg}$A7^YIXxJY+MsptebKC^k0S=Ko z_VE6lr!L3mht5RP8^uVB2%MH-y882_B$FiJtGGDA@6QD9PpIFowh+31oRs>GXpgTH z;T_3|#bbno+Jo2N5h26bx*)gFDkl$`uKmTlS?+@tRym zhDKfoF;97$RmtXy@2^gZ#oDMethwAd`Frh-6L3+|4*J*+Iri${d^oZmD2n3^OMiZp z>sTDE5{X{_>~%?2Vyiu{BgT0+LN!DE!>PY$gPg#pS4T+^f>vL@pJHfGh!N(~fvt7p zQj=pF)&m(t_!*oB?aZRA3UMkWB&YGe9~u$aG^=MfUA<)V1au<)hx((mibm%urPE=M9x_0i8 zs9+3i9b?uKEmV-(9GcEBa_;DHAy2Au817E+ARY#aZ5CILrah-FEDgligv)dY`%j;WG+ZSaP$r3bL)mr zZ2yC^%sP$!DqDzM?=uL@Xh7`F>Zxh2)eLy?3iG}AS;rYSaT-mZQ9SR`cIUKFyc&Mb zZoDS3Jzgf6wnF%4(7oIe&E`*yfqF5$5w}dTT1Eq|{a&TBS8qK3@dlpRZ0>J(j_!7f zT8RIL)LKS_Xc<>Has-!*Y#FvL#2DkV`e(y=4Lb5##iL->h1&DaY>u9%6`6YGhm7H# zjhc%#O|dryov|>BvjFbh!YAkLnAPDCx5-+3D+conNRf6Lxgu{JnUOIiUwM zBd>hP_2-Ax7399?e14}wftJ5rtj7-z&XeuM=ja7*pcO5fLoGzTT900sJ<93cAgcJs zQ!&oEbbRJd`SBH%pYM5*G>dcVmKVdFt*~F8SLJRrht4{j7dg7j-Ro^OmuE|+V&(L9 zXVg07cq%?|jR$!pmSHDj74$!IN4iu>oGVviaBcTU zGhtN0V>ZR3@A7R&e^^#zq^NN|p!=K-2Me;xJ4|sH54SX#^i1pv?yBmk-hA?mQ=g7F z%&CY^E33?YGvFu;vwMmfIwGHGIu__gCu0V}z|)YTqCX18(*=^KM;;4~^Yjl6qvG|{ zmq98W0m@bfQCu>v4+PDeZJGJQkoyq?$!qh1-7LRv&4-K0kg5#Pb$Gh}{ewejHYWv8 z5yQK`oRc>xEL8?|-EfJwOSlPo~Ysw{mP$3cJN>L4NCa$Qsjy`?xaIs<^+?x*r{j*u$@W2=qwk+;Dm- zCs~)KiZF-bR6h$Dv$XwGOEVlq)b_%lHJlUmBRdMK zya^U4A3bLlvQ z(035{M!k+sMTr;?oucx=_a&PkVf^z2y}CFMAmx#}b0*69b}B_+rNFxso>jRubK9^k zs6SBxU$5A6S5>fjz8!qHY`sFDn}RGitv6ZoK6naAC-qy}S`m9I+>J>479q1<9p~mB z%`~V@G^M@YOc4khw?X(o$=X@WeNt9P-Di(+>Nmqo#RmwpGr(5cG27Rjm+M)bi3~tY zw{n`zsQP}6@e*QZX*DF<d&^-K$2lqkwR|>)-EG!kJ3g&CR!X(kCEVBNEUPo%gr-m~y2m0P`Kdy0 z=p_}a8CvLzn4;L)Q{(eRPne4*26$`fgHdam&jDik5tD8PV)XRGk$D@0yAQH&tZ-Cf zr%J+bL2sk9H<7`q@3K4c^kLcI-+>_U1Tqh$i9;u9bi(U*rWOOU{_^fq)Rcs7nIz8pWxaFZt+Hn!PS(8zR*xx{y zWY1<6>GIF0BDBk}Sf>Fiwn?XM8(u{Y>*v2fRw0Tn{Bcx0`a;zT$=gUrGmVZNARCAN^ z;yzSTV@?LU)StNviF7yy3bF9*{x=)N)`LF2TH)kp4SYXjeX*iodGuh3a`c?QU=V!C zSI>}QDL!B+f1FB9`@`~A^?bK<#aVz{do+Y8z|G6DErJ&%kt4hOX1aS5!skQM7S|-J z&WrkMxZ?4E^k%G2iL~BUM_BH|g*BrnS7a;RJjqg*1!Y8wH2BpE0z`~|3VJI1StZ!% z=z$;Bc2SFqRiq1xw-bX@`hJ;fpxmAJkSZxc3-3LzxH8ZgKsa_twMoi0G)DwPZ=9?|oivHSP^pY?P z9vLt0E>WSK53Juy&&zBOnz1o^mzvoRl10I`+AK&GZJFN?JGF?eNj~&4k=-nJEWIDn zBV-jk(^P(asyk`$t`za>)v^nCr`@$)lTkL(#}upzO{fOzkRb$E@4aY+44=Jf3PnS# zlseAZ*L!p6s)1#%uDg{R*h0A*Yoi=a)qLYSOTj$%={#>>i$I>7E8blDP71$)rU=ei zy|-IfI#oOwX^|^r&1=G$MKi09&a5Kx!ms!DR`Mfe3%(Gc zz5S5gpS9?ORZ_%e6;5J*MIsL^X^`qg`=Nn`$6rzv6>j<_j_7=jjzhZ?M)6I!de*Zr z9jAvmo2GAyWEf%jRpwK6EymKp%UbUwwlS^da#9YC={I_seU7Q2&~=)7CYH-<`D29! zgY5qEARN|Bx=zH~rMm~HQ*aPi?l=^nUr^Y%Hfoh_HB_EGLtCHr*u?xScyg*LOpF(1 zW{N#{n~N&utuT!I0p!WF5#Hnp;k1mzZ=`J-igworc;*2t#MjR8>AmLo(TJZyB4=c5 z#&Bh%u<3g^3gqT})xtftauU|m?XL@!+s5%$_^cGIU9RFjWUmNM=Vl~A0sz?jZdENK zz1lD7uHSqZTqEWhx#1Jo>^b@cms*kBXP8Yw;~mdwBKPE2w;6V^7>pFv9J}YYmtkrn z%{3QcJRq-Dy9C;oyRp9zOxEd;&vq zKe#%8o+NWlzRA$v2<$(s6|Yf|<;wQk{TDd+6iu0%`9TSr&S6`3K_K%N8$FiA(?Ss@ z7nL=an@M)#WA|C9sY8TFbeo7s;~bOt`NMx}US={ZxZ5stgvQT-7xu*az~75<%u^M7 zI`gpu$#3lUQt~w@sE>^?|6W4LI4baI4$7F#$5Jn^_(xt-T!N{V>y#sO(aL)=tYH7t7itSfjZ=x*w%DB4iv+W0M{`kn+`u> zGZ+BaH7lDhloOfzdJjtOPu}FSDJOLWSU8sSdsqB@cX{8;-)2B%Mhtulhau{2m|?L0 zI_5N1?M;!_o?bp!Y&YI9-5#&(r;F~Hd4YESLenE(+Q{1CUlzCu>Qw0zZiL zE)sBb_zSGD8?{Tu)D>CEC*M%_-p++4adb~HYd~xSd!EbEm_mpTtm@sl#`U6q{dRFu z4XDoRiN@raIGd|xq#qCujwK2-d4C7GKmypMT>gI3cuL|xg(&_^fRy_;zMUXVD6yYP zdKk=N)NJKg0>ACRb`&^SJ@2r_x{QSxETkyK3&Sv-0mVZ!jkl`Ve*s#?xzJdZQ&8Au zxZEfd(*lt9Nsm|kp;mQ962my%3kL62s-V#C@~1QwM3 z&*aI9w@68|&NOEjUL+lG2lb4~n)Cz*OGZ!ihD-ixw0#TinPa;YjS-Uch+tUQz#{f3 zJ?GwCw0;1P4P)+FuKrCu&Ig+NUC+}S9aF+d|K3@Yv^NZD{N@%$R(mP_`tWPa=3))9 z8)pr+&9AHp5paV~HAaGNS*N{NVIL}l(oW)lT6fLFY?LF?VJmK}c{tS4u|(X|NBL-@ zsIs?BQDD{FjPwpzTkxpZIhhPqitYLQev*%+_! zms{|d66pjAP$avFQdCrv3RflsKsv4$pS+xjd9gM%J-@y6p>R01>bD3A zG|fGRnN028TKVuBpJ#ctOl`m#r))^JSKNf#0#(C zqce!qdCd#Z)LIGltzrVO8D*p_U=Yj;b$c{SVpO8qD!-q)m;m^7=QP)*S|6*;#Zin! z-bC&X_>q>|f3$L}{?Z45nACLi`xp)wH*zTefL19 zW#`CxR6g-g!%l5Sa)pI$$R5FS9#3)P=CIy5JBB0d`vGv9SJvK@9-_Svpu)3vhR_{K zPLvP9d7jv$5dYqBk1qgT=B7^S?+W3iiyv7 zid)7_4NWks=&lkvM7SQPcw($X%jqQQ>`E+~vq7>V9O`AQ3$HK}y555Q#%3S<99^PSxC zxH=2)a%&_+113Nw9*Cqf13H(YmUW2{9lxb|CLoSlnx)I${dM8V0F6zJ--TdUEacUJ z;L4);o)6xD{~({XaUb6&DNu#~cylIG>^Z5)>`KrYkgiQsiZ{o6^ODapKSKw#GBdVOUeSvT$1F<%etkGI5XsoJW2JV$vsif~KR5OPasbV52ZiQve~+YZxh`Om(; zasW>*c>%G%&1QYt8n{j{FU~cFa?Mr;NvMRQYI4stf0}jpqB}ReP(9$wF8l_py*!a< z&;1KhKG#REpRcUTt~k!D2x;(Yh!qlPkbl+p9e9J-a7KjbH)p8#UoJey-^awITQ2CX zx6d(YfadquT%0SwywG@#I$Cq9^V2IqeEQYzGtBVOY2`6^t8X4p+3!9bJg*IGV*u&+ zzx|)%Ik)a~-4H%tT(F*PtKKpDt}en+SZqb>X#vn2LxVhUsS&FKCT)#9WY+flKOS@2 zlzlG5*g_)iY7};d1-D*HOS+n{b?92W!ds^>Pq zR^?*ch9P+i%R${!MZ~kN`u-W4I5VJ9gTKKGuQ;^?eG+;Maoq$k@uiAxDkt%y9diL7 zxtpsWg%6p$MBvYm?KNV5^!r`@bO#{0dpxO!^3Y>pa5e{>q#wUmMDowU1b-Hc=+7o$ z`j2=qJblhI0cJ-&ik}P0Jwt^Vm={B_)nVY>G!6bX!mrb7bcRg*^~nRr97syn{4d;M zfxNsC)q(KU_|;jxec_5OL)e$_|&C|Kn8{n5GaG+9` z57UFYLSAC09!6kVgRE@kGiR0deYs1obE$ugyN6ebmrew!n-B7^GORnVJ%=!g->sH9 z0@Lx0?FAHK>jzC`5Xj_hn<2o1#S!?2QDnX!iA&}1X>18#)|ID8fB;gZJc2@Jzrmbni@L60m zx$ZB3ELMTHO*k{JMHmNVHC&JawsyY!HL18DusOobrrSPaGFu@QjYGis*WfDf`}7hh zC%@jesf8TXb3+!oE0L?fp(p0PAG=Y`%J8D@GG;qvTnoF^v(GXA_L%1vVj>Nxh0ANq zk8gTP`BEoRD2nn%1$U{h0aV8g9D+9HgE6p5UI8Tvq^a64ZhwK0@v8Z9N0Lg)fopD@2 zHq6RNjz?V2_WQ|-vl_`UPMTg8^e!Ww7{?Xzj433dvT8#IR%fG}Rk7Y*kWIs_t?te^ zUr5gM_>yPVes8Minwl~J;aQf&F<+jf@Xeua>Xl5@Li7VgzfA|ecF}d28x2K}D>O=i z>hsk}Hpc?Us?KKJjcx@#&D{ENrC&?6jM+mUFN%A(IGNsBNh~ducHaIH0NgH2f&d); z@-e|m9rGV}1zFm1@j45DTDgqbMelP@7#(E4y%}kjk z<3xcG!*xqm{>&$7CTl#JtGL+FWn;+MWDUN*jm37EDyaEuBX-w1ArFw)XPV$~gS#%4 zOZP__CKb^;Y3f(56iNUKJjP__xk8A_GyH@5Qw2w`a=f@j zLCtbA{FnKMF4_{`_9U=IS4El+=vnYenItA#ipp@d1{LuJG}18mm)2zV_JKkrCnUC& z281R>fUtCtDde6{i0?msKS81IduAr5&a{cu2Pt|#!~BH7n`z*FeVAivD#}2Mc*R<% zzH0Bq;55DNfPb$t$l`a60j%|?Pb=>fTNb&*njBa4Ji1R!Z$TZi%dp@g!{6`}Rem?4 zOWc04Nl*uX(AT+qMavR_6xqB_LTq$m464ovG@%rL|8XI2z(0q1uSpT^o0mNq)XfbM z<6Kl<-`tZEit#(0Jh*3_F5dx?Cu?b3U^I=5HLDMsMzA1%N%PASZH(Emc!!qyTK>5y@|* zSqgls9dUPFceKke#)Q}amOps=(2hhlZQ`l+&TDJ)2dL2Hu={%(ox|=6F@+%5D5yp8 z8c3IdyxgfI!w>=qUbi8>qs6)M7E~SMCZlW=4&d4&aG^Ti{q0H$KxfSTk&d?jwxD#^Q(gy+ES|%}o6ad4PC5)ipam++X#-OU zzfcc)&)}ajr2rBp0}9oN2le9X_y;{IYC0kM zy4?^V1(>l*&>8nw+%A^ z-Z^MW@K>EU=l5_i=TZ-=_>#A!Qb!+SypElg9UwEzIuahbw%>M~>7qPt0Ytf$b=ZJ%>rDyY?$$%7%uz1W}qk0MQ&} z*)X-oo*uBgEM)~owbJwt8C!BS)f-#&3v1YJ6nGQ<+5U7dt~JbdI#9a!)Deea3ZhfH z+kMz?UI{=mB8`w~Y5zm_s}G1yRGBheUcW$=T0{G7w3_%O91=@!SYU4H=E_W59;{;YDj*HntRsdc6qOW8= z@?O9FHc}~QwkyO;q~ktcaL|-VOwCzWNe9Yz;hB_pkPw4se3Kww6Vwto1mB*=0kozF zC>(33-CKAv+@Rx|XZFwUz^5Wu@(-6Q&>Pl3ah(i$eZ}Wd`+H`9vT2r1$tszQT#Z@q zV6_4!AdF(ZfOP5qK#)>$YKyyrn~o1`Q3q~_f9L1`c4Ej(1W`VnZJ+O1$ZHo`YJiUR z^bWwIs1j@ZcbZg)zX7Ai;&%tYnu{`|H!rMC$D|{`~#d zEoQ*oKDAi>rzF*1>5>O1p=3}h^AW276S{#pgd10vs{j+|?wU|_kQ2Yn0I+4cc5b#6 z>NnpCgzfExcq_o*scPJdz|d^SI2QV>5^A4@I(5!nxOo*DQV{^SLHC$Lb&7;_$@6#m zWNK>#iowV3zF4R+77IFe&F1=Y4j7*KSt2EW(9hVk!KGb%gJS^lfvJtlBUp#KQ_}98 zS)t0P>dO{FBg%UGa}p_Bms0Re>HqnHgef{gTNHvQ)=IAq9$GD{Dzq z{O_)3m zJ3&h#zB!T)G#^KE>o+OI*NPo*9VFQ1NIs5jQMc7;NSXQEsY(mQ}WIo?v^es&iF?TVO)^YgHx(~S5j62V~27Ggv%x+BxYuACg3*Abt{ttIQOUJ(($30_DDwLYG#uBqBse8Il#_0ByrmAlc^u}(5 z{vzHi3?jf^$%USKR8KAAy8vWtp>gAy!zIX>exM6!a<%bLV6}A2rjFiXO2}H|Mw-g_ z8>Ky#1wzrMp`?z(%V`I#SYI}RwhpaciIhNP!dS5%^>O`weXt|hW=B}%r{R^@jRd{@ ztzL$8nx!hx7}CKUDEbj<>84yzGl~+h(JciXdmYh2_8mUJ7U?z((0Gu}IH(Tgmxd2U zuAs?jwWHGj8yR~N;*Thg8-hU!Vl>vu-bY>Q8e~mX-O?AWK0$hWGU)iI7|?{84sghQ zJhBt;_fI(v)bWy5823ErJS~9vQgg}D=r|~CL9Bczk6!G&9#EP)E_=~`gqmD#+s5th zr@osw=xo0_g76*yv|KlpAW+6TAeM}CdJdQY%$mY^G4H#8-7)maRi`3!4)0dMkkde7 zW??@&2VMMA-d7u-mhL5i9HgB#@Rg{1P`4&SoSa$F5MYWym-<${BK#v{kLzyu9TVq) z5ZQ0C&|r1`{eU@WZrAoouYKPMI(LNU()Y<~tBLxd@Q5nw+$K^wAg^}G{aOHt>wur2z;~BSM4#)$(6RJJOs0V>*AzDEgmyi#_xJfNEUFD$d zAN2dHIcZscdwCoOxF*M~u4M5LmPg@jxKhY~YNZz~Q3$4PB9WWRruYE@*|4;kvjr>Ev@B6vG8QQ7{?>vk@;7*rK}&wNHAkgP9KVYtC!klQP9yScSGJW-yyboBY=$_~s~7 z%~HlcKX6HyqCf-5p7`U}-DG_bU$fS$hm3g+c{zVBZ`v+?KkY1RLHIR%Wwo@of;;NL zrHS!+)b;J0Kp5x6QD{(%@?fOzhwO0EOGV>OfWiC}NFuf9oeP!gEZEw$*&?-y85NB- zA65c`;jZJS>4QM;Yw9W>D=%~ObW*(JHQR&8RI8ir^tVBUzKZ67=j`1qP%0(Tw6owX zEWYLn#pQ5A8z>%nt`mm`A6zRnQFX#1XQk3%XQlogc;^C1}#6Z~!{hIxj}z83gH-tjVW_EX#c7+VWQ z?ij9rdd_*8{4z(DJ+Sd6(}sv_&*kN5Mev>xA09aHCRBjB-hEf6;v|8c1<&MTGnoekOR*@+R z)V%tVgYNVf7;;c9?7FARLZDLV&nO}89wnhc%Q$1BS2LL1H%6WEGDK+;%(3z;JWE?| z&VlmH&XR0W^SME(MHo>ssxNlnkq}XbGP5Py8@)-lWHjj@m=P5s> z>|!`r!dYXq{+Ars$O*;=QC5W-^KHQK8>{NqJp|MttmVyJH6T_}-r)|lexJ-TUe>2S z++8-jmNf?xHKRq2s~TnyGBO+UdU5Jq(g!4GV*~zRwAMTo=qc7ksOd?0dJK1g-o;>s z2t)ws($i;tlhum5$f5eipSAZfJFq=4h$o%rqUZHVHK_<9d%H#+=|;$YAL1`z(AiC$ zULdpRtCRvbc8_~apMBcDZdrkg1lIOhEA;tu;E^j9dfX$d5q&20-iMWHByyq$Jr0(@-4#NEg?7-9j=}-C}@hnl6A9ae-Z~BLLC-0AW|4`q$+|#Rx zp_b#Xj!xU+vKFX0Pp-58bTWk7*UCqG;;)RF7<{$f@VcpDL(3m==qi>@hp_T!*@;h0 z>p>7{7)9ey5PC-2eMU>uiJNss!V^q)GN!RF`?}wxlX{YXSHfI{KvnyNQLO`&?v;ew zr$UN##eFEJX44+cWn=Y~+8{U1U+PNY(w)n1W!#0uS@1>XUl5AD%Lne9jL4S*Z__6e zZy(WIYyd&AvAq4wavdi~2fS8hWY{5~0>e{)Zzn07@~}6a2pk?%V6H2BgwP^^gchm) zjsv@=8+|2I$mS}n7(( z_hnxLo9;hg4iUhl|NBFX{v3P|tp6X4jQUTu+KD6@fSPy!Kr?&c*Q2=-775=*ZWg)$ zU&H~(ga-Dg?t*Nc1oq8HER&kx3Cuy{UY62ON3Q$qkz?Pf< zYX4*a$3#H4R`M&IgolTf(}k0=G`Ya9Seg&qB}vd2J-{(InE?g(0c*7FT`2!T8rFqv z$(TdA?(Er5?1o@I9XN(V9*C+JD;l;9K##ZC>U2AsbcyBfFFxjZh3J#HLJ#C_iLe5L zonq#IL$z~x;O#AwmN^H8-^1J_5odsIQ_umX-I&gwaTnA|z1st|a!>ws;bwwzo<5TG zoJbC(aqZr{O5O0%lm|S7>$7NH63}`6jgI?qT2KMdzB0-{T``A zsE%>~Uf)FE8IsT{9R1?UEql?X+|F(MBDSyO?fQ}a)6euc&3j=qZ+~w)IUL~#H1h(Nv+}!FTHvgyP3{`26^fAI^7G%1YgHIXo|q|=TBfq{WPY@l9iPj zIO*J(c4>b)mHl@2qbqw(C@a94e&YrE-?Q8PL*`e;9V1xs>X_YEcRQZ;6vqw`cS8Wh zngL~Qcen+b{RKS%Xhhg9eBPLw7BM;3iQIF5_Tnvc$mOrnM4#Ny8+>29Zh!nHOQ;}@ z|G5h0-5Fs?=x$$!OlbaKjO0A<44HwsIvVv#PT2w9pjsH}UUL}nJe>so@xoD|do>3d zPu7XrX+$Le|GHsqHP#dwwABfO+m@u(cE*#CCV2qVfjxb#c0&8LBpXPJPaJ|}&rfqM zf{6(U5Uj^Y6hn%r+s;yLC-7Fqwn*&nK<#)Yh}*Cc$=BU66K*KhyL03+^dF#-WEg2k z72qen3w%H^V>XTOIRi#H$bG#9RVNupU+P@d&l*1G)eXR@_>$83@_h)HrTrd&ISg8v zW()4v?G94-;%f(Jk3fUgAU9XRLb(Ra699V=28Xsg>4+~Ztv3LzWNA1JN<~{pf6j|9 z1Twxu-DR)auBXsIw@e1*jsmQgrA^AX+(u&f38>CWLhWy(6rouWsw|UN4@Rqj ziG~(#W8SPCbsv~JLBSsmnd2yna`d&zUF?J>J}GxVeR$9mFmR0mM)u+Q4)f3iM5rot z+MWf!l+&^*Q1LmIgP)9Y7F^#P4^)JvA8a+-z=NR?d7L-k)Atw~veXd%g6qH?W&j1N zNI^UOgAz9;IigUy0`xKNF*nDGiHIzef$At9bRHPhX3v$G4un;%Caz{VS8&XOGihK{ z_`KKme>tNXAL(xQ_a>S49t>7UUs4-(28CSO+l-oyLoQPEcn}a7?c9vQ^#P5eJ2z-3UIH8xgS84xbWQ9stzzl zAsn(=B}1;mZEjLcXsT8z2mpB>K0@|GKo4-~*rBO#U?!Th>aQQfJdX}cFhUQZF-lLG zlze8JKr;n2RgLYgZd1MYnSMKMOQ@;O4easUXXHhK@gxu#l7|RKK)U)t*W^kRcFuMV zvLIY3uYNE8=;l5sH9y2i%zy=i#)lD;kh{$k#rD$`4Ss=>lQ2ty4%v7OQ$Kr$4V`>0 z0>vhYh9VVS{P0F)SocPz4a@@g)H0Sven2|&jst@(v!JehGav4w9QQ-0h1cn#AgQkF z_rWsOyzAItxjZ9?LgD1B9sCN64@Esit}$R1$Rdj1Zs`7bqNk6a0dHH2Ce?G+yFHNA z&X$r{3_p(!r)N}ry}KNSYLd>o@>uGa&zT|K9FaoW%z9SsT4hJxv#R^MfWv?CCWnZW zx0DLG!X?dLxrydU0+fbv*_HR``D{h_a`3_YKi|aktWtJj3*NL48#^s-=_+VGhNW7$ zbP4VAbY#~)=;%z$20W35X;Yw8Ql?Uvi#n6_E_xPNAEFpzPu7IM)HqfVHmcoMTin68 zmL#YxLFAs__2CzdTpDq6=6i!sU9i>;oEgz4$Dl!Wije(JL~zP+rat3Bm~lfjf_ZcU z8Zk!PX1D9oH@YO`^)HZY6#Bq*wE5O`ftV*_C zhG1Fz$g{SrmVee4yrkMTxX)U+W7aTN)rivn#3?;7Q)JvvM^RsOUHY@WVH$k)SJ~E! zCAf+NZ$9HyS7+LXK9XgFXxRzzPn_ClB27PO*hU%Fze>bXEP`*m<55I8a`uGd zs{J4dFj}@LeS4rPLy9DNdY6e2=JO$_Z@Cr3RA=D=p4jll#_G(PaIfxe>Z$kbsaqqI z6Z+h$Z%S(7x?BPToyVz}MRKI-X00K2<+`H*Ep8^dIiGLx!@-zAYcy=~7P3B;S`=(c zWf#W{9Rmin-m9eFU9VedaIR-&S5BG3$%#-G!#@U*@`&ZW}3T-Lmqxp99#6O*(YUNk{a9D{$qR9B1jT z%w3?%9>TKNRfH?k6ww>RYB^s{Z$H=tKS>2SHF6WGObz1wfq}SKXBq_>wa&jM{~Z44 zl428D`5m6R=%j%!>$+;;1hXcd>ub68Uf>#-q?4<|o1F%a0uG;bJ5c(jVk zjctmwGfR53hpmT(H0udp^!A$tbjkGfXIInXy%Y&$uaOm5>tAJ&4%RQ2cS#VBCGqg%e|MwxhcwVmg8K}f*!w+_a(S<%%u9NG za_8sNd-BZUOj`Q{{kvPB5ntqyv7SzK_{b$ihMMZtVyVq|#@*{>A-zL&^6+M`A(_?4 zgT{KE30BwX4et4T$o*B>c)|ppS;_C+Uf}Zn^485Q=E+tmMrcgyJa9z0^5ym(-i)Xq zd85F6lnj~w!yWRuag+!!VKM-LAmEZX`E`HE3?(WHZV*KVoR>@-^}jcOo|^K)ES2H# z&DurCZJc|I%HC}m9=gv&>+hHC2I27`##efr|6qD#&I(-|3!(dY26 zOC@%C;;iX*h&Sn`T5tUN^pCdyXLOE}hpsrH`Rpz>8{AvG zB>DMeR344>Qi4%*cG<%6>_m3v4R7W6+_6^ktB}J7CfW3E%fOmnNMX;7WBSc;| zfILdw_9lUh+P-Pcv?8Y}9W%Sm$v2x|hIOU{zqL_~;?6x?6#9Xt_;FF|x6lA?9ANDb z2cr+TBI&Svr+fq}(=ED?0>jR|J;e+cNXWS_A|KEFDhtsm5?~>JYVh~2%Rk%_Is`6&$p=(Sfoh4 ziQZAn&+uc~8=YhC`#CYr*x|vI<)OE0p|;;9O*_jTG;utAZ*vFc9E>X~m4<8Y_3LOE zP;w!*Zt@^eP8&lFSpye?i9U0P=;?;J_@BYy6IQnX(YMgT$%Bn)O#5W!LH0SK@aNQq zESP)rRiP)ct+D@b;@oQ4%MTfFR_8llKq3oSya`nEm`?=zaFnU4tlvtPRWz5OI@wg8 zrk`yN4cY`nMH|1TheGJUTdP?Dn!{;BY0xa966b=WSwv9)Tk*U7XaDUDK^yHon)sDg z-QR^BxIH(X8@51GS9U`bGynZVx|6ysYKWcvR-~a#YbFTg2T<60%``RCVyzg2*Ypoe$j(LxJ++zmcQIRD)Lv;oV3yV-*PD&jM z3r7$Bo5Vi}chpIS{o#MG9n@uSVU=`WLc_)BCvtiYSXh*#$bZ-m()!^MNi2D(o0_iJ zKgJ1OUD8VUb?m~EV$KrZW)a0lc9j_a^hrE`QyupIj$o=Og|2FU_-CzH|TQl{(l_MTT=V=l(tMzZ>yC83_f4hL;IL3) z{3_pY<1b&n$Xi%sYierZ)|FKt6YxgUm${?AxxaYv%#YD(Reb}4?3|qER>6zjXBUNW z@Np%5pRHlz<0wlu6Z(~u@O-O!oRnWsTbqd6L8!Te*xM$tDKjULD33Hf8;*$km%+gg zi&}SZZruR;@Oz5Ir002-7#{_n{y1YtTJ-XB^KX} z_5F3eZ5;I1s&CP!U^U%%r4slP(>SCZjj#I5I&hOeu4rj#sT&*H z=&dcDgC$FB1a`mpDK+(!q%XP!3)hbYS<1Ah`HU7xxV|vHRDmv{9bJi#S-I^S$#r8| zj`tRG^P*OezqAGk^jsA8LR1qpIWlHt zxfAci@|6U$Z?2=b_8e?D*OOJNrfpiK(5(6SJF&$YPd&w{PA~@zCdO zUX0yg$O(X6Hs`@ig(*m38k#6eo2hDXWKvh7_cwpJ{4qpRCCi;HLD;$zl2S_CW+nlZ>P~JUsl| z-d>LnPt)Owt+M@A#PP2P`#xz(3oS2kr^UQwub<Wvn$P!2p$kn83ml!D+aE~$*4yh16rVL1 zHFo`uhgzu`HUGIf-geFMT})S`Elu^9edxg?^u*U-cXzk_IU)6nXcpgtwCC8O^^)|g zFwMzdYuJ`>YHN&SOA87MzZnYP(mlT&c&@EG<%Y82^3sy0bs5Vwp|K|csN)*D8RpSh z1zyL@@T#h7!gDbiv;B&^v6>WOgKVNsZQL4wPaPa4u?vB|R64^@(4h!ANjYC!qmDa-{C1IT0+_BXSD~~rzTpAP| z&pi6QyHaj9^T}K@=IJE=?ZI9!w3cbFn(J zVL}rx*)?*`qH@a0)P_n;Y)2f2^d7Xe1e{Vv^?1(ntE;OsP9Cnh$QC$%*s>YiSh4jn zU{QMYvi@_C-dppH>5rQx$2%tT$w?ZSZAt66e)wlznoIL7ItmM69}>Bmn9zleI?QzJ zb0}MuHXZHXEj4&!>NjThnPy&0`9tGQv2Ad8cNVRViB~0;Ub&R_!3EVTwfn7Ym`M!vbHYQ}KX>hTnzHGou){oa z=+SnrMC|18(a-sv6GWo)bibNUb*5dp_ldcOmEggz4`CweWLyhXB8MxNL~1|!Vd3l% zyfmM)l_QJ;b01H`ilCF2c%d!a7OhgeGTAgKATv2|J9;9?w(%RYZsLW-YOKlw{vOe2 zrP^@e`3w9saqh*&rA(?K{n?+%R7E3|-Z(P#L#$>80wiDxKa@_Ah=#p&r>0%qR zqU=C2vm%+$gYoF_TIHl`k83*KisznAqz+Vhy(C27bpB40Fxt~Y{rAz)&!)*RgFB~i zA}Q(HLJnlHd>wJPY*ce+Ry;&fa+q#*mwE1Onlq-4axA{y=tJ|1iT`RLYJWy)Y?^BV%Uq z_kOG9HLo>~dGADX^kM?7$Vy(OoqIjp;*Yqwzf3*(>-oVPug}F^W*XE0FMdsID`8$T zCBD=hU$1h;3y!tP30OC(AIxh8snY5?{W*b%Vdw(;&sYrEzHkw6Qp=70hN3XjgiN&y+0{WOI~LoC`K24hJOKN{s4~HAESGC27IMyG_~Ur{IX}r=O0q;VLk>7xB@|*;#6lN49Tm9#7P}?SUHT*9Y+*#s!+==q>&Bn-nT-d8RYZV&9aha?W(OSFCnCOajdo{L}kf<;8+W1e-c zfL(8;{x@2Yg4mAN8qd316T)lJLKjlFnf$MZ#E!@c4%pr^b{ThS8%VUwUY&{c3}A&1 z#KH%(4bN9;r8hK;zc5C(;(3_hv9{M9w#-CG*idB!D=h`7J>*>INwbwPIRYO%&ZR&6 zGZogz>@HD$=AD3_If>$4?&TaQtBPV1o?Bgm@l4{wVgt$SO5?p6(V`2jtZn@t(cSNS zavLAf@fY7)7vUQJ9C`u(02v^Ro`^zj;$A@k5sO(RBK_XWoVDo#KDGR{v?AQDZI|O; zRch3Fx9jm|O}-dtND+`9x=qHiFzV$V7T&-3dO0HbcC|zAth}uCcJJ);Q!*94o=TfT z)Wt)V43+F+LsWnAaFfiw^J|S}`JCFix;csCEr~9x6b@vqr!ZzPS%{Dq`>5x#@9^g> zI5u%F^hzoJ_T2bLnbc5~ytOykrNx`P_-$zbB6y9SHav5+26=8@oxK z&XCY|*G@0`6eUm6m~V5qMt)JdoYMH^Hd*{2nu{huzrnq7&!~m&a8Y*NAgER=fW@Gj z(fKeZIoez~MMUd{$Q?BgAaJ9E`dpM9mBhggn6tx)M4G#R8EDbXvRCu z1>|K%1l0O$1x2Xy&e6IJTX{Ki2)ie9rH4_IL8$k1L6$!kIex1j+A5IqURGG@ih^ z8hbs-HfSz6lY66X*|KxKD@oZud)V+v3*NkeeRpMIj8#;eh=_)kR;;0HcH2wu=-zKW zilsM>e$900^K$0~ZgI&(4Tp{kO$^Eq(`HSv>X z%~f_2X{7j)GsXLr(QJvz%uJVkZpCu|oJ`U!JL0(S2Z}8uUEb4?C&b4!X#R#W%8ipN z@uw+NGRT#GtHSW`R)&0^TyRi$kk!GC-R_TXZBIX;BEoN>V^YFBt@^V9!ai;ZH)rvUU)v1I%oX)lh7`7lBX6a_|ZC=DNsJ0P& zCRZ*~o$>TC&OROWxUTx~OuhmClRp8})HB~`b3zG>tqT*3+aG8) zf982&jYKPcFHgT=;WWq2M14Dbeey$pq3&2}3QaZ!S#6R5?v;=zIrj#@En4Tm$xa|s z2m9Q4WhCz5f*`3};Emc^oVUXr%Cpg?(udKK$H;TXxN_wOZx_00@d2kg-L2bN;UluS z9s4J$d%+Ym`TsphNEbGG`~cz>_In5cF2(>zycmMaeqP1kR- zroOLC4Pp%_!)`%*DH(|GuD-6y&4fo-<7~EP?*ZQr%O=C=wU;L zu+EelLb>)Qc-m+1LPxin->&yhifxr2Ous&6YT=N4^qX|RKZRTQg)w<^qQO^I2vDY9 zIXgPGHvF96gw0!X^HhXjFqNn1dh?0-L1a5^jfwMFMJ;sRPl3(karv)LVbm6L?XmVt z`MHT73UYI0;U8)B%FLd5F5H-mL=WU9l_$h0Y9}v{$&_y5R{XH1sHWsehcJq4jk$&6 zlCVp;lJ6wXSv$!zpCdQ7tuCRt(aEE#`USDIi)F0H);;%}+U6yo4Gk3$G;`kv1)-&} zH+SFrTGM)MtK}CIkSc$g?@8k&px}CG(GeHS;$S^oq8!&%VYulpnz}f>>nhs4c5eQz zWk)tZ)!b8iyQX&;3&1Yj?@-n_K#*et@PBTeBB)SB;?Plex#+Eu0RZA>B%=bZ3l4{< zgnW(MMBFZv+^{yfVg^AMGBwFtZy)qr%QP2pcY>)%6x2Ecg%zu%tf(_Li!9C70ipZTWs(@^s;gg} z_CQbcM&=vUIkzPXJ4Ah?qNeWJ(tCUVa*wbo;(SSIy;7`%7e)+e1zSX(Oko-D&-y<= z3zq&+ItXlXEl^6n=|ATbWI(k%+uN98Jo$W|B(y{E{Ykpo*np!xrUCna< zFIA1&@c%Z?_Ab0*i~doX?eUmUd^Dx&sS8)AF66f4y1N70+erzGh!00EKU*Yh|w?F<=o znc0uGw|raUs9bhe5u!C}^7gVpfkZ(1h#z1G`$U?LOJit*~20 z_&z_!T{;T13TMY0dSW5+N$8TRq#aKm6u!#65VJt?r(s|jW=T<-?VMftA44gVCM67EKZe-i(~o#T+X>OP z3|33k=lHqf-RV-9F!UI)pkD^JLDj3tyM4#6?CGcQ7rk|BPv_YfCa$(t7}koo68ZJv zuDY6silXw5e+tNQcKqHuy(q5t+xby@UX(OzXQY;p7f)vS`CIe~I3oBNviryWGV)Iw z&k?_Z-Pw1TsiC!EOm5f#iNL|D8^B1}Sev6M|I%~6z>&{!N(f;l9P{sT<&VO+4*#)e zCScp1a|`o-+&;Z&A(nhxab&j1-|H#ly7XY|(x10OzzsfBuzdR;$L0ywo3Uw0fBNd{ zTZ~W{GI8Dvf2qMzxNh8ZaQI`X#BYF+L*jMbzt`)?wT`#VpJ~Rzse)A-O20y9E&dnke9XSP3ox+X{DFyQnxH14r$-h=Q$e>Yq@4x+M1+Q!F06hF$5y1 zUj%P*jeSv$BL7fRQCX>IZEa1QMDg|7N3*i#s<7uMUa*Wtc0tCeI^S>}U?}92mBYTS zN9_N)N)vWtwvLnV0^1#9V-}!}PN;64!N!e(xs*SXN^cs7&??(IJ9^Qk-$=;aAud%r z56Hl55dAh5$0*oIv9hvKT}_Q4Ha1p!H&O+$g}XrhozbYUH%w2Zq=uTcqNtM~*mu6bz!~exBMb!8{4jpBpRjG}@!l3}; z(7T{nYuN+}~r+4u`CYL}6<_ebnj{1f^?G=RT z`Q6Xnz?bK~@-$C7k1{D8V~k7&vqZ{oNy?TcCMJH>9SpJTKaEQqg!s!r zMirmiOzG62adB}`rq5W+1kOVXFR%e#>J8Ar|HU)KWixCv>h#kYa3{FTCx~a^x2YIW zhtC>N>DGd6N1iwR?!o+CB{zTt;8nWx(_O@T_Xzl+Vw)HQ>0ib}hqk6?%-{v|Z(!rY zyCxvObp~g@E?~nI$UHUtYhKXTR)pb0y&U*X59Es&zFImVtODO^n5Bt*^&2?1adpSU zpFSl+ZpPv$QR-jR79p2H*u*Ow6{n%i>ncF)tv$-W=$8fGi$0I>)Qn8ca@yv%lw}V| zP3`JTHpgI;h}U)Cmp+H6x{o+rU0waXz4SlEZ0ug5sbJl`L_iqfIa`L8h42=KKZJcV zEzp-M!I#BO`@+o=#JGuwKfsj&V$B!V|Nc`G-@d~2&o60_ZrhyUi<-+}agx?<)*CZ6 zGU6`NvV(3?DJG zO8DCAb97<6PZ0vMpg^I$ql3UsaTP2apWf6DTp`}gEP{#Cd}50!+=7VDt0Letvh;Y0 zHc>&^hp^j-XHtjHDzr?o<8U-DlBb;PK`?%Cgt6e$9xv+fVc>+8D)le z{lLycS(#8Gl;tR&U=y%_Zu2W+pMoi!8i0ohM zYA`)?8>mVj=;hvno)mDGet8Hv9V<7`^WP?yc>NCilE^M!L>HP%q%CSmh;*zDI<`WR z<0F&a*k}yQ{Zm__O+So>Ws3oef1QPj-~|S;0eF254vwty^8Wa1Pc)ET`B{Q>OY^Tr zE(*f3B)KmVj7+mOF!g?fgoGWkKo0LM+HoW9*_Rsnih3?C$?ve+8M`{H zF8TxWy@8Fyw5E}dHTxPnGggSb>clp0HcvO$MEu`Fc*=ngQHKX?Cg2Uf17Ypb(1 z)Yo@6s?g_f*e5nMl~$wrNPKeY-pr@4a{D=j1pWu%Tqcc%HuW1a?7)?hb85@ozJ0sA zW^^u!d|^+DvU+X2Zuo`q#=vzqGkeUgC)omF+!>jbqp;Nf3ozd}86m!(r>UjY1`&b9 z(ZOE1X$y{C)uRN6VC>IGU9h+Q03`50FyS)wWAWGJd1 z+?7$o4A#`?H|t`qSL`aKm6erq8q3Se8FrS07m}XC@Y=uSQhBq=LBJJ)grtbEMHV$b z|CI7vnBcopXkd%1on6@?o!A(U&@bb9%$kv;e+UhUll6S}^mI<(vvr13&mlZ`x8TaE(`*XBdY15sse7yHU=(B?z~{P@B2-8?kB|paZrWtR+v|#(S_-V zmZ4&Uxgi5vn%;m>+otjQ#wmXS^R}qV^Lx`DL#4^ezZB|p0EOM@tq0L8KXp`&_j2!w z{Ryg11ek?`%gA!4!lHv>>ju|8ozc19cmj1Ld3hJ!TtPo+aIPDB@^k7EPQsb7jaCXb zdyV5YoBQsBbM|DGRz=ZXH?CPYZe)2O) zhw{q(+R#4+^JYDT2pd`0$mJB`Z%^fEA}$4U=cW0^MT}QM{)mF1Zeqo9{6&UX;4H;Y z{C(2b_Y_PxZMyp9fA5lv`uG5GpP0wjB>{Av&@7PpcSo-L*V#Ycgb&lg=Hlk7@RtXH z+wv4?|DKd=J^|k$x|2`(?+$(2G(V=b$RAgFKvdWqS{TDbUI+=%lYhPmA0`6h5tYnV z_}dC-L}<^yuKoYAvv-zmrXaI}O=@XlbMB6{y|qGpA!juKs@(ybTHGX+iLkaMOFsH! z%J=MlVS%)9I%ni72vZf@+)y9P%I>T;rFHBE&H!SR8rv{P{_1nbP{*3~!i@W^0($(r5W} zemMAFuZihmU9~}cf-4UsCl@}9VPUV~I@;YB2$Q)wd}dvz>#YtXe01s6=H1E19@e9SH`IyW5#M~EuSJ%@io|yB>@tMIH8ik5RhDO z*|aKXYW3&@r`|RPDrf;z1hfGO~nWh4uiC&Tmaj=Xa!3-BQJ#bxx0XDs7^l9z<1?_yT~}g>R2~{MRt6l*FOlJ7c;}ot%J0) z9z?obg9ez$_>zi0B;(B&#u==Q*Sts8pUg;G5ImF)-i+T*0h+QCAI-zyxK$SI%du zq-~Ktp{Ul;)=nHB-70tyig~UGiKDMoICc`ilzF^XNI}Lt1&7PCn!jYb-2!gbomFsx zo07ONx*?J;(PrMztRBd2Amfq;XoCDp>%iIz+-;XrGaJC*^7m&DBkoSJ00t+YN>p}= z8{V>|hH-hVIsnp5ZkALdnf;+>G3SsDH{wISAMA!O1I8=h0{O^O7{#Xm^^4X&g~!#g zLGnHHl_n+>H-$z_2N1gIfVP;>4L0tW2GW zZua)wXrLU&EyLu8F30*=SQ%!hK2WfzlgA%lBZK(>R@fF#YJ4pAEX&KV4X z5uksGpg$DKP74cJCC%x*%*c9r2OX8Z{;SGy9HJBrjrTp%Ur9Ch5kH{|Zr|c2apo9a z&Z*2xUi%1{folLb-Sl(dP#8U!7>Vh141yrOfOM1W3{_M0VN=d+_H_Z0nf)kM!TymR@+>A~;kOIgU`>69Tk$m+*_a(hX zPt=&}Oqja9J_n>IUp{~Syr|B zgx8OISFMM*`pl)%mhp$JOB>=I4Q57&bL><5D8}=|?rkk}_~Q|ne=E1>@9Sd#wz#Vn zxZ+D`VCwhEE=c@H&SZ#3tQh|nu;FuQUM;JlHe=WMl^2!Bm%>tJmx7qW`Br+-H3))# z_l%%xF?OH^0J|a(lZ$NevT_w9kx0LwJvbC;VyOx;flx4ks#bI7a*mXT!7`VDii!7U0}@D?Pmg z2adhGg=AZ*`}#a?cIQ+tpxUv+Fq#-#99FRTsaw#ui{R*59@pXb zdKz<%)pd0S+dW^)?serOiHK+Owc`Q&m-B2NO*W4IAoUrs%C*qyK0p7LG-Tw0<>t1& zL+#Fgzw@P_^-$}Y&#}kby*!&Q>sx8ZhUaKRm{m5j&cZOK{uo1PIJ&F(G|CLyjeGxI zncvq?6%%BS1!5ZSA2IEZQ`A;2x>##^Xl?R4aSS9o^AQ6^M)xYs+en2Mk{yyY?#sFw z`D9BpWy&Y4?u!41<#k&4rPO|zbN)24im-4<&a;h_TZHcECmel_d2fSlwBG1*?9I;0 ztIA_em;bpoPSI!b4q36IY|dQ&mlhBF7c5UcGa1abDIYfF3hV1l^r)t1m#OToe2r1N z9IEn)snBNMP{xA&sbmZ;qL-{qI%-N+BMp``acZf1`~J7(_?NWrZL`23vpA`p#7qr# z60UBvg@yq7uOGz6MYtKh*XazH2129{8(*AqUl#A*Yw$-=Ma@tBpZ_sdUteX|^{+`s zH~zhrG=KZ^|8BtlyCa}*Vt^Ex zfmD28ffYq*e`c-T(i#~W(#|umOXEVoar_+l`O=~yTn0v%Ka$e|<1bPr!F3IS#uB7! z2+lqc6SJwRMSSAxmoMb|J=(-ZR?%WO2OSowToTTi z>VPX7dSR?bN>K|gm4pIN@w=AX%pk z`t{B2QH7$g(_YunU~IZ9U`TqBXcb4?M+RfGDXo3LHwA@Y4Lnl|7C5YsL14_&sBqZ5 z=3w{IqcyF45D#0b^T+53xHk^>zR{StKR|?tDa9AL&Ttq3vs4JXeYr@zWA{lwMx(^i z`(V-km}>`1+*h<`mGdmwz?AuWTf5xyP3v!VXr zGhbC?_r&daOmacRcar83f(2nfuEVF!JQ%aAw7}_}-YqEL06$P&>RY`2kNWb}GqEg1%)-qF!$3e`GZRpzV*c6s2&8ES zR`}%g>jK;d04<4p&Z3dn8)#fWKkarjIGOl-?t*8CZ4sU`4!CIxH(zwB1pM^p=af1G z-`ymI$(8F%=YCBIn0@iK+$p3Z!{9RvGqW=C2nL6T6-`Xi)>m5cd`K51;N`uLbNI-L z?I()oXy@_xaWTk$x|dP4E=L$q?+*jDZ#51NJRn zkbj8&2lQGVw2Ce^2-*a9|3Obn{EF&oMI$4c8`ZK*W7`N{ux&cgYU_C^_R;5n0MvS1D%)gq<*Cj*i7M@u_ctLlS^9z-dC{J8LHDS-Vy zK}XA3qGALZ7a`;+;|oVd4FszL*wLFxMLWg(k-*a%fWLqAk&R9tX~H!}0fc17&xR)4 zzIcr)B@CMIwl9MV>58ub*fr|1yqMp4m|Obyw=xmSlS4@5U_RNm(U4+_hMfKMuI2MI zW5hxl!9wOd@GX5uVLV(6V)hYRz_o`BqFp-78VAC!-DUuY2xN;SGp*&bXQ+Pr9s(_h zcQ7fQx6tLg;k1v1;PHh3_!Z2t6f(yUn8cJJ2skH$9%-4RAFM47zU$Qvvgs(49u*Z8 zM3LhgSriyV{}X`bm{7J%1Ts}u_`V=b!>?h4NS^b>_kvkq9m^l|(4x`ki)G(M)B$2& zo&McT4QQ}^VXJj|*K@lDS$C4@z0iGf{juENR{opo?}<8=z7%FxR(`2KRH|{jCN|fW z9TO{U#zj!aN57Rxa~42bI|mp$66LTUBHei}XYQn7%x417QZ&zC2w;fPA<=r5_dt2A zcA3X*_uGRxCc3?#WY5jEtM@xExUEzUOObIee66v|EHXr?tAM5>S{E%hEcSDZ3BlHD zpwqbOweP4QFkr9JuqVDW?ff{G#S?NRlS?&!a=(855YXuo-}kZI9Bw+&N_CsRh}vH& zoGb6(i3Vgn!p!v`&CB_!T|b5R{$i%Ome%F}d!ViC%s61*Mquxmp%IyAik@~RExJU4 zLiFd7%|gJGOfC|*%mD6^gT2L4U9kW4=`N6TAtFk3eRo%12avU}wDt9Iie2Rv>i4=T zRyq)H>f9whjf9auDKX#?Sf2c9%DHPTywasBDI0*OQ~R9s66GLo*aqCNGLDAfSlR<2 z0NMK(!q}L*aP%^QW*K75?K1N7+hXo-a}WspWq?%x81{Ow@)8wy?weGHMypqU^cfdXh-G5T)E~AR@`@nW-c(mn9w6hR zAs*3&PCZynx2{;NnK|dS_&RV_BW*th_^O1hhK7dQ`stZQ-9b7zbI&z_$icjYkp+Yp z__&5m@_7TkwXYiT@>&msw>tT|P#^%f;CGsg<&|uT!X?i|VVaeGj2vT1@Q>TTI1|Ya zxk?F9$>)?-N>omWP)DhxihirFbXl7iWSrky9k1(}cJacD!G9rfNS@G7)h-(S{9u2L z;S0|)RiV(Nr$)HQ5*t&xhbj8j*3R-UYwo$7u$9s3Q15}YiUF05f@Jp2zv+D`T^Kr? zmgDAA;-)f1Hcf}l(E{l6-SPe-diEJ7gh@8%oK0ZS=;X z>e}LjeyvFyPs}i(yhf7~Oh&Hpf8HXgW~TYxv$%6_NbfvJM*oAu`SewX?9gG2=$!BS zrE`v)&j!Qfx7dZoDwA*1Yxv!u{l=(jGZJlO*FMM4+po)tQB6GmA5;^Fs$<-One&#~ z@Z$@~lXRDgM6;VhmN1Xl18?me`uHj(6QcmrP_bjEeJYR;j;W2AmHmF=va*pedf=kaIByvgk!jlWl;Kht_BA-c7_k>SSs-Gn;b*1%G-pvBjppC^x(d$hkX ze`Ua9cgH!&+CzU=!NjCACPp>-YsI~+JaeTStylKrflJv7y*B8TN0HNIWLf6wl`$%> z60~TL`i?0`k>X(rcyw@=mY0q4S>qK%L_|6*ApW1OnA!MD7hhW|g5(knoxW%x1q|Qk-N2lwV3`gcd@q}``;O4`=;Le7UJ#i%u;?D#}haTjKnTALsmrgD36S(KSO zxRy0-Cn7UEr1G&BV}j|&oG#@Trd?4b&vmkLEh0WF3$e(24390-Vm;UBG-Yf}wfI5s zNm%AhsQPM7@j>DAk{|+tKs?96JWy0VeE3l6IPI;8EZU$=4bpvpB46FNjLP;=kNDnR z3AhW06!vS*O3njM@||+UBi1I&Nx76ChA5Z?vp3P?+dkN?cYJr2U2M}RgGuFMnO#3> z8P96`;0>y(^sSz|K6RGkG>O&n(!;N26THQ*xXCDw!rUed911QOIuyE1j92iVmyh$V zO0LOS;j?9gN-bcRPo~<8vZn_Jt9(TJ{HmvId1z%7$?Jty%+T-)h(M^f?&~uzT5`LV zLENX2>bWT|47-Z?#}M+{?Tb_;bjQCw-!D6WtlwJ+{CsYk*@fpxg`*N2$YMjMa0jpV4J)2!Z)n#De8^2Q9s z({Xqa0cY8~>u6=1x*--Aa&hnr#kV9Z@|`v4J=1%tIB`7th7ac{mj+oPHEN%hYOt^gl6ZA&Q7` zgIS<7O)@%VWJI6eUCYSG=neWNjfFl`*dPSxJbG~eNvl$nH!T~jDGbk&Ob)1(-HQ;@ zJ~~=Eo*481zYxlzUzs#4|FMFfY_fUT%s+o2fwYs zozLMy$H_$+qZ#s(ma=YUn#I+{eM&~oL*ktKvJune`5^XxzvECL;s)6cf7WJky_V3^ z+_&iOAyoD0I7o)0TW*y$^|MPDHJ4}v0*=~fQ}tHA-;bY|?#{C@>d6#o?*PLF>5TN@Vl z%ZUlX<_iP|2oGulnXWvEz;pLS%KM zlg!DS7N1j@qQ2hi7({(H4qo%%j(*4n88K|A5<27RYe|a}05Rb2s>h?9d0Gs)r%qMB zck>!Ha!Dv=bWD1_TvFHlrBC2cJHz2TIgR&fvIDIKS$@ffnAdutu@?^e?{&U7!tG>V z{&df_UMq2U^4^gD0!?oSNkeQR;;!5DTIs!^qZ(^sV_SfG)M&kHe17{w37V$Lb%v&= zGD4TbKF?g8FXpBeNN^0FE@$r^V2H>jmrZ29O>;VgHRo*TXC*!hg-@(GXIN*&wem(y z&rY1LD6P{tSA6W#KU&)}mU-})vi&R5gm{*6cW{f-;_lh*3H^Jd!@@hu6*)HO-Ugo} zy^1x%9wsCQV>)|mePKcz@^eLqMoHxz$obMXD=@nQL?Q zpcHO{1{ zCf`2%`%-VEK{v_d;jq+={WUA^`GsQk?x>9)_^TH-f+Sxdw1>mCr3W*vJ(Q@|6oUA%8e@{}{VVTiIJ+{)KPIf)oQ$j3VP%$J zZI70*k9|}~5UgLjd;W6OGO-<}k7PeB5f&os`alO3!Q#-_n?_@_rq>dYygIHiA_fW!Iyu5&+A-~H zFmoGG3j28iC$LFA>yB$r{Km@8O<8QnE0zW2WhjY{IdTiL`f3SYMQY+b0qan+Sq}kbmfXci13MD@X&a7_r z1bNIP4VR|q(eh81_T!_yyH;|Ko+PZ1bX4PXltz56yF4+$ne8ACG9$rp0s#BkNIms? z)^v%uG0;@rokWQl>^(np+ANi59EqLf81;}X^L!v1?^v}XRkpl+(N7Z8a(E)50z92K zYALji4%-}^8@y7Z9v=Hy#o3DP9_|EoHxRL+9zafu(=v9rGKlxsf8V|TmQO8-_F#7< zEjqsX&`p4G!sRAVOSK}u<=z;+$vT5u5Bl6qg*bO6cH%F6oJM0u(LPO)=oTEWC4)p# z39hb1Ex!uTPN0zbQ=8AR_-BdX#cG!t%-o+ee{LS2>;{&AOEn??(tpu^NcnqVK|xl1 zz3A1mTsFLq@2YUYokl_bnO&%ax-rke@wD>N#7 zOHx>Rb7~%c7E}(gOqb%Nb?6H+b^5}tiy_(td$Or&0tbhgf}k^t)6QZURf4yr-YX+n ze7}0c%t9kVGZ{g(z|ILm$_Fsbx)gf*H-zVRIB+CmIq0oaO-=uM}vGu)n0nr;`Y&yOL*-y1};Nu8;X zW)C`B50yuTNb;SY&mXiH_bC+CSt zV=7F-nwS>g(fWswxLPitP2+FoW75i|z7Ue1da};Q0p=Z{kcH{Z6hbH;eRFs~2kPL} z6C%}!Hii*o^W=TavSp+2Lr`%eA{Y3fe4Nkj7)lbJ<|P)W#*}effK<1jZvbT#(5vHB zLH%^dIuZrh0+DKwrBxjKw1@P?G$x!aL1eugtg(1Efup_|g!34o1A&~J<~NN6Hm~>$ zYYOtpTio#NrS(2n*}syItc;9XUs2;~-;q(0gxq)(m&6XL`o9Zaa1AL;zt?r7YwBFb z9#8ZGi`tr3!F==^x>Woy2mG{%JLFA#AyV_1W&ql=58s%pdwnj6j^R34IQ z7#eb6%D`SU0Jzg-3ZZtekz!65R*8CbgZiw}~oq+j_z z)o|(P=yV7U8!|z)dsqILZ|J6H%k-`&HVUI*PEUsfBL`dcu~6Ob)jEr%!TcNssF$Us z7P5(n<Aw}8`(l7=FZ`sQb>wKZOA6Ad7JdS89YXdRX+ zLN%>{N2?b)*GYqreBWxF3X#z9eHFS-mM#{O4SKbX`JHJJCKOz{^icK~4W|_7)j2=j zOjJVB^E0QB3bTK-B)rLUj(4m}P;Gyw)Q6kyXgzQaa_1N9?-?0Xx^3Op`=k0i(+R52 z7njhBv42wM3u>~Zj_v~&8=w?vPxCXvvx5Pjhci41)5ng+f3%#Ui^<4fL~Zvg20oU$ zcP~>|!g{EPVXY@wSmPY6sAcy9eH*#Uda1p$9JCQhxS_KgilQMHvKIFiA|J*Ns1ED3 zoe_f~(EC(R#%k=|PZR<1m(0t_Nxj*aYHek8Wr3etC@YFfWl_v=!TIq0-vqucbm$!w zifg1ebR6(3Yi^qNevr-%&~`sT_!}70SA1zZ0{TF$E>FQKS5n@bfnsg}Rs`D3ENz_E!7VMlAu zVW@L}AA(@!9ZcRJ2k2;HO%qLzPm~)6jetGW4tHcohnjc3x`MF3(Q?HmO~xhln+k8` zjCn_pa;fSAo~8qP4ah2=9D7OmXBwc`9U7Cans8NmTXpHX(jGch_cM));b!qx_l=$x zkBUJ9&@J}UP}p(tB~(R;T?Ix@cxO=KEl+&CFVt`&Q;EET)BmOmMOl1I%-O|5kcxIu z75|r1wAR#Eh*|k?=q`L%v->|vQCQ7ozSb^Iyj(^R*#l1lB06~Ll(r*Q1G6VL2hzS= zgrcXhC&TuCPTcWh!#{g$(#y6jI_-6dr$ym%!nMcxOC2O>U>qOLxI#%cV(f}|U*EOB z!7%@h?E$49bBBq~>c66)IN0s<{uEYVntlY=d&@oJf0#+hW_1`_#S@DEP8Mo5fomvc zxPeSNbW|598nQA)Q~bUD1v;|1QVExT6;p~}P+Wvu=*m2zDEX^ zd&kDOfK~+WJ0ovzK~90$vi@b?NZ4Y0`Nib?wb<-d5v^wy<<#giw3jQTUo!zfj1=0z z_l4+|W~N?AumpNvfChMcYyzkKl<+Pzv}=aggTX)n1LU&mg7A=OU|5gvt>;YL-58F-};#NJ_poItQ7{LCg>Hr=z2Kt_) z=hK9{x0SCW9eb(_uHX?#X2TG2?8rXxz)eVEBgj_@M_DL<7q(jDv>YTU`VCRuWlHPe ziXzoRvTrmIVfQb#=65boo+5~Iuu6~7Qa2kz2Y-ZdixJcS6gWXkpNLcp%)-RRveL&h zy$prB4r()0$haB8diWkfSxkufD`_Zm@MxWp+d_bw5)klL)gX}Pfz0kWo>T#{;NYO5 zcRz{oLUW#LKo`Ehpe{z_xtk!wMHG;-^kGmX4=PT1sCENogurDt%*t%O217W?sNjRc z^q-unUobu)f>F+ZM}XRmRSN9?cm|}A1rF+0IKBrFo1ze5;osB7evkn`@w$>N zD~>eml%owIVh6GT%AZaJ&EuIs!&)LpxDoJmfFzfRoK~&En!z}kX(Qg&j3);5lOT2= z2E#P=HY*1o$!6bb$tgrkCL9Jt_??}eBce_m|2p7+j1o1d8zFW3C=0oXULSBnZd$|R zfWKTVY#0yy5~)|{aZBZkkvp7N#+~f z+sKZC9|GomGZ=SeT?wf*gA6Fp+o|J$A%C_CIdIJiU#3t_AXn9=R2RYMVP!xH()%>z z`F$qnD021|w3;4P5|f^Bo_cO6QgQXKKy`xDlsPoQUkhMxHAk`|BFg$tNtp|B3Ky8& z0atjyJK%W0)BmfD6$0gnBUFZA9R3)EX>IRgwI$A1um%WZy23Db?ZHSmPYLjBK;azc z@jXeT3=SK&A1N4|_*~#J3gt#nx;4G=EAOyhP=%YDyCjynGJ^}5_kV>lzgS?TZMUD* zZ6JqWAxV#TV|RI|_R7l3lT#6%JCEdwEv7#NpP(xLo7cAJNfn!~Kf>+$pJ7`Fcw$D= zlEjdc2Rk_*31;0Mv7gC^mhJ>qh24yd9?Cu&61%dHIp=`_?aY|Cxb_fo&aBt3 z8T76p$!0^>45lX{<(~lKHvhr6kj)0Wd-`oM$@6F{EuPw@<|d-P6h>!7&yHA^#)gEP z`Gy=Y@LUX%*%?i1xouw$_O>iq{PD^keY=WYTzm;%t>Y6Y919*nsgvr1uEe=y1@Z2`fY3x-J@_Y4fEtxN6wFuoDXPZBbYcpR4|gmR!ZkJfYjrtroGvbmzS-3k+Tnw;`v zZEbBglee;JYIHG}*P*0q&Tf7CK9AEALc|2IaJ->@anS|$jH?gLvs&bEv5^X^`x_oh z#jHxNOpTY=`r-Him>xsx_^$to{uRK~|t_<$3~Z}{12^@MxjrRz_cAIIOr8P23#C(z_P7y5C8}08LH|xK2hKA;T->| zaJ*Q3Hsva4cd|V?R zZP^G&0CtMz967-WL9MYm##Soa#`U_6hhISUVF4v;_nV5`w&oN5Cah6|zt=IL-QNQd z=%*$2`y#Hc&RN78vFywKeBW=ohyIlPo(7_fRsHTi$c`A#@B1+U)w9%r&lNcPgddc` zl{zfcDJ}l=?2SAS(f{92Uh?@jn;dOYqNYPQX9qyyi+|bQskTrfYbRaKMYPU&XE&%q zkhSD7DH^Tf^SJ(Y)|_vUxIJP^#`te$>L?2agC(1BA;e>Y)7|)e{x}#mQ|=$O-7R8_ zrDDAOefBkO1J+XWsf_AT^Cq0i?Z(HUt&RF5$`zfFi_;{ZgD%zG$zq#3R=LLFO?J3L z80=9&+bHYytko{qL#gve;JI@qt^COT7ADy)A!enpfgty(GT(5+Lnmb)f30h3=9`^I z)Yq6vb61uJCJj4&c!@;@eUxI+YK(ok6Z~U>rQCY?yS3BoQf%en>7ie3jQk5hA3j}{ zX(DAZ;g#U-ANHu>>VbIy;;{d~=e=j)ygLv!%IW#}@^F?MU%G)vwodY|_$=3_#~nGD zN+M6RMea9S+&2E(6Tv=2u>QG3z@0deMCY&Z; zwcTfx#^PUA_Wnz^FfuOR&AWtDvZ8w#Z_)y}cdOIK)J|X7YGqWP}6C zF*j$URoib9i7{tqA08Wm#7$%O2p z2Fum#0tvK8g{t{N2ct!>-<$uVs_Ou!>i_@TD9WDMt}P-GBDwY^*?UGsMzS-mNcJAt zdqkHcglo&pF3H~6du0AUM}5EF-~V~;b05!f?)!Ys`+U~>^?tuz=WSb}0sdi^9$S3% z4=-i&mX5nKYds&Q8JztUM>7>-!h@gqig&RludQ4E)tRQ~L4qJYudGwYy~=TYsr)*&go$TLf&Nt>sA^?~$B z8B3WfMav{B1uudfMz5VN?%G?EEpNU>_PVf8)QRu-wx=^Fu(*}%N2JjCSefsxylI|) zfjss**k#uUq?V~bDY5tlZd-*a zl%*_*p%F3UVlO&X2~`|kptWavnir2Iseaq5L{p?q`h?k)WUOT0AYnGAIj=dVHnNUe z@5Aa)h!*%TxYW>7an^rJ1=hxv$K!legH-rgNqDJB;?Md={uLUP;fayDFG?1HP!q(v z2X(d3?h2>Sm*@HLC!HiiBLJ46_Gb1fDbP0NIfJj+c~!YX)rteD*ZNu%NprK+R!v6d zLw%R3d96hp%SCoi9fO=rj#Q+?UZlnY*0cJ-)AMW;?qG}3hR}KWS)q(q+uQY3-qy+g z-CpDPYft%{E&5ol{rqI*@gN`9;zP^uxuJ;u{((-;BteQ`(fRMkHt%-5RPs`Y`BSyW zqU68t_Iae(<1TAXt^~C%>wEVo1W9v#5cc-*I`{LS!ylBA8`w37=$W?d%D_#arwcq{ z*t1ueOIebtGjjV8U=5(^dayiR^aKuU@6#2NWEVc6RjV^nY6w11uZ=kGP z`28UmNxr_KAYNG)2HY$pMW3^#M^3nQ7)?UEEJ>iZgXldnjxyS>W&uQa@t=Q zNFVG!{$=h3ZQ>ek{!Ornr?l6)2l<47e$$|j?CDY6!EYRrSQk>+ zPKK*IQKa>@`I~DKedz}tvFQaDgr2Hn<`NPijg>-NU|TIZb|)#P^$Xr6Ha@6>PEsE!%E{~t*F`HjJlmwVP{c|yXLA9 ze_|;_24MQt2O9Y%KL=-Qs!bixcypGlPekfmA`sk&-ydkn)24lbJ+n6cka%ShNAQ`% z1c)oS0Jc_)$CWCXVFINC01nQY&w|8B*%IVC zri?TOjj7UvW{r#`1TUeFdPZr>Ub85^gMZ zux#}5eBe#2NR?odJ=bdBrUU?S6{P-P-AMxrWZ zDS`ThvY+!XLmkuT@=h{6UWu%k0V-Q3bxkZvzLa)kXtMc5&lX6@oPsX@l7y&KzJUtV zfxP+oC1H-8(7PYKBoUHQ`%6(nN-igwix&^5{fz?jz&3>AG^iMg{&R|pyF-n|eh+)j z_m=AuJM~Hl(`RbS zxo=(H^Hgt({n9!e0)gr@sO!4wdh*z-1^Z$C6)NK38&~(iHfI|lS zTGTf)MhNQoAIgqL}U1lmrdQ4F3 z*cCNJj3mMV*qF7=YfC?JKbJviQ7j&h$&T#+u3py0GUk-sEQmFajaZ-nbPGsHCyb%1 zt83?d_zH2mzUl8z3{fGI^|w}5sxXiokaWo1qg^pFShFhO>#TW!fy+plYYXi{^}Fh_ zCAK(hUK~}RPXbCW9QGC$&v>k7*VM``_PH8}{dF3ok2mc+kP)d`F?9_b8>*ZoQI*5WRvr#y>=Uw#q`-yB zu^g`um{5ohGH=I?5?MhmIRBC`B>n+(`az$QC7toY+9s19|BNmx zrd+D#%H}SW4L+1|u z-!1b#4ByrV?Xh40|08cN?JM~2$G7tsIJr>ua#pJ8k4zSS5yRk6`H(xtle&6HkR~h3 zZ<20T2rmj<6K)Eo&UgW@GB_~n!pLy)+mj`&GOl8huFc8;)dzy$?e_qEy?(e%4f@pv6)#@1in@DLC zG02Y^wl6ufgK0{V9+6~n9iUc}U*6W>K_DoQQu)zYB1JYF zp%^e10D_DV%Xv)y4?Tp@4?HDeJP{*$2)Sh*-Gn6FeGx9;=!m=ePjk>F16spsXm5DP z7&-Y=nSr|dCg2JIG4qukNbhL;nb9&t2+4v7O)o6edP)Byx4(3XRKtSo`4$0tsOJy$ z60x4z*yNU%$6|3QOJu)?F7*buNRo^@X*nEFjf`BM4KuhdTr3)hhHC&u<<%^_T?%f9 z(Dw0Th_q~)3Nq0Ex2L0q=;zIa4u5FRN}!Y14&|;RAX*NQSq3p=dJZ6w!G`VG-^|97 z;1C!prf){5Y~SnACWCx(4ZyL;CV+Zr0nHy>49NKyZw+$D561$!=Ec~JQ!4Q6K{?4d zXt_(9P&C@Y5vXj@C#tSQNSM^m@pzMo+#=3||78q|xw_~{t>RC8br@9g0dq zhGxB4wMbScpL-|CDb7Y2PTG`%uTVt~;jn#(q)DmK>zU*E;B<~%JPImk6A=}C1hS~l z7v5TTTHoKHQj57MTQU&nGLX2mbZrym39}e03QoGfaHF|BEmjnILF7IcZ(tV%ij!kB zzXI?MqHdpWF`UC$Sy{J%P|eSnW57bGu-f#GSG-vC??FW*IN!fNt+X9sDychq$8A)t z*(0*TJ@^%rH+KPGkjLj}Fn#V_5h|L`V(hs7_kY~DxVS%>fw~+I5(5M%nq)pH_1-@& zCL<^B0Qt|$L=aI11iMN8pPhg0P9ZN{>E0lSHvOEdCp_2u9A))N zon~Phz#G4I<>loK4c-+SZe0@*E4U(BxhUN(bMXUABEUhykigL(6$qkEF@18J>J&&o z*I6S90#ZOOp$M5bq9lhb184zA>&2UF!_g&nqkV&ew{!M-CJb_$&3 zZv=4SQn}KGwRB;}tY7kjQ&PhgOA>wpy03>X&QJ&7a`4kRr`1X}brIGd{r!9yuYqdr zj=s;#>%WOkENS_&Lo!R&j7g@UrKT_%A%ZG9K;vH!!io^v|AD#a{x`*~3HX>m{bBLP zM!+@7PN-@)q5dRyq*J0E6dsB^9>}GMTHYu4+b~APhp|6~3)J9%&JuV9MuodNUZdj; z-ha>ml4mlmzUd;)?+k&G>UJif7f1FA=S3ZgukY6j)#7pHjIC`N)dfcO^}K%it&25`eQ6KV3sAoZy-&jY)i=0(o@OFJ(B z#J3GZylTtM5a=zi^41+@y#$>6m_aph zqxhQ!Y^|+$uo@~p74wqna&bM|mHJ@T%GB>|;$rgWaXBnOLB=k*}G`PX-rdOS|-rX)eE7xDOm&_(2qnu3zGeer%a46rU!|>(%NRS@8GaTr-1wbqQU-bxo5xn zG6B5%39--?!Qiqzr%>avmW1{>pU|(ceZnjoT3Y9IY}Fp6<}}w%T;p|I5p`*(;+aA6 zEi*sor2*CnUijW5HcfAXPyl>|pvheTB-&03=7k=tRFC z9c(ZxYscd5)H_T|nht!j(XlXgULL~rYV|q%iFV&%V=kLl`A!|D$spl@ef)o z_t;xTVjByR6B8k4SQw+UQdZcCG#DF{N&s)nWq(!A^I$#o4rspa2j>-S&&??$gyvHV z+CEO;F)9PBus~~`azVSX$7<>K@zbJ#xX0*tjq5WY#9Q%XFzSwd&}P>=AvSx`~=rHngatPb2sPOgLSi z#sL5cH{SdaY3U6rpZdi2iD7$u1hm3L{+F`Y^{;-7i=^3y6A(uN&*No9-QO0oFvKGK zEoXeA$@2W{v~dD1_1fPZt~*jy3Y@Zm8Y7=}JKQuI1&NlgiG0ldNFz6Q8Ef@7RD0JP z74)q^0zra1hDN(c`+fJ%YL)6>UmP+@%%I{KeO|p}ek33hVqYT?eN!wKL67a|pWoIT z!qWDUW)%p!H6{mSNMc0wshA64+lR3mek>y3)!lVZzwd>|_u8$Yrqz|#O%Nh}I{>=`>D(3R3qc}&8O~g?m(epFlwhMbe5;hLfH0v&Q3;v_8O&hcBKlx zj;+~8j7)JVg-fK2b5ez1U{x_)=J3yTlQ{D6$5TLkvDpO*cMtMGsoM0i_!y}RC@yp8l6rbJ zhBlp;*kJB6G<7TidUbGaN^)I#ww7k{o4~1EINa;A{h5b$otyK+gteA#^I5DNh6*f! z#Ks)=H2R}A^vdbexjHQyd@-qzGMBoGskA z2`tVpQ?vb&oY`oLnZOuN!J6y1(c>KRjg#16gEWmO*k=+jPwKhN+fs(%L-O?66_N~Y z`B_at9&_gKg&9fCi7Q@h*V5>&GVz)`>C2FCqorgXIa21=*=`5)3OY$grFd@K8ti+H0|Y3m1%lFXDiDI$LYJthB@A5ot{*I+bF^h8;2eK@ya z3zbu)gUfA?Z3h(#e`GO?9@@euFsb*xx%e#fW4|jjp5Yk|W?RK^+$vER^zi785xzDa zDPDhU&eb1XT4UMs@LMH|yr2%zAkbh=OQ!3+L1oi0$nDD7>nz`P7gyg1k603%Te;U^ zh!#t`)s)22^HG6uIjrlTd$dN(E`9P`L;1e<$uA=-ly`4))maA7pGHi^02=LaUn6P9 z;SDeaw<;jgif0Uk^iC6SiK-h(UkL>bR*XFCT}RyGOXCW_^^XTs7J8DQyRt|rI(g!lCQ$vdK$;9v^epN}5l5`9K8HsUOA zE&^`?y4(Q|jPLh`MZC2{xuJ;U{Vz}ZE=Fuprj@yyc=B zrnr8CRG2q@n9u2y=SkY4W$OkrDv^fiRTE|kkhX6xKREKnk30Jv1#SXKgnQ_L<6V+w zT+DJL3p}Ta+#cQO3DO_ z?_ghcJgyZtl0jg-qsMef(&#)dGTd8@m0@}yfJJooj9vdZ)z^^Q$)#IIweqLjJoS`l zE6;wWFJ-v=l)C-Q%!chlJh2Q)a@R5fBk!D;-n(+$^b6hrL?a^n`vLu z#V60^R9g)#;yS(fxZ>i5?nyI>!8$t>Bha!Gn8ZXE9}WpvT1>8k87>bhj|B$pH^3+I zK3FeyhY=KiP5a`+GJ+!{hMFo0Skj1#52F{1L~p4C$^FL;_y(p(^jND3lA`}@(h(L* z*?`FYH1;7Ht5)>Fm5&0&Nskw&&F z`run4|Fw)u64`#EQRZ9h-{v5&2G|*+;?AhsVgFhtfi?qvA@4YY{&>h2Cr1sz#C3Wn z68m4vtnvt^A}mTR4U&I6BA3NI6!$Y1{{1=lgBVu7 e`)&UG(zg1Yij&xZ7Vz3i3 Date: Tue, 2 May 2017 15:40:42 -0700 Subject: [PATCH 277/305] fixed xzstd --format=xz was missing a break, making the execution continue into lz4 error message --- programs/Makefile | 6 ++---- programs/fileio.c | 4 ++++ programs/zstdcli.c | 4 ++-- 3 files changed, 8 insertions(+), 6 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index 2ab29e1ef..0c50f50bd 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -140,15 +140,13 @@ $(ZSTDDECOMP_O): CFLAGS += $(ALIGN_LOOP) zstd : CPPFLAGS += $(ZLIBCPP) $(THREAD_CPP) zstd : LDFLAGS += $(ZLIBLD) $(THREAD_LD) -zstd : LZMA_MSG := $(NO_LZMA_MSG) +zstd zstd-nogz zstd4 : LZMA_MSG := "" +zstd zstd-nogz xzstd : LZ4_MSG := "" zstd-nogz : ZLIB_MSG := $(NO_ZLIB_MSG) -zstd-nogz : LZMA_MSG := $(NO_LZMA_MSG) xzstd : CPPFLAGS += $(ZLIBCPP) $(LZMACPP) xzstd : LDFLAGS += $(ZLIBLD) $(LZMALD) -xzstd : LZ4_MSG := $(NO_LZ4_MSG) zstd4 : CPPFLAGS += $(ZLIBCPP) $(LZ4CPP) zstd4 : LDFLAGS += $(ZLIBLD) $(LZ4LD) -zstd4 : LZMA_MSG := $(NO_LZMA_MSG) zstd zstd-nogz xzstd zstd4 : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) zstd zstd-nogz xzstd zstd4 : $(ZSTDLIB_FILES) zstdcli.o fileio.o bench.o datagen.o dibio.o @echo "$(THREAD_MSG)" diff --git a/programs/fileio.c b/programs/fileio.c index 944b11ab4..e188936b2 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -617,6 +617,7 @@ static int FIO_compressFilename_internal(cRess_t ress, switch (g_compressionType) { case FIO_zstdCompression: break; + case FIO_gzipCompression: #ifdef ZSTD_GZCOMPRESS compressedfilesize = FIO_compressGzFrame(&ress, srcFileName, fileSize, compressionLevel, &readsize); @@ -625,6 +626,7 @@ static int FIO_compressFilename_internal(cRess_t ress, EXM_THROW(20, "zstd: %s: file cannot be compressed as gzip (zstd compiled without ZSTD_GZCOMPRESS) -- ignored \n", srcFileName); #endif goto finish; + case FIO_xzCompression: case FIO_lzmaCompression: #ifdef ZSTD_LZMACOMPRESS @@ -633,6 +635,8 @@ static int FIO_compressFilename_internal(cRess_t ress, (void)compressionLevel; EXM_THROW(20, "zstd: %s: file cannot be compressed as xz/lzma (zstd compiled without ZSTD_LZMACOMPRESS) -- ignored \n", srcFileName); #endif + goto finish; + case FIO_lz4Compression: #ifdef ZSTD_LZ4COMPRESS compressedfilesize = FIO_compressLz4Frame(&ress, srcFileName, fileSize, compressionLevel, &readsize); diff --git a/programs/zstdcli.c b/programs/zstdcli.c index f6ce2e1a6..32fef9993 100644 --- a/programs/zstdcli.c +++ b/programs/zstdcli.c @@ -514,11 +514,11 @@ int main(int argCount, const char* argv[]) /* Quiet mode */ case 'q': g_displayLevel--; argument++; break; - /* keep source file (default); for gzip/xz compatibility */ + /* keep source file (default) */ case 'k': FIO_setRemoveSrcFile(0); argument++; break; /* Checksum */ - case 'C': argument++; FIO_setChecksumFlag(2); break; + case 'C': FIO_setChecksumFlag(2); argument++; break; /* test compressed file */ case 't': operation=zom_test; argument++; break; From 3791d2105d14fa7bfc5da120325ab5ca045c4130 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 2 May 2017 16:38:37 -0700 Subject: [PATCH 278/305] added xzstd4 target support for all formats, xz/lzma/lz4 included --- programs/Makefile | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index 0c50f50bd..409b976ca 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -140,15 +140,17 @@ $(ZSTDDECOMP_O): CFLAGS += $(ALIGN_LOOP) zstd : CPPFLAGS += $(ZLIBCPP) $(THREAD_CPP) zstd : LDFLAGS += $(ZLIBLD) $(THREAD_LD) -zstd zstd-nogz zstd4 : LZMA_MSG := "" -zstd zstd-nogz xzstd : LZ4_MSG := "" -zstd-nogz : ZLIB_MSG := $(NO_ZLIB_MSG) +zstd zstd-nogz zstd4 : LZMA_MSG := - xz/lzma support is disabled +zstd zstd-nogz xzstd : LZ4_MSG := - lz4 support is disabled +zstd-nogz : ZLIB_MSG := - gzip support is disabled xzstd : CPPFLAGS += $(ZLIBCPP) $(LZMACPP) xzstd : LDFLAGS += $(ZLIBLD) $(LZMALD) zstd4 : CPPFLAGS += $(ZLIBCPP) $(LZ4CPP) zstd4 : LDFLAGS += $(ZLIBLD) $(LZ4LD) -zstd zstd-nogz xzstd zstd4 : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) -zstd zstd-nogz xzstd zstd4 : $(ZSTDLIB_FILES) zstdcli.o fileio.o bench.o datagen.o dibio.o +xzstd4 : CPPFLAGS += $(ZLIBCPP) $(LZMACPP) $(LZ4CPP) +xzstd4 : LDFLAGS += $(ZLIBLD) $(LZMALD) $(LZ4LD) +zstd zstd-nogz xzstd zstd4 xzstd4 : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) +zstd zstd-nogz xzstd zstd4 xzstd4 : $(ZSTDLIB_FILES) zstdcli.o fileio.o bench.o datagen.o dibio.o @echo "$(THREAD_MSG)" @echo "$(ZLIB_MSG)" @echo "$(LZMA_MSG)" From f47284fec0563a5e5c1eab148475d44f686e7f3c Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 2 May 2017 16:55:57 -0700 Subject: [PATCH 279/305] reorganized Makefile for multiple targets --- programs/Makefile | 28 ++++++++++++++-------------- 1 file changed, 14 insertions(+), 14 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index 409b976ca..204352b4c 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -138,17 +138,17 @@ all: zstd $(ZSTDDECOMP_O): CFLAGS += $(ALIGN_LOOP) -zstd : CPPFLAGS += $(ZLIBCPP) $(THREAD_CPP) -zstd : LDFLAGS += $(ZLIBLD) $(THREAD_LD) +zstd-nogz zstd xzstd zstd4 xzstd4 : CPPFLAGS += $(THREAD_CPP) +zstd-nogz zstd xzstd zstd4 xzstd4 : LDFLAGS += $(THREAD_LD) +zstd xzstd zstd4 xzstd4 : CPPFLAGS += $(ZLIBCPP) +zstd xzstd zstd4 xzstd4 : LDFLAGS += $(ZLIBLD) +xzstd xzstd4 : CPPFLAGS += $(LZMACPP) +xzstd xzstd4 : LDFLAGS += $(LZMALD) +zstd4 xzstd4 : CPPFLAGS += $(LZ4CPP) +zstd4 xzstd4 : LDFLAGS += $(LZ4LD) +zstd-nogz : ZLIB_MSG := - gzip support is disabled zstd zstd-nogz zstd4 : LZMA_MSG := - xz/lzma support is disabled zstd zstd-nogz xzstd : LZ4_MSG := - lz4 support is disabled -zstd-nogz : ZLIB_MSG := - gzip support is disabled -xzstd : CPPFLAGS += $(ZLIBCPP) $(LZMACPP) -xzstd : LDFLAGS += $(ZLIBLD) $(LZMALD) -zstd4 : CPPFLAGS += $(ZLIBCPP) $(LZ4CPP) -zstd4 : LDFLAGS += $(ZLIBLD) $(LZ4LD) -xzstd4 : CPPFLAGS += $(ZLIBCPP) $(LZMACPP) $(LZ4CPP) -xzstd4 : LDFLAGS += $(ZLIBLD) $(LZMALD) $(LZ4LD) zstd zstd-nogz xzstd zstd4 xzstd4 : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) zstd zstd-nogz xzstd zstd4 xzstd4 : $(ZSTDLIB_FILES) zstdcli.o fileio.o bench.o datagen.o dibio.o @echo "$(THREAD_MSG)" @@ -176,7 +176,7 @@ zstd-nolegacy : clean_decomp_o zstd-nomt : THREAD_CPP := zstd-nomt : THREAD_LD := -zstd-nomt : THREAD_MSG := $(NO_THREAD_MSG) +zstd-nomt : THREAD_MSG := - multi-threading disabled zstd-nomt : zstd zstd-pgo : MOREFLAGS = -fprofile-generate @@ -191,18 +191,18 @@ zstd-pgo : clean zstd $(RM) $(ZSTDDECOMP_O) $(MAKE) zstd MOREFLAGS=-fprofile-use -zstd-frugal: $(ZSTD_FILES) zstdcli.c fileio.c +# minimal target, with only zstd compression and decompression. no bench. no legacy. +zstd-small: CFLAGS = "-Os -s" +zstd-frugal zstd-small: $(ZSTD_FILES) zstdcli.c fileio.c $(CC) $(FLAGS) -DZSTD_NOBENCH -DZSTD_NODICT $^ -o zstd$(EXT) -zstd-small: - CFLAGS="-Os -s" $(MAKE) zstd-frugal - zstd-decompress: $(ZSTDCOMMON_FILES) $(ZSTDDECOMP_FILES) zstdcli.c fileio.c $(CC) $(FLAGS) -DZSTD_NOBENCH -DZSTD_NODICT -DZSTD_NOCOMPRESS $^ -o $@$(EXT) zstd-compress: $(ZSTDCOMMON_FILES) $(ZSTDCOMP_FILES) zstdcli.c fileio.c $(CC) $(FLAGS) -DZSTD_NOBENCH -DZSTD_NODICT -DZSTD_NODECOMPRESS $^ -o $@$(EXT) +# zstd is now built with Multi-threading by default zstdmt: zstd generate_res: From 710497d8ea99f9d6b399e805bd7d86a55083e57c Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Tue, 2 May 2017 17:18:24 -0700 Subject: [PATCH 280/305] updated programs/README.md, to introduce compilation variables make it possible to enable/disable features individually --- programs/Makefile | 22 ++++++++++---------- programs/README.md | 50 +++++++++++++++++++++++++++++++++++----------- 2 files changed, 50 insertions(+), 22 deletions(-) diff --git a/programs/Makefile b/programs/Makefile index 204352b4c..491b3ed3a 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -138,19 +138,16 @@ all: zstd $(ZSTDDECOMP_O): CFLAGS += $(ALIGN_LOOP) -zstd-nogz zstd xzstd zstd4 xzstd4 : CPPFLAGS += $(THREAD_CPP) -zstd-nogz zstd xzstd zstd4 xzstd4 : LDFLAGS += $(THREAD_LD) -zstd xzstd zstd4 xzstd4 : CPPFLAGS += $(ZLIBCPP) -zstd xzstd zstd4 xzstd4 : LDFLAGS += $(ZLIBLD) +zstd xzstd zstd4 xzstd4 : CPPFLAGS += $(THREAD_CPP) $(ZLIBCPP) +zstd xzstd zstd4 xzstd4 : LDFLAGS += $(THREAD_LD) $(ZLIBLD) xzstd xzstd4 : CPPFLAGS += $(LZMACPP) xzstd xzstd4 : LDFLAGS += $(LZMALD) zstd4 xzstd4 : CPPFLAGS += $(LZ4CPP) zstd4 xzstd4 : LDFLAGS += $(LZ4LD) -zstd-nogz : ZLIB_MSG := - gzip support is disabled -zstd zstd-nogz zstd4 : LZMA_MSG := - xz/lzma support is disabled -zstd zstd-nogz xzstd : LZ4_MSG := - lz4 support is disabled -zstd zstd-nogz xzstd zstd4 xzstd4 : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) -zstd zstd-nogz xzstd zstd4 xzstd4 : $(ZSTDLIB_FILES) zstdcli.o fileio.o bench.o datagen.o dibio.o +zstd zstd4 : LZMA_MSG := - xz/lzma support is disabled +zstd xzstd : LZ4_MSG := - lz4 support is disabled +zstd xzstd zstd4 xzstd4 : CPPFLAGS += -DZSTD_LEGACY_SUPPORT=$(ZSTD_LEGACY_SUPPORT) +zstd xzstd zstd4 xzstd4 : $(ZSTDLIB_FILES) zstdcli.o fileio.o bench.o datagen.o dibio.o @echo "$(THREAD_MSG)" @echo "$(ZLIB_MSG)" @echo "$(LZMA_MSG)" @@ -170,7 +167,6 @@ ifneq (,$(filter Windows%,$(OS))) endif $(CC) -m32 $(FLAGS) $^ $(RES32_FILE) -o $@$(EXT) - zstd-nolegacy : clean_decomp_o $(MAKE) zstd ZSTD_LEGACY_SUPPORT=0 @@ -179,6 +175,12 @@ zstd-nomt : THREAD_LD := zstd-nomt : THREAD_MSG := - multi-threading disabled zstd-nomt : zstd +zstd-nogz : ZLIBCPP := +zstd-nogz : ZLIBLD := +zstd-nogz : ZLIB_MSG := - gzip support is disabled +zstd-nogz : zstd + + zstd-pgo : MOREFLAGS = -fprofile-generate zstd-pgo : clean zstd ./zstd -b19i1 $(PROFILE_WITH) diff --git a/programs/README.md b/programs/README.md index 203fd7b49..d7922a096 100644 --- a/programs/README.md +++ b/programs/README.md @@ -11,8 +11,29 @@ There are however other Makefile targets that create different variations of CLI - `zstd-decompress` : decompressor-only version of CLI; without dictionary builder, benchmark, and support for decompression of legacy zstd versions +#### Compilation variables +`zstd` tries to detect and use the following features automatically : + +- __HAVE_THREAD__ : multithreading is automatically enabled when `pthread` is detected. + It's possible to disable multithread support, by either compiling `zstd-nomt` target or using HAVE_THREAD=0 variable. + Example : make zstd HAVE_THREAD=0 + It's also possible to force compilation with multithread support, using HAVE_THREAD=1. + In which case, linking stage will fail if `pthread` library cannot be found. + This might be useful to prevent silent feature disabling. + +- __HAVE_ZLIB__ : `zstd` can compress and decompress files in `.gz` format. + This is done through command `--format=gzip`. + Alternatively, symlinks named `gzip` or `gunzip` will mimic intended behavior. + .gz support is automatically enabled when `zlib` library is detected at build time. + It's possible to disable .gz support, by either compiling `zstd-nogz` target or using HAVE_ZLIB=0 variable. + Example : make zstd HAVE_ZLIB=0 + It's also possible to force compilation with zlib support, using HAVE_ZLIB=1. + In which case, linking stage will fail if `zlib` library cannot be found. + This might be useful to prevent silent feature disabling. + + #### Aggregation of parameters -CLI supports aggregation of parameters i.e. `-b1`, `-e18`, and `-i1` can be joined into `-b1e18i1`. +CLI supports aggregation of parameters i.e. `-b1`, `-e18`, and `-i1` can be joined into `-b1e18i1`. #### Dictionary builder in Command Line Interface @@ -23,7 +44,7 @@ which can be loaded before compression and decompression. Using a dictionary, the compression ratio achievable on small data improves dramatically. These compression gains are achieved while simultaneously providing faster compression and decompression speeds. -Dictionary work if there is some correlation in a family of small data (there is no universal dictionary). +Dictionary work if there is some correlation in a family of small data (there is no universal dictionary). Hence, deploying one dictionary per type of data will provide the greater benefits. Dictionary gains are mostly effective in the first few KB. Then, the compression algorithm will rely more and more on previously decoded content to compress the rest of the file. @@ -35,7 +56,6 @@ Usage of the dictionary builder and created dictionaries with CLI: 3. Decompress with the dictionary: `zstd --decompress FILE.zst -D dictionaryName` - #### Benchmark in Command Line Interface CLI includes in-memory compression benchmark module for zstd. The benchmark is conducted using given filenames. The files are read into memory and joined together. @@ -48,7 +68,6 @@ One can select compression levels starting from `-b` and ending with `-e`. The `-i` parameter selects minimal time used for each of tested levels. - #### Usage of Command Line Interface The full list of options can be obtained with `-h` or `-H` parameter: ``` @@ -62,33 +81,40 @@ Arguments : -d : decompression -D file: use `file` as Dictionary -o file: result stored into `file` (only if 1 input file) - -f : overwrite output without prompting + -f : overwrite output without prompting and (de)compress links --rm : remove source file(s) after successful de/compression -k : preserve source file(s) (default) -h/-H : display help/long help and exit Advanced arguments : -V : display Version number and exit - -v : verbose mode; specify multiple times to increase log level (default:2) + -v : verbose mode; specify multiple times to increase verbosity -q : suppress warnings; specify twice to suppress errors too -c : force write to standard output, even if it is the console - -r : operate recursively on directories --ultra : enable levels beyond 19, up to 22 (requires more memory) + -T# : use # threads for compression (default:1) + -B# : select size of each job (default:0==automatic) --no-dictID : don't write dictID into header (dictionary compression) --[no-]check : integrity check (default:enabled) + -r : operate recursively on directories +--format=gzip : compress files to the .gz format --test : test compressed file integrity ---[no-]sparse : sparse mode (default:enabled on file, disabled on stdout) +--[no-]sparse : sparse mode (default:disabled) + -M# : Set a memory usage limit for decompression +-- : All arguments after "--" are treated as files Dictionary builder : --train ## : create a dictionary from a training set of files +--train-cover[=k=#,d=#,steps=#] : use the cover algorithm with optional args +--train-legacy[=s=#] : use the legacy algorithm with selectivity (default: 9) -o file : `file` is dictionary name (default: dictionary) ---maxdict ## : limit dictionary to specified size (default : 112640) - -s# : dictionary selectivity level (default: 9) ---dictID ## : force dictionary ID to specified value (default: random) +--maxdict=# : limit dictionary to specified size (default : 112640) +--dictID=# : force dictionary ID to specified value (default: random) Benchmark arguments : -b# : benchmark file(s), using # compression level (default : 1) -e# : test all compression levels from -bX to # (default: 1) -i# : minimum evaluation time in seconds (default : 3s) -B# : cut file into independent blocks of size # (default: no block) - ``` \ No newline at end of file +--priority=rt : set process priority to real-time +``` From cf4f9403fa9aac8f20216095f68b67b12bbdeeca Mon Sep 17 00:00:00 2001 From: Nick Terrell Date: Wed, 3 May 2017 11:07:39 -0700 Subject: [PATCH 281/305] [kernel] Update README with SquashFS patch Take patch from PR #682 by @iburinoc and update benchmarks. --- contrib/linux-kernel/README.md | 43 ++++++++++++++++++++-- contrib/linux-kernel/squashfs-benchmark.sh | 39 ++++++++++++++++++++ 2 files changed, 79 insertions(+), 3 deletions(-) create mode 100755 contrib/linux-kernel/squashfs-benchmark.sh diff --git a/contrib/linux-kernel/README.md b/contrib/linux-kernel/README.md index a62838517..16bc2d48f 100644 --- a/contrib/linux-kernel/README.md +++ b/contrib/linux-kernel/README.md @@ -1,6 +1,6 @@ # Linux Kernel Patch -There are two pieces, the `zstd_compress` and `zstd_decompress` kernel modules, and the BtrFS patch. +There are three pieces, the `zstd_compress` and `zstd_decompress` kernel modules, the BtrFS patch, and the SquashFS patch. The patches are based off of the linux kernel master branch (version 4.10). ## Zstd Kernel modules @@ -22,14 +22,15 @@ The patches are based off of the linux kernel master branch (version 4.10). * The patch is located in `btrfs.diff`. * Additionally `fs/btrfs/zstd.c` is provided as a source for convenience. -* The patch seems to be working, it doesn't crash the kernel, and compresses at speeds and ratios athat are expected. - It can still use some more testing for fringe features, like printing options. +* The patch seems to be working, it doesn't crash the kernel, and compresses at speeds and ratios that are expected. + It could still use some more testing for fringe features, like printing options. ### Benchmarks Benchmarks run on a Ubuntu 14.04 with 2 cores and 4 GiB of RAM. The VM is running on a Macbook Pro with a 3.1 GHz Intel Core i7 processor, 16 GB of ram, and a SSD. +The kernel running was built from the master branch with the patch (version 4.10). The compression benchmark is copying 10 copies of the unzipped [silesia corpus](http://mattmahoney.net/dc/silesia.html) into a BtrFS @@ -50,3 +51,39 @@ See `btrfs-benchmark.sh` for details. | zstd 9 | 2.92 | 43 MB/s | 406 MB/s | | zstd 12 | 2.93 | 21 MB/s | 408 MB/s | | zstd 15 | 3.01 | 11 MB/s | 354 MB/s | + + +## SquashFS + +* The patch is located in `squashfs.diff` +* Additionally `fs/squashfs/zstd_wrapper.c` is provided as a source for convenience. +* The patch has been tested on a 4.10 kernel. + +### Benchmarks + +Benchmarks run on a Ubuntu 14.04 with 2 cores and 4 GiB of RAM. +The VM is running on a Macbook Pro with a 3.1 GHz Intel Core i7 processor, +16 GB of ram, and a SSD. +The kernel running was built from the master branch with the patch (version 4.10). + +The compression benchmark is the file tree from the SquashFS archive found in the +Ubuntu 16.10 desktop image (ubuntu-16.10-desktop-amd64.iso). +The compression benchmark uses mksquashfs with the default block size (128 KB) +and various compression algorithms/compression levels. +`xz` and `zstd` are also benchmarked with 256 KB blocks. +The decompression benchmark is timing how long it takes to `tar` the file tree +into `/dev/null`. +See `squashfs-benchmark.sh` for details. + +| Algorithm | Compression ratio | Compression speed | Decompression speed | +|----------------|-------------------|-------------------|---------------------| +| gzip | 2.92 | 15 MB/s | 128 MB/s | +| lzo | 2.64 | 9.5 MB/s | 217 MB/s | +| lz4 | 2.12 | 94 MB/s | 218 MB/s | +| xz | 3.43 | 5.5 MB/s | 35 MB/s | +| xz 256 KB | 3.53 | 5.4 MB/s | 40 MB/s | +| zstd 1 | 2.71 | 96 MB/s | 210 MB/s | +| zstd 5 | 2.93 | 69 MB/s | 198 MB/s | +| zstd 10 | 3.01 | 41 MB/s | 225 MB/s | +| zstd 15 | 3.13 | 11.4 MB/s | 224 MB/s | +| zstd 16 256 KB | 3.24 | 8.1 MB/s | 210 MB/s | diff --git a/contrib/linux-kernel/squashfs-benchmark.sh b/contrib/linux-kernel/squashfs-benchmark.sh new file mode 100755 index 000000000..02dfd7325 --- /dev/null +++ b/contrib/linux-kernel/squashfs-benchmark.sh @@ -0,0 +1,39 @@ +# !/bin/sh +set -e + +# Benchmarks run on a Ubuntu 14.04 VM with 2 cores and 4 GiB of RAM. +# The VM is running on a Macbook Pro with a 3.1 GHz Intel Core i7 processor and +# 16 GB of RAM and an SSD. + +# $BENCHMARK_DIR is generated with the following commands, from the Ubuntu image +# ubuntu-16.10-desktop-amd64.iso. +# > mkdir mnt +# > sudo mount -o loop ubuntu-16.10-desktop-amd64.iso mnt +# > cp mnt/casper/filesystem.squashfs . +# > sudo unsquashfs filesystem.squashfs + +# $HOME is on a ext4 filesystem +BENCHMARK_DIR="$HOME/squashfs-root/" +BENCHMARK_FS="$HOME/filesystem.squashfs" + +# Normalize the environment +sudo rm -f $BENCHMARK_FS 2> /dev/null > /dev/null || true +sudo umount /mnt/squashfs 2> /dev/null > /dev/null || true + +# Run the benchmark +echo "Compression" +echo "sudo mksquashfs $BENCHMARK_DIR $BENCHMARK_FS $@" +time sudo mksquashfs $BENCHMARK_DIR $BENCHMARK_FS $@ 2> /dev/null > /dev/null + +echo "Approximate compression ratio" +printf "%d / %d\n" \ + $(sudo du -sx --block-size=1 $BENCHMARK_DIR | cut -f1) \ + $(sudo du -sx --block-size=1 $BENCHMARK_FS | cut -f1); + +# Mount the filesystem +sudo mount -t squashfs $BENCHMARK_FS /mnt/squashfs + +echo "Decompression" +time sudo tar -c /mnt/squashfs 2> /dev/null | wc -c > /dev/null + +sudo umount /mnt/squashfs From c08e56861e2a3c4e5df5e6a63b580e7661f519c1 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Wed, 3 May 2017 14:32:33 -0700 Subject: [PATCH 282/305] updated dict graphs to 2D mode --- NEWS | 2 +- doc/images/dict-cr.png | Bin 23323 -> 90047 bytes doc/images/dict-cs.png | Bin 25052 -> 93837 bytes doc/images/dict-ds.png | Bin 27053 -> 89590 bytes programs/Makefile | 8 +++++--- 5 files changed, 6 insertions(+), 4 deletions(-) diff --git a/NEWS b/NEWS index 407f3b8ac..7d9c9c94e 100644 --- a/NEWS +++ b/NEWS @@ -5,7 +5,7 @@ cli : new : zstdmt symlink hardwired to `zstd -T0` cli : new : command --threads=# (#671) cli : changed : cover dictionary builder by default, for improved quality, by Nick Terrell cli : new : commands --train-cover and --train-legacy, to select dictionary algorithm and parameters -cli : new : experimental target `zstd4`, with support for lz4 format, by Sean Purcell +cli : experimental targets `zstd4` and `xzstd4`, with support for lz4 format, by Sean Purcell cli : fix : does not output compressed data on console cli : fix : ignore symbolic links unless --force specified, API : breaking change : ZSTD_createCDict_advanced(), only use compressionParameters as argument diff --git a/doc/images/dict-cr.png b/doc/images/dict-cr.png index f555a46c7b99b512966ca81ad8283d7f164ae160..f3a9ce2bdda64c7f370642d596a5cc1fdf50bf54 100644 GIT binary patch literal 90047 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

5pNgHLG2GX04f@dE_fzhrfD)Cb*raeoCa+P z_P-*sm|-)wlK|Z0@i;D!S6_r(&|$gq(TbXA zdL;YADw(FP0z5QiGGeO{n-2V)Y_(fZ#rlVG7xQqGtn`*|>(%Qy+)8*JWYD7f2ivRH z)ENVRaHZLRN^&?Gt)TuG9yDi@v9y|QVi7!|zyVkFe50*rr9wUAIMyjKaii3Z?^0gZ zcv07Zaxb2z!_)0%QI)TxdqfCG7$Aa+w0x!D&#?{Kz)3jtTqw{Go$ET=yoxW_lI6Gq zm`X+NaCS7fX9oP*+Ka7m%A8pxz_ebnyrnCX=VSl_P^O}T@)OyQ(|Nmb;^I2qNb0^_ z8TFHd8xmCHyZrJ|bfU6ssXDD2%cib~WxUyk%$p}rzXOf31|{iowSGXUQYOKAC6fll*MIpFKy_86a4S5|4(_Dv&*I3DH+#5ZS=`AJh$z=%3A@f h(PMGq5^}3($B9ps?bW3FzXHdxKc{}G_*3TP{{UpXM-Tu2 literal 23323 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 diff --git a/doc/images/dict-cs.png b/doc/images/dict-cs.png index ccc02b0d17134efca5b4df2e932e901b474eaea2..55e5ef518fed8b9c598a79ec614c8064be8edd7c 100644 GIT binary patch literal 93837 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 literal 25052 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 no liblzma, building zstd without .xz/.lzma support HAVE_LZMA := $(shell printf '\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_lzma$(EXT) -x c - -llzma 2> $(VOID) && rm have_lzma$(EXT) && echo 1 || echo 0) @@ -116,6 +117,7 @@ LZMALD = -llzma else LZMA_MSG := $(NO_LZMA_MSG) endif + # lz4 detection NO_LZ4_MSG := ==> no liblz4, building zstd without .lz4 support HAVE_LZ4 := $(shell printf '\#include \n\#include \nint main(void) { return 0; }' | $(CC) $(FLAGS) -o have_lz4$(EXT) -x c - -llz4 2> $(VOID) && rm have_lz4$(EXT) && echo 1 || echo 0) @@ -127,9 +129,6 @@ else LZ4_MSG := $(NO_LZ4_MSG) endif -MD2ROFF =ronn -MD2ROFF_FLAGS = --roff --warnings --manual="User Commands" --organization="zstd $(ZSTD_VERSION)" - .PHONY: default all clean clean_decomp_o install uninstall generate_res default: zstd-release @@ -221,6 +220,9 @@ clean: clean_decomp_o: @$(RM) $(ZSTDDECOMP_O) +MD2ROFF = ronn +MD2ROFF_FLAGS = --roff --warnings --manual="User Commands" --organization="zstd $(ZSTD_VERSION)" + zstd.1: zstd.1.md cat $^ | $(MD2ROFF) $(MD2ROFF_FLAGS) | sed -n '/^\.\\\".*/!p' > $@ From a00e9599f1db6123d773b222834826f98c1f4324 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 4 May 2017 17:24:29 -0700 Subject: [PATCH 283/305] removed -g from DEBUGFLAGS It inflates binary sizes, which is negative for the Windows build. It also makes it impossible to check if 2 different source codes get nonetheless compiled to the same binary, since checksum will be different, due to integrated source code. --- lib/Makefile | 2 +- programs/Makefile | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/lib/Makefile b/lib/Makefile index d8d8e179d..f5f610372 100644 --- a/lib/Makefile +++ b/lib/Makefile @@ -22,7 +22,7 @@ VERSION?= $(LIBVER) CPPFLAGS+= -I. -I./common -DXXH_NAMESPACE=ZSTD_ CFLAGS ?= -O3 -DEBUGFLAGS = -g -Wall -Wextra -Wcast-qual -Wcast-align -Wshadow \ +DEBUGFLAGS = -Wall -Wextra -Wcast-qual -Wcast-align -Wshadow \ -Wstrict-aliasing=1 -Wswitch-enum -Wdeclaration-after-statement \ -Wstrict-prototypes -Wundef -Wpointer-arith -Wformat-security CFLAGS += $(DEBUGFLAGS) $(MOREFLAGS) diff --git a/programs/Makefile b/programs/Makefile index 0c920a87b..bb40253b9 100644 --- a/programs/Makefile +++ b/programs/Makefile @@ -41,7 +41,7 @@ CPPFLAGS+= -I$(ZSTDDIR) -I$(ZSTDDIR)/common -I$(ZSTDDIR)/compress \ -I$(ZSTDDIR)/dictBuilder \ -DXXH_NAMESPACE=ZSTD_ # because xxhash.o already compiled with this macro from library CFLAGS ?= -O3 -DEBUGFLAGS = -g -Wall -Wextra -Wcast-qual -Wcast-align -Wshadow \ +DEBUGFLAGS = -Wall -Wextra -Wcast-qual -Wcast-align -Wshadow \ -Wstrict-aliasing=1 -Wswitch-enum -Wdeclaration-after-statement \ -Wstrict-prototypes -Wundef -Wpointer-arith -Wformat-security CFLAGS += $(DEBUGFLAGS) $(MOREFLAGS) From c21a9c35960d7dd18c91b04b9af760f6c071773c Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 4 May 2017 17:26:58 -0700 Subject: [PATCH 284/305] removed zstdmt generation from Appveyor artefact build --- appveyor.yml | 2 -- 1 file changed, 2 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index 67c4f72d7..29eb1ac53 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -91,8 +91,6 @@ cp programs\zstd.exe zstd_%PLATFORM%.exe && appveyor PushArtifact zstd_%PLATFORM%.exe && cp programs\zstd.exe bin\zstd.exe && - make -C programs DEBUGFLAGS= clean zstdmt && - cp programs\zstd.exe bin\zstdmt.exe && cd bin\ && 7z a -tzip zstd-win-release-%PLATFORM%.zip * && appveyor PushArtifact zstd-win-release-%PLATFORM%.zip ) From 9203dab529cdd8c46bdb9b3dd5d6a6cfd3c27fab Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 4 May 2017 17:30:37 -0700 Subject: [PATCH 285/305] Appveyor build artefact creates zipped cli binary --- appveyor.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index 29eb1ac53..df7567b8e 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -88,8 +88,8 @@ ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] lib\dll\example\build_package.bat && make -C programs DEBUGFLAGS= clean zstd && - cp programs\zstd.exe zstd_%PLATFORM%.exe && - appveyor PushArtifact zstd_%PLATFORM%.exe && + 7z a -tzip zstd-win-binary-%PLATFORM%.zip programs\zstd.exe && + appveyor PushArtifact zstd-win-binary-%PLATFORM%.zip && cp programs\zstd.exe bin\zstd.exe && cd bin\ && 7z a -tzip zstd-win-release-%PLATFORM%.zip * && appveyor PushArtifact zstd-win-release-%PLATFORM%.zip From 36153af5ec8f281ce507ee4ece4e78a4059d5452 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Thu, 4 May 2017 17:40:40 -0700 Subject: [PATCH 286/305] creates a binary archive without the `programs` directory also improves compression ratio to -mx9 --- appveyor.yml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index df7567b8e..1f8b8cf8a 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -88,10 +88,10 @@ ( if [%COMPILER%]==[gcc] if [%ARTIFACT%]==[true] lib\dll\example\build_package.bat && make -C programs DEBUGFLAGS= clean zstd && - 7z a -tzip zstd-win-binary-%PLATFORM%.zip programs\zstd.exe && + cd programs\ && 7z a -tzip -mx9 zstd-win-binary-%PLATFORM%.zip zstd.exe && appveyor PushArtifact zstd-win-binary-%PLATFORM%.zip && - cp programs\zstd.exe bin\zstd.exe && - cd bin\ && 7z a -tzip zstd-win-release-%PLATFORM%.zip * && + cp zstd.exe ..\bin\zstd.exe && + cd ..\bin\ && 7z a -tzip -mx9 zstd-win-release-%PLATFORM%.zip * && appveyor PushArtifact zstd-win-release-%PLATFORM%.zip ) ) From 25b283339691af8cd9747f1b843618c4105e2450 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 5 May 2017 15:41:23 -0700 Subject: [PATCH 287/305] keep dev branch status only master branch status is removed, due to misattribution during cron jobs --- README.md | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index 7caee5fd3..26305f51f 100644 --- a/README.md +++ b/README.md @@ -6,10 +6,9 @@ and a command line utility producing and decoding `.zst` and `.gz` files. For other programming languages, you can consult a list of known ports on [Zstandard homepage](http://www.zstd.net/#other-languages). -|Branch |Status | -|------------|---------| -|master | [![Build Status](https://travis-ci.org/facebook/zstd.svg?branch=master)](https://travis-ci.org/facebook/zstd) | -|dev | [![Build Status](https://travis-ci.org/facebook/zstd.svg?branch=dev)](https://travis-ci.org/facebook/zstd) | +| dev branch status | +|----------------------| +| [![Build Status](https://travis-ci.org/facebook/zstd.svg?branch=dev)](https://travis-ci.org/facebook/zstd) As a reference, several fast compression algorithms were tested and compared on a server running Linux Debian (`Linux version 4.8.0-1-amd64`), From 3d55e1fbee1c5b2c188fd9980babc4dac2379d9e Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 5 May 2017 15:48:42 -0700 Subject: [PATCH 288/305] added dev branch Appveyor badge --- README.md | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 26305f51f..97b1a7d1d 100644 --- a/README.md +++ b/README.md @@ -7,8 +7,14 @@ For other programming languages, you can consult a list of known ports on [Zstandard homepage](http://www.zstd.net/#other-languages). | dev branch status | -|----------------------| -| [![Build Status](https://travis-ci.org/facebook/zstd.svg?branch=dev)](https://travis-ci.org/facebook/zstd) +|-------------------| +| [![Build Status][travisDevBadge]][travisLink] [![Build status][AppveyorDevBadge]][AppveyorLink] | + +[travisDevBadge]: https://travis-ci.org/facebook/zstd.svg?branch=dev "Continuous Integration test suite" +[travisLink]: https://travis-ci.org/facebook/zstd +[AppveyorDevBadge]: https://ci.appveyor.com/api/projects/status/xt38wbdxjk5mrbem/branch/dev?svg=true "Windows test suite" +[AppveyorLink]: https://ci.appveyor.com/project/YannCollet/zstd-p0yf0 + As a reference, several fast compression algorithms were tested and compared on a server running Linux Debian (`Linux version 4.8.0-1-amd64`), From 11bff3fbd3a7d9ce64ccfec13ba3193d9fdda37f Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 5 May 2017 15:55:03 -0700 Subject: [PATCH 289/305] added dev branch CircleCI badge --- README.md | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 97b1a7d1d..eee92f917 100644 --- a/README.md +++ b/README.md @@ -8,12 +8,14 @@ you can consult a list of known ports on [Zstandard homepage](http://www.zstd.ne | dev branch status | |-------------------| -| [![Build Status][travisDevBadge]][travisLink] [![Build status][AppveyorDevBadge]][AppveyorLink] | +| [![Build Status][travisDevBadge]][travisLink] [![Build status][AppveyorDevBadge]][AppveyorLink] [![Build status][CircleDevBadge]][CircleLink] [travisDevBadge]: https://travis-ci.org/facebook/zstd.svg?branch=dev "Continuous Integration test suite" [travisLink]: https://travis-ci.org/facebook/zstd [AppveyorDevBadge]: https://ci.appveyor.com/api/projects/status/xt38wbdxjk5mrbem/branch/dev?svg=true "Windows test suite" [AppveyorLink]: https://ci.appveyor.com/project/YannCollet/zstd-p0yf0 +[CircleDevBadge]: https://circleci.com/gh/facebook/zstd/tree/dev.svg?style=shield "Short test suite" +[CircleLink]: https://circleci.com/gh/facebook/zstd As a reference, several fast compression algorithms were tested and compared From 01a1abfdb5a78cd4f63b0199be036b1546c4135c Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Fri, 5 May 2017 19:15:24 -0700 Subject: [PATCH 290/305] cli : -d and -t do not stop after a failed decompression The problematic srcfile will be named on console/log, but decompression/test will continue onto next file in the list. --- programs/fileio.c | 67 ++++++++++++++++++++++++++++++++++------------ tests/playTests.sh | 31 +++++++++++---------- 2 files changed, 67 insertions(+), 31 deletions(-) diff --git a/programs/fileio.c b/programs/fileio.c index e188936b2..ba15555dd 100644 --- a/programs/fileio.c +++ b/programs/fileio.c @@ -206,8 +206,8 @@ void FIO_setOverlapLog(unsigned overlapLog){ static int FIO_remove(const char* path) { #if defined(_WIN32) || defined(WIN32) - /* windows doesn't allow remove read-only files, so try to make it - * writable first */ + /* windows doesn't allow remove read-only files, + * so try to make it writable first */ chmod(path, _S_IWRITE); #endif return remove(path); @@ -983,16 +983,19 @@ static unsigned FIO_passThrough(FILE* foutput, FILE* finput, void* buffer, size_ /** FIO_decompressFrame() : - @return : size of decoded frame + @return : size of decoded frame, or an error code */ +#define FIO_ERROR_ZSTD_DECODING ((unsigned long long)(-2)) unsigned long long FIO_decompressFrame(dRess_t* ress, FILE* finput, + const char* srcFileName, U64 alreadyDecoded) { U64 frameSize = 0; U32 storedSkips = 0; ZSTD_resetDStream(ress->dctx); + if (strlen(srcFileName)>20) srcFileName += strlen(srcFileName)-20; /* display last 20 characters */ /* Header loading (optional, saves one loop) */ { size_t const toRead = 9; @@ -1005,12 +1008,17 @@ unsigned long long FIO_decompressFrame(dRess_t* ress, ZSTD_inBuffer inBuff = { ress->srcBuffer, ress->srcBufferLoaded, 0 }; ZSTD_outBuffer outBuff= { ress->dstBuffer, ress->dstBufferSize, 0 }; size_t const readSizeHint = ZSTD_decompressStream(ress->dctx, &outBuff, &inBuff); - if (ZSTD_isError(readSizeHint)) EXM_THROW(36, "Decoding error : %s", ZSTD_getErrorName(readSizeHint)); + if (ZSTD_isError(readSizeHint)) { + DISPLAYLEVEL(1, "%s : Decoding error (36) : %s \n", + srcFileName, ZSTD_getErrorName(readSizeHint)); + return FIO_ERROR_ZSTD_DECODING; + } /* Write block */ storedSkips = FIO_fwriteSparse(ress->dstFile, ress->dstBuffer, outBuff.pos, storedSkips); frameSize += outBuff.pos; - DISPLAYUPDATE(2, "\rDecoded : %u MB... ", (U32)((alreadyDecoded+frameSize)>>20) ); + DISPLAYUPDATE(2, "\r%-20.20s : %u MB... ", + srcFileName, (U32)((alreadyDecoded+frameSize)>>20) ); if (inBuff.pos > 0) { memmove(ress->srcBuffer, (char*)ress->srcBuffer + inBuff.pos, inBuff.size - inBuff.pos); @@ -1018,14 +1026,22 @@ unsigned long long FIO_decompressFrame(dRess_t* ress, } if (readSizeHint == 0) break; /* end of frame */ - if (inBuff.size != inBuff.pos) EXM_THROW(37, "Decoding error : should consume entire input"); + if (inBuff.size != inBuff.pos) { + DISPLAYLEVEL(1, "%s : Decoding error (37) : should consume entire input \n", + srcFileName); + return FIO_ERROR_ZSTD_DECODING; + } /* Fill input buffer */ { size_t const toRead = MIN(readSizeHint, ress->srcBufferSize); /* support large skippable frames */ if (ress->srcBufferLoaded < toRead) - ress->srcBufferLoaded += fread(((char*)ress->srcBuffer) + ress->srcBufferLoaded, 1, toRead - ress->srcBufferLoaded, finput); - if (ress->srcBufferLoaded < toRead) EXM_THROW(39, "Read error : premature end"); - } } + ress->srcBufferLoaded += fread((char*)ress->srcBuffer + ress->srcBufferLoaded, + 1, toRead - ress->srcBufferLoaded, finput); + if (ress->srcBufferLoaded < toRead) { + DISPLAYLEVEL(1, "%s : Read error (39) : premature end \n", + srcFileName); + return FIO_ERROR_ZSTD_DECODING; + } } } FIO_fwriteSparseEnd(ress->dstFile, storedSkips); @@ -1221,9 +1237,14 @@ static int FIO_decompressSrcFile(dRess_t ress, const char* dstFileName, const ch size_t const toRead = 4; const BYTE* buf = (const BYTE*)ress.srcBuffer; if (ress.srcBufferLoaded < toRead) - ress.srcBufferLoaded += fread((char*)ress.srcBuffer + ress.srcBufferLoaded, (size_t)1, toRead - ress.srcBufferLoaded, srcFile); + ress.srcBufferLoaded += fread((char*)ress.srcBuffer + ress.srcBufferLoaded, + (size_t)1, toRead - ress.srcBufferLoaded, srcFile); if (ress.srcBufferLoaded==0) { - if (readSomething==0) { DISPLAY("zstd: %s: unexpected end of file \n", srcFileName); fclose(srcFile); return 1; } /* srcFileName is empty */ + if (readSomething==0) { + DISPLAY("zstd: %s: unexpected end of file \n", srcFileName); + fclose(srcFile); + return 1; + } /* srcFileName is empty */ break; /* no more input */ } readSomething = 1; /* there is at least >= 4 bytes in srcFile */ @@ -1259,7 +1280,8 @@ static int FIO_decompressSrcFile(dRess_t ress, const char* dstFileName, const ch } else { if (!ZSTD_isFrame(ress.srcBuffer, toRead)) { if ((g_overwrite) && !strcmp (dstFileName, stdoutmark)) { /* pass-through mode */ - unsigned const result = FIO_passThrough(ress.dstFile, srcFile, ress.srcBuffer, ress.srcBufferSize, ress.srcBufferLoaded); + unsigned const result = FIO_passThrough(ress.dstFile, srcFile, + ress.srcBuffer, ress.srcBufferSize, ress.srcBufferLoaded); if (fclose(srcFile)) EXM_THROW(32, "zstd: %s close error", srcFileName); /* error should never happen */ return result; } else { @@ -1267,9 +1289,15 @@ static int FIO_decompressSrcFile(dRess_t ress, const char* dstFileName, const ch fclose(srcFile); return 1; } } - filesize += FIO_decompressFrame(&ress, srcFile, filesize); + { unsigned long long const frameSize = FIO_decompressFrame(&ress, srcFile, srcFileName, filesize); + if (frameSize == FIO_ERROR_ZSTD_DECODING) { + fclose(srcFile); + return 1; + } + filesize += frameSize; + } } - } + } /* for each frame */ /* Final Status */ DISPLAYLEVEL(2, "\r%79s\r", ""); @@ -1364,15 +1392,20 @@ int FIO_decompressMultipleFilenames(const char** srcNamesTable, unsigned nbFiles dstFileName = (char*)malloc(dfnSize); if (dstFileName==NULL) EXM_THROW(74, "not enough memory for dstFileName"); } - if (sfnSize <= suffixSize || (strcmp(suffixPtr, GZ_EXTENSION) && strcmp(suffixPtr, XZ_EXTENSION) && strcmp(suffixPtr, ZSTD_EXTENSION) && strcmp(suffixPtr, LZMA_EXTENSION) && strcmp(suffixPtr, LZ4_EXTENSION))) { - DISPLAYLEVEL(1, "zstd: %s: unknown suffix (%s/%s/%s/%s expected) -- ignored \n", srcFileName, GZ_EXTENSION, XZ_EXTENSION, ZSTD_EXTENSION, LZMA_EXTENSION); + if (sfnSize <= suffixSize + || (strcmp(suffixPtr, GZ_EXTENSION) + && strcmp(suffixPtr, XZ_EXTENSION) + && strcmp(suffixPtr, ZSTD_EXTENSION) + && strcmp(suffixPtr, LZMA_EXTENSION) + && strcmp(suffixPtr, LZ4_EXTENSION)) ) { + DISPLAYLEVEL(1, "zstd: %s: unknown suffix (%s/%s/%s/%s/%s expected) -- ignored \n", + srcFileName, GZ_EXTENSION, XZ_EXTENSION, ZSTD_EXTENSION, LZMA_EXTENSION, LZ4_EXTENSION); skippedFiles++; continue; } else { memcpy(dstFileName, srcFileName, sfnSize - suffixSize); dstFileName[sfnSize-suffixSize] = '\0'; } - missingFiles += FIO_decompressDstFile(ress, dstFileName, srcFileName); } free(dstFileName); diff --git a/tests/playTests.sh b/tests/playTests.sh index 021fd59fe..e69574588 100755 --- a/tests/playTests.sh +++ b/tests/playTests.sh @@ -92,7 +92,7 @@ $ZSTD tmp --stdout > tmpCompressed # long command format $ECHO "test : compress to named file" rm tmpCompressed $ZSTD tmp -o tmpCompressed -ls tmpCompressed # must work +test -f tmpCompressed # file must be created $ECHO "test : -o must be followed by filename (must fail)" $ZSTD tmp -of tmpCompressed && die "-o must be followed by filename " $ECHO "test : force write, correct order" @@ -142,21 +142,21 @@ $ZSTD -q -f tmpro rm -f tmpro tmpro.zst $ECHO "test : file removal" $ZSTD -f --rm tmp -ls tmp && die "tmp should no longer be present" +test ! -f tmp # tmp should no longer be present $ZSTD -f -d --rm tmp.zst -ls tmp.zst && die "tmp.zst should no longer be present" +test ! -f tmp.zst # tmp.zst should no longer be present $ECHO "test : --rm on stdin" $ECHO a | $ZSTD --rm > $INTOVOID # --rm should remain silent rm tmp $ZSTD -f tmp && die "tmp not present : should have failed" -ls tmp.zst && die "tmp.zst should not be created" +test ! -f tmp.zst # tmp.zst should not be created $ECHO "\n**** Advanced compression parameters **** " $ECHO "Hello world!" | $ZSTD --zstd=windowLog=21, - -o tmp.zst && die "wrong parameters not detected!" $ECHO "Hello world!" | $ZSTD --zstd=windowLo=21 - -o tmp.zst && die "wrong parameters not detected!" $ECHO "Hello world!" | $ZSTD --zstd=windowLog=21,slog - -o tmp.zst && die "wrong parameters not detected!" -ls tmp.zst && die "tmp.zst should not be created" +test ! -f tmp.zst # tmp.zst should not be created roundTripTest -g512K roundTripTest -g512K " --zstd=slen=3,tlen=48,strat=6" roundTripTest -g512K " --zstd=strat=6,wlog=23,clog=23,hlog=22,slog=6" @@ -201,16 +201,17 @@ $ECHO foo | $ZSTD > /dev/full && die "write error not detected!" $ECHO "$ECHO foo | $ZSTD | $ZSTD -d > /dev/full" $ECHO foo | $ZSTD | $ZSTD -d > /dev/full && die "write error not detected!" + $ECHO "\n**** symbolic link test **** " rm -f hello.tmp world.tmp hello.tmp.zst world.tmp.zst $ECHO "hello world" > hello.tmp ln -s hello.tmp world.tmp $ZSTD world.tmp hello.tmp -ls hello.tmp.zst || die "regular file should have been compressed!" -ls world.tmp.zst && die "symbolic link should not have been compressed!" +test -f hello.tmp.zst # regular file should have been compressed! +test ! -f world.tmp.zst # symbolic link should not have been compressed! $ZSTD world.tmp hello.tmp -f -ls world.tmp.zst || die "symbol link should have been compressed with --force" +test -f world.tmp.zst # symbolic link should have been compressed with --force rm -f hello.tmp world.tmp hello.tmp.zst world.tmp.zst fi @@ -225,10 +226,10 @@ $ZSTD tmpSparse -c | $ZSTD -dv --sparse -c > tmpOutSparse $DIFF -s tmpSparse tmpOutSparse $ZSTD tmpSparse -c | $ZSTD -dv --no-sparse -c > tmpOutNoSparse $DIFF -s tmpSparse tmpOutNoSparse -ls -ls tmpSparse* +ls -ls tmpSparse* # look at file size and block size on disk ./datagen -s1 -g1200007 -P100 | $ZSTD | $ZSTD -dv --sparse -c > tmpSparseOdd # Odd size file (to not finish on an exact nb of blocks) ./datagen -s1 -g1200007 -P100 | $DIFF -s - tmpSparseOdd -ls -ls tmpSparseOdd +ls -ls tmpSparseOdd # look at file size and block size on disk $ECHO "\n Sparse Compatibility with Console :" $ECHO "Hello World 1 !" | $ZSTD | $ZSTD -d -c $ECHO "Hello World 2 !" | $ZSTD | $ZSTD -d | cat @@ -238,7 +239,7 @@ cat tmpSparse1M tmpSparse1M > tmpSparse2M $ZSTD -v -f tmpSparse1M -o tmpSparseCompressed $ZSTD -d -v -f tmpSparseCompressed -o tmpSparseRegenerated $ZSTD -d -v -f tmpSparseCompressed -c >> tmpSparseRegenerated -ls -ls tmpSparse* +ls -ls tmpSparse* # look at file size and block size on disk $DIFF tmpSparse2M tmpSparseRegenerated rm tmpSparse* @@ -257,11 +258,11 @@ $ZSTD -df *.zst ls -ls tmp* $ECHO "compress tmp* into stdout > tmpall : " $ZSTD -c tmp1 tmp2 tmp3 > tmpall -ls -ls tmp* +ls -ls tmp* # check size of tmpall (should be tmp1.zst + tmp2.zst + tmp3.zst) $ECHO "decompress tmpall* into stdout > tmpdec : " cp tmpall tmpall2 $ZSTD -dc tmpall* > tmpdec -ls -ls tmp* +ls -ls tmp* # check size of tmpdec (should be 2*(tmp1 + tmp2 + tmp3)) $ECHO "compress multiple files including a missing one (notHere) : " $ZSTD -f tmp1 notHere tmp2 && die "missing file not detected!" @@ -379,7 +380,9 @@ $ZSTD -t tmp2.zst && die "bad file not detected !" $ZSTD -t tmp3 && die "bad file not detected !" # detects 0-sized files as bad $ECHO "test --rm and --test combined " $ZSTD -t --rm tmp1.zst -ls -ls tmp1.zst # check file is still present +test -f tmp1.zst # check file is still present +split -b16384 tmp1.zst tmpSplit. +$ZSTD -t tmpSplit.* && die "bad file not detected !" $ECHO "\n**** benchmark mode tests **** " From 0be6fd342984d375965255bff77f8936bcd85f8a Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Mon, 8 May 2017 16:08:01 -0700 Subject: [PATCH 291/305] merged CCtx and CStream as a single same object To be changed : ZSTD_sizeof_CCtx(), ZSTD_estimateCCtxSize() --- doc/zstd_manual.html | 2 + lib/compress/zstd_compress.c | 124 +++++++++++++++++++---------------- lib/zstd.h | 3 +- tests/zstreamtest.c | 18 +++-- 4 files changed, 84 insertions(+), 63 deletions(-) diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index 2e77e7742..8cac4294f 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -244,6 +244,8 @@ size_t ZSTD_freeDCtx(ZSTD_DCtx* dctx);

+
typedef ZSTD_CCtx ZSTD_CStream;  /**< CCtx and CStream are effectively same object */
+

ZSTD_CStream management functions

ZSTD_CStream* ZSTD_createCStream(void);
 size_t ZSTD_freeCStream(ZSTD_CStream* zcs);
 

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index c08b315da..d7f90dff8 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -79,6 +79,8 @@ static void ZSTD_resetSeqStore(seqStore_t* ssPtr) /*-************************************* * Context memory management ***************************************/ +typedef enum { zcss_init, zcss_load, zcss_flush, zcss_final } ZSTD_cStreamStage; + struct ZSTD_CCtx_s { const BYTE* nextSrc; /* next block here to continue on current prefix */ const BYTE* base; /* All regular indexes relative to this position */ @@ -115,6 +117,22 @@ struct ZSTD_CCtx_s { FSE_CTable* matchlengthCTable; FSE_CTable* litlengthCTable; unsigned* entropyScratchSpace; + + /* streaming */ + ZSTD_CDict* cdictLocal; + const ZSTD_CDict* cdict; + char* inBuff; + size_t inBuffSize; + size_t inToCompress; + size_t inBuffPos; + size_t inBuffTarget; + char* outBuff; + size_t outBuffSize; + size_t outBuffContentSize; + size_t outBuffFlushedSize; + ZSTD_cStreamStage streamStage; + U32 frameEnded; + U64 pledgedSrcSize; }; ZSTD_CCtx* ZSTD_createCCtx(void) @@ -253,8 +271,8 @@ static U32 ZSTD_equivalentParams(ZSTD_parameters param1, ZSTD_parameters param2) { return (param1.cParams.hashLog == param2.cParams.hashLog) & (param1.cParams.chainLog == param2.cParams.chainLog) - & (param1.cParams.strategy == param2.cParams.strategy) - & ((param1.cParams.searchLength==3) == (param2.cParams.searchLength==3)); + & (param1.cParams.strategy == param2.cParams.strategy) /* opt parser space */ + & ((param1.cParams.searchLength==3) == (param2.cParams.searchLength==3)); /* hashlog3 space */ } /*! ZSTD_continueCCtx() : @@ -285,8 +303,11 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc, ZSTD_parameters params, U64 frameContentSize, ZSTD_compResetPolicy_e const crp) { + DEBUGLOG(5, "ZSTD_resetCCtx_internal \n"); + if (crp == ZSTDcrp_continue) if (ZSTD_equivalentParams(params, zc->params)) { + DEBUGLOG(5, "ZSTD_equivalentParams()==1 \n"); zc->fseCTables_ready = 0; zc->hufCTable_repeatMode = HUF_repeat_none; return ZSTD_continueCCtx(zc, params, frameContentSize); @@ -312,6 +333,8 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc, size_t const optSpace = ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) ? optPotentialSpace : 0; size_t const neededSpace = entropySpace + optSpace + tableSpace + tokenSpace; if (zc->workSpaceSize < neededSpace) { + DEBUGLOG(5, "Need to update workSpaceSize from %uK to %uK \n", + (unsigned)zc->workSpaceSize>>10, (unsigned)neededSpace>>10); zc->workSpaceSize = 0; ZSTD_free(zc->workSpace, zc->customMem); zc->workSpace = ZSTD_malloc(neededSpace, zc->customMem); @@ -335,6 +358,7 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc, /* init params */ zc->params = params; zc->blockSize = blockSize; + DEBUGLOG(5, "blockSize = %uK \n", (U32)blockSize>>10); zc->frameContentSize = frameContentSize; zc->consumedSrcSize = 0; @@ -364,6 +388,7 @@ static size_t ZSTD_resetCCtx_internal (ZSTD_CCtx* zc, /* opt parser space */ if ((params.cParams.strategy == ZSTD_btopt) || (params.cParams.strategy == ZSTD_btopt2)) { + DEBUGLOG(5, "reserving optimal parser space "); assert(((size_t)ptr & 3) == 0); /* ensure ptr is properly aligned */ zc->seqStore.litFreq = (U32*)ptr; zc->seqStore.litLengthFreq = zc->seqStore.litFreq + (1<stage!=ZSTDcs_init) return ERROR(stage_wrong); memcpy(&dstCCtx->customMem, &srcCCtx->customMem, sizeof(ZSTD_customMem)); @@ -2528,9 +2554,8 @@ static size_t ZSTD_writeFrameHeader(void* dst, size_t dstCapacity, size_t pos; if (dstCapacity < ZSTD_frameHeaderSize_max) return ERROR(dstSize_tooSmall); - DEBUGLOG(5, "ZSTD_writeFrameHeader : dictIDFlag : %u \n", !params.fParams.noDictIDFlag); - DEBUGLOG(5, "ZSTD_writeFrameHeader : dictID : %u \n", dictID); - DEBUGLOG(5, "ZSTD_writeFrameHeader : dictIDSizeCode : %u \n", dictIDSizeCode); + DEBUGLOG(5, "ZSTD_writeFrameHeader : dictIDFlag : %u ; dictID : %u ; dictIDSizeCode : %u \n", + !params.fParams.noDictIDFlag, dictID, dictIDSizeCode); MEM_writeLE32(dst, ZSTD_MAGICNUMBER); op[4] = frameHeaderDecriptionByte; pos=5; @@ -2840,6 +2865,7 @@ static size_t ZSTD_writeEpilogue(ZSTD_CCtx* cctx, void* dst, size_t dstCapacity) BYTE* op = ostart; size_t fhSize = 0; + DEBUGLOG(5, "ZSTD_writeEpilogue \n"); if (cctx->stage == ZSTDcs_created) return ERROR(stage_wrong); /* init missing */ /* special case : empty frame */ @@ -2877,12 +2903,14 @@ size_t ZSTD_compressEnd (ZSTD_CCtx* cctx, const void* src, size_t srcSize) { size_t endResult; - size_t const cSize = ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, 1 /* frame mode */, 1 /* last chunk */); + size_t const cSize = ZSTD_compressContinue_internal(cctx, dst, dstCapacity, src, srcSize, + 1 /* frame mode */, 1 /* last chunk */); if (ZSTD_isError(cSize)) return cSize; endResult = ZSTD_writeEpilogue(cctx, (char*)dst + cSize, dstCapacity-cSize); if (ZSTD_isError(endResult)) return endResult; if (cctx->params.fParams.contentSizeFlag) { /* control src size */ - if (cctx->frameContentSize != cctx->consumedSrcSize) return ERROR(srcSize_wrong); + if (cctx->frameContentSize != cctx->consumedSrcSize) + return ERROR(srcSize_wrong); } return cSize + endResult; } @@ -3084,30 +3112,6 @@ size_t ZSTD_compress_usingCDict(ZSTD_CCtx* cctx, * Streaming ********************************************************************/ -typedef enum { zcss_init, zcss_load, zcss_flush, zcss_final } ZSTD_cStreamStage; - -struct ZSTD_CStream_s { - ZSTD_CCtx* cctx; - ZSTD_CDict* cdictLocal; - const ZSTD_CDict* cdict; - char* inBuff; - size_t inBuffSize; - size_t inToCompress; - size_t inBuffPos; - size_t inBuffTarget; - size_t blockSize; - char* outBuff; - size_t outBuffSize; - size_t outBuffContentSize; - size_t outBuffFlushedSize; - ZSTD_cStreamStage stage; - U32 checksum; - U32 frameEnded; - U64 pledgedSrcSize; - ZSTD_parameters params; - ZSTD_customMem customMem; -}; /* typedef'd to ZSTD_CStream within "zstd.h" */ - ZSTD_CStream* ZSTD_createCStream(void) { return ZSTD_createCStream_advanced(defaultCustomMem); @@ -3124,8 +3128,6 @@ ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem) if (zcs==NULL) return NULL; memset(zcs, 0, sizeof(ZSTD_CStream)); memcpy(&zcs->customMem, &customMem, sizeof(ZSTD_customMem)); - zcs->cctx = ZSTD_createCCtx_advanced(customMem); - if (zcs->cctx == NULL) { ZSTD_freeCStream(zcs); return NULL; } return zcs; } @@ -3133,8 +3135,6 @@ size_t ZSTD_freeCStream(ZSTD_CStream* zcs) { if (zcs==NULL) return 0; /* support free on NULL */ { ZSTD_customMem const cMem = zcs->customMem; - ZSTD_freeCCtx(zcs->cctx); - zcs->cctx = NULL; ZSTD_freeCDict(zcs->cdictLocal); zcs->cdictLocal = NULL; ZSTD_free(zcs->inBuff, cMem); @@ -3156,20 +3156,20 @@ size_t ZSTD_CStreamOutSize(void) return ZSTD_compressBound(ZSTD_BLOCKSIZE_ABSOLUTEMAX) + ZSTD_blockHeaderSize + 4 /* 32-bits hash */ ; } -static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize) +static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, ZSTD_parameters params, unsigned long long pledgedSrcSize) { if (zcs->inBuffSize==0) return ERROR(stage_wrong); /* zcs has not been init at least once => can't reset */ DEBUGLOG(5, "ZSTD_resetCStream_internal : dictIDFlag == %u \n", !zcs->params.fParams.noDictIDFlag); - if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict_advanced(zcs->cctx, zcs->cdict, zcs->params.fParams, pledgedSrcSize)) - else CHECK_F(ZSTD_compressBegin_internal(zcs->cctx, NULL, 0, zcs->params, pledgedSrcSize)); + if (zcs->cdict) CHECK_F(ZSTD_compressBegin_usingCDict_advanced(zcs, zcs->cdict, params.fParams, pledgedSrcSize)) + else CHECK_F(ZSTD_compressBegin_internal(zcs, NULL, 0, params, pledgedSrcSize)); zcs->inToCompress = 0; zcs->inBuffPos = 0; zcs->inBuffTarget = zcs->blockSize; zcs->outBuffContentSize = zcs->outBuffFlushedSize = 0; - zcs->stage = zcss_load; + zcs->streamStage = zcss_load; zcs->frameEnded = 0; zcs->pledgedSrcSize = pledgedSrcSize; return 0; /* ready to go */ @@ -3178,9 +3178,10 @@ static size_t ZSTD_resetCStream_internal(ZSTD_CStream* zcs, unsigned long long p size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledgedSrcSize) { - zcs->params.fParams.contentSizeFlag = (pledgedSrcSize > 0); + ZSTD_parameters params = zcs->params; + params.fParams.contentSizeFlag = (pledgedSrcSize > 0); DEBUGLOG(5, "ZSTD_resetCStream : dictIDFlag == %u \n", !zcs->params.fParams.noDictIDFlag); - return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); + return ZSTD_resetCStream_internal(zcs, params, pledgedSrcSize); } /* ZSTD_initCStream_internal() : @@ -3212,11 +3213,8 @@ static size_t ZSTD_initCStream_stage2(ZSTD_CStream* zcs, zcs->outBuffSize = outBuffSize; } - zcs->checksum = params.fParams.checksumFlag > 0; - zcs->params = params; - DEBUGLOG(5, "ZSTD_initCStream_stage2 : dictIDFlag == %u \n", !params.fParams.noDictIDFlag); - return ZSTD_resetCStream_internal(zcs, pledgedSrcSize); + return ZSTD_resetCStream_internal(zcs, params, pledgedSrcSize); } /* ZSTD_initCStream_usingCDict_advanced() : @@ -3261,7 +3259,8 @@ size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, ZSTD_parameters params, unsigned long long pledgedSrcSize) { CHECK_F( ZSTD_checkCParams(params.cParams) ); - DEBUGLOG(5, "ZSTD_initCStream_advanced : dictIDFlag == %u \n", !params.fParams.noDictIDFlag); + DEBUGLOG(5, "ZSTD_initCStream_advanced : pledgedSrcSize == %u \n", (U32)pledgedSrcSize); + DEBUGLOG(5, "wlog %u \n", params.cParams.windowLog); return ZSTD_initCStream_internal(zcs, dict, dictSize, params, pledgedSrcSize); } @@ -3287,7 +3286,7 @@ size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel) size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs) { if (zcs==NULL) return 0; /* support sizeof on NULL */ - return sizeof(*zcs) + ZSTD_sizeof_CCtx(zcs->cctx) + ZSTD_sizeof_CDict(zcs->cdictLocal) + zcs->outBuffSize + zcs->inBuffSize; + return sizeof(*zcs) + ZSTD_sizeof_CDict(zcs->cdictLocal) + zcs->outBuffSize + zcs->inBuffSize; } /*====== Compression ======*/ @@ -3314,8 +3313,9 @@ static size_t ZSTD_compressStream_generic(ZSTD_CStream* zcs, char* const oend = ostart + *dstCapacityPtr; char* op = ostart; + DEBUGLOG(5, "ZSTD_compressStream_generic \n"); while (someMoreWork) { - switch(zcs->stage) + switch(zcs->streamStage) { case zcss_init: return ERROR(init_missing); /* call ZBUFF_compressInit() first ! */ @@ -3323,12 +3323,14 @@ static size_t ZSTD_compressStream_generic(ZSTD_CStream* zcs, /* complete inBuffer */ { size_t const toLoad = zcs->inBuffTarget - zcs->inBuffPos; size_t const loaded = ZSTD_limitCopy(zcs->inBuff + zcs->inBuffPos, toLoad, ip, iend-ip); + DEBUGLOG(5, "loading %u/%u \n", (U32)loaded, (U32)toLoad); zcs->inBuffPos += loaded; ip += loaded; if ( (zcs->inBuffPos==zcs->inToCompress) || (!flush && (toLoad != loaded)) ) { someMoreWork = 0; break; /* not enough input to get a full block : stop there, wait for more */ } } /* compress current block (note : this stage cannot be stopped in the middle) */ + DEBUGLOG(5, "stream compression stage (flush==%u)\n", flush); { void* cDst; size_t cSize; size_t const iSize = zcs->inBuffPos - zcs->inToCompress; @@ -3338,29 +3340,33 @@ static size_t ZSTD_compressStream_generic(ZSTD_CStream* zcs, else cDst = zcs->outBuff, oSize = zcs->outBuffSize; cSize = (flush == zsf_end) ? - ZSTD_compressEnd(zcs->cctx, cDst, oSize, zcs->inBuff + zcs->inToCompress, iSize) : - ZSTD_compressContinue(zcs->cctx, cDst, oSize, zcs->inBuff + zcs->inToCompress, iSize); + ZSTD_compressEnd(zcs, cDst, oSize, zcs->inBuff + zcs->inToCompress, iSize) : + ZSTD_compressContinue(zcs, cDst, oSize, zcs->inBuff + zcs->inToCompress, iSize); if (ZSTD_isError(cSize)) return cSize; + DEBUGLOG(5, "cSize = %u \n", (U32)cSize); if (flush == zsf_end) zcs->frameEnded = 1; /* prepare next block */ zcs->inBuffTarget = zcs->inBuffPos + zcs->blockSize; if (zcs->inBuffTarget > zcs->inBuffSize) - zcs->inBuffPos = 0, zcs->inBuffTarget = zcs->blockSize; /* note : inBuffSize >= blockSize */ + zcs->inBuffPos = 0, zcs->inBuffTarget = zcs->blockSize; /* note : inBuffTarget == blockSize <= inBuffSize */ + assert(zcs->inBuffTarget <= zcs->inBuffSize); zcs->inToCompress = zcs->inBuffPos; if (cDst == op) { op += cSize; break; } /* no need to flush */ zcs->outBuffContentSize = cSize; zcs->outBuffFlushedSize = 0; - zcs->stage = zcss_flush; /* pass-through to flush stage */ + zcs->streamStage = zcss_flush; /* pass-through to flush stage */ } case zcss_flush: + DEBUGLOG(5, "flush stage \n"); { size_t const toFlush = zcs->outBuffContentSize - zcs->outBuffFlushedSize; size_t const flushed = ZSTD_limitCopy(op, oend-op, zcs->outBuff + zcs->outBuffFlushedSize, toFlush); + DEBUGLOG(5, "toFlush: %u ; flushed: %u \n", (U32)toFlush, (U32)flushed); op += flushed; zcs->outBuffFlushedSize += flushed; if (toFlush!=flushed) { someMoreWork = 0; break; } /* dst too small to store flushed data : stop there */ zcs->outBuffContentSize = zcs->outBuffFlushedSize = 0; - zcs->stage = zcss_load; + zcs->streamStage = zcss_load; break; } @@ -3419,7 +3425,8 @@ size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output) BYTE* const oend = (BYTE*)(output->dst) + output->size; BYTE* op = ostart; - if (zcs->stage != zcss_final) { + DEBUGLOG(5, "ZSTD_endStream (dstCapacity : %u) \n", (U32)(oend-op)); + if (zcs->streamStage != zcss_final) { /* flush whatever remains */ size_t srcSize = 0; size_t sizeWritten = output->size - output->pos; @@ -3429,13 +3436,14 @@ size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output) op += sizeWritten; if (remainingToFlush) { output->pos += sizeWritten; - return remainingToFlush + ZSTD_BLOCKHEADERSIZE /* final empty block */ + (zcs->checksum * 4); + return remainingToFlush + ZSTD_BLOCKHEADERSIZE /* final empty block */ + + ((zcs->params.fParams.checksumFlag > 0) * 4) /* optional 32-bits checksum */; } /* create epilogue */ - zcs->stage = zcss_final; + zcs->streamStage = zcss_final; zcs->outBuffContentSize = !notEnded ? 0 : /* write epilogue, including final empty block, into outBuff */ - ZSTD_compressEnd(zcs->cctx, zcs->outBuff, zcs->outBuffSize, NULL, 0); + ZSTD_compressEnd(zcs, zcs->outBuff, zcs->outBuffSize, NULL, 0); if (ZSTD_isError(zcs->outBuffContentSize)) return zcs->outBuffContentSize; } @@ -3445,7 +3453,7 @@ size_t ZSTD_endStream(ZSTD_CStream* zcs, ZSTD_outBuffer* output) op += flushed; zcs->outBuffFlushedSize += flushed; output->pos += op-ostart; - if (toFlush==flushed) zcs->stage = zcss_init; /* end reached */ + if (toFlush==flushed) zcs->streamStage = zcss_init; /* end reached */ return toFlush - flushed; } } diff --git a/lib/zstd.h b/lib/zstd.h index f8050c136..3179c8c78 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -281,7 +281,8 @@ typedef struct ZSTD_outBuffer_s { * * *******************************************************************/ -typedef struct ZSTD_CStream_s ZSTD_CStream; +typedef ZSTD_CCtx ZSTD_CStream; /**< CCtx and CStream are effectively same object */ + /* Continue due distinghish them for compatibility with versions <= v1.2.0 */ /*===== ZSTD_CStream management functions =====*/ ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream(void); ZSTDLIB_API size_t ZSTD_freeCStream(ZSTD_CStream* zcs); diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c index 0e09e1856..6c210f2ff 100644 --- a/tests/zstreamtest.c +++ b/tests/zstreamtest.c @@ -53,13 +53,15 @@ static const U32 prime32 = 2654435761U; * Display Macros **************************************/ #define DISPLAY(...) fprintf(stderr, __VA_ARGS__) -#define DISPLAYLEVEL(l, ...) if (g_displayLevel>=l) { DISPLAY(__VA_ARGS__); } +#define DISPLAYLEVEL(l, ...) if (g_displayLevel>=l) { \ + DISPLAY(__VA_ARGS__); \ + if (g_displayLevel>=4) fflush(stderr); } static U32 g_displayLevel = 2; #define DISPLAYUPDATE(l, ...) if (g_displayLevel>=l) { \ if ((FUZ_GetClockSpan(g_displayClock) > g_refreshRate) || (g_displayLevel>=4)) \ { g_displayClock = clock(); DISPLAY(__VA_ARGS__); \ - if (g_displayLevel>=4) fflush(stderr); } } + if (g_displayLevel>=4) fflush(stderr); } } static const clock_t g_refreshRate = CLOCKS_PER_SEC / 6; static clock_t g_displayClock = 0; @@ -155,7 +157,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo void* decodedBuffer = malloc(decodedBufferSize); size_t cSize; int testResult = 0; - U32 testNb=0; + U32 testNb = 1; ZSTD_CStream* zc = ZSTD_createCStream_advanced(customMem); ZSTD_DStream* zd = ZSTD_createDStream_advanced(customMem); ZSTD_inBuffer inBuff, inBuff2; @@ -344,6 +346,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo if (zc==NULL) goto _output_error; /* memory allocation issue */ /* use 1 */ { size_t const inSize = 513; + DISPLAYLEVEL(5, "use1 "); ZSTD_initCStream_advanced(zc, NULL, 0, ZSTD_getParams(19, inSize, 0), inSize); /* needs btopt + search3 to trigger hashLog3 */ inBuff.src = CNBuffer; inBuff.size = inSize; @@ -351,14 +354,17 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo outBuff.dst = (char*)(compressedBuffer)+cSize; outBuff.size = ZSTD_compressBound(inSize); outBuff.pos = 0; + DISPLAYLEVEL(5, "compress1 "); { size_t const r = ZSTD_compressStream(zc, &outBuff, &inBuff); if (ZSTD_isError(r)) goto _output_error; } if (inBuff.pos != inBuff.size) goto _output_error; /* entire input should be consumed */ + DISPLAYLEVEL(5, "end1 "); { size_t const r = ZSTD_endStream(zc, &outBuff); if (r != 0) goto _output_error; } /* error, or some data not flushed */ } /* use 2 */ { size_t const inSize = 1025; /* will not continue, because tables auto-adjust and are therefore different size */ + DISPLAYLEVEL(5, "use2 "); ZSTD_initCStream_advanced(zc, NULL, 0, ZSTD_getParams(19, inSize, 0), inSize); /* needs btopt + search3 to trigger hashLog3 */ inBuff.src = CNBuffer; inBuff.size = inSize; @@ -366,9 +372,11 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo outBuff.dst = (char*)(compressedBuffer)+cSize; outBuff.size = ZSTD_compressBound(inSize); outBuff.pos = 0; + DISPLAYLEVEL(5, "compress2 "); { size_t const r = ZSTD_compressStream(zc, &outBuff, &inBuff); if (ZSTD_isError(r)) goto _output_error; } if (inBuff.pos != inBuff.size) goto _output_error; /* entire input should be consumed */ + DISPLAYLEVEL(5, "end2 "); { size_t const r = ZSTD_endStream(zc, &outBuff); if (r != 0) goto _output_error; } /* error, or some data not flushed */ } @@ -770,7 +778,9 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, double compres outBuff.size = outBuff.pos + adjustedDstSize; remainingToFlush = ZSTD_endStream(zc, &outBuff); CHECK (ZSTD_isError(remainingToFlush), "flush error : %s", ZSTD_getErrorName(remainingToFlush)); - CHECK (enoughDstSize && remainingToFlush, "ZSTD_endStream() not fully flushed (%u remaining), but enough space available", (U32)remainingToFlush); + CHECK (enoughDstSize && remainingToFlush, + "ZSTD_endStream() not fully flushed (%u remaining), but enough space available (%u)", + (U32)remainingToFlush, (U32)adjustedDstSize); } } crcOrig = XXH64_digest(&xxhState); cSize = outBuff.pos; From 791d74427925351c776de1e94ea701a41e894e20 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Mon, 8 May 2017 16:17:30 -0700 Subject: [PATCH 292/305] Updated ZSTD_sizeof_CCtx() can now contain buffers if object used as CStream. ZSTD_sizeof_CStream() is now just a thin wrapper of ZSTD_sizeof_CCtx(). --- lib/compress/zstd_compress.c | 7 ++++--- lib/zstd.h | 4 ++-- 2 files changed, 6 insertions(+), 5 deletions(-) diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c index d7f90dff8..d55ea5059 100644 --- a/lib/compress/zstd_compress.c +++ b/lib/compress/zstd_compress.c @@ -165,7 +165,9 @@ size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx) size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx) { if (cctx==NULL) return 0; /* support sizeof on NULL */ - return sizeof(*cctx) + cctx->workSpaceSize; + return sizeof(*cctx) + cctx->workSpaceSize + + ZSTD_sizeof_CDict(cctx->cdictLocal) + + cctx->outBuffSize + cctx->inBuffSize; } size_t ZSTD_setCCtxParameter(ZSTD_CCtx* cctx, ZSTD_CCtxParameter param, unsigned value) @@ -3285,8 +3287,7 @@ size_t ZSTD_initCStream(ZSTD_CStream* zcs, int compressionLevel) size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs) { - if (zcs==NULL) return 0; /* support sizeof on NULL */ - return sizeof(*zcs) + ZSTD_sizeof_CDict(zcs->cdictLocal) + zcs->outBuffSize + zcs->inBuffSize; + return ZSTD_sizeof_CCtx(zcs); /* same object */ } /*====== Compression ======*/ diff --git a/lib/zstd.h b/lib/zstd.h index 3179c8c78..bb447ee17 100644 --- a/lib/zstd.h +++ b/lib/zstd.h @@ -468,7 +468,7 @@ ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams); ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx_advanced(ZSTD_customMem customMem); /*! ZSTD_sizeofCCtx() : - * Gives the amount of memory used by a given ZSTD_CCtx */ + * amount of used memory is variable, depending primarily on compression level */ ZSTDLIB_API size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx); typedef enum { @@ -597,7 +597,7 @@ ZSTDLIB_API unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize); /*===== Advanced Streaming compression functions =====*/ ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem); -ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs); /**< size of CStream is variable, depending primarily on compression level */ +ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs); /**< same as ZSTD_sizeof_CCtx */ ZSTDLIB_API size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize); /**< pledgedSrcSize must be correct, a size of 0 means unknown. for a frame size of 0 use initCStream_advanced */ ZSTDLIB_API size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */ ZSTDLIB_API size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize, From fc5145955a0c53a9157579ce9dc2abcadeff5e32 Mon Sep 17 00:00:00 2001 From: Yann Collet Date: Mon, 8 May 2017 17:07:59 -0700 Subject: [PATCH 293/305] updated ZSTD_estimateCCtxSize() added a parameter streaming, to estimate memory allocation size when the CCtx is used for streaming (CStream). Note : this function is not able to estimate memory cost of a potential internal CDict which can only happen when starting with ZSTD_initCStream_usingDict() --- doc/zstd_manual.html | 6 +++--- lib/compress/zstd_compress.c | 23 ++++++++++++++--------- lib/zstd.h | 8 +++++--- tests/paramgrill.c | 4 ++-- 4 files changed, 24 insertions(+), 17 deletions(-) diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html index 8cac4294f..d70915458 100644 --- a/doc/zstd_manual.html +++ b/doc/zstd_manual.html @@ -372,7 +372,7 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v

Advanced compression functions


 
-
size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams);
+
size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned streaming);
 

Gives the amount of memory allocated for a ZSTD_CCtx given a set of compression parameters. `frameContentSize` is an optional parameter, provide `0` if unknown


@@ -382,7 +382,7 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v


size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx);
-

Gives the amount of memory used by a given ZSTD_CCtx +

amount of used memory is variable, depending primarily on compression level


typedef enum {
@@ -507,7 +507,7 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v
 

Advanced streaming functions


 
 

Advanced Streaming compression functions

ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem);
-size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);   /**< size of CStream is variable, depending primarily on compression level */
+size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);   /**< same as ZSTD_sizeof_CCtx */
 size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize);   /**< pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
 size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
 size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize,
diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index d55ea5059..1248032bd 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -244,7 +244,7 @@ ZSTD_compressionParameters ZSTD_adjustCParams(ZSTD_compressionParameters cPar, u
 }
 
 
-size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams)
+size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned streaming)
 {
     size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << cParams.windowLog);
     U32    const divider = (cParams.searchLength==3) ? 3 : 4;
@@ -260,12 +260,17 @@ size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams)
                               + entropyScratchSpace_size;
     size_t const tableSpace = (chainSize + hSize + h3Size) * sizeof(U32);
 
-    size_t const optSpace = ((MaxML+1) + (MaxLL+1) + (MaxOff+1) + (1<blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << params.cParams.windowLog);
 
     /* allocate buffers */
-    {   size_t const neededInBuffSize = (size_t)1 << params.cParams.windowLog;
+    {   size_t const neededInBuffSize = ((size_t)1 << params.cParams.windowLog) + zcs->blockSize;
         if (zcs->inBuffSize < neededInBuffSize) {
             zcs->inBuffSize = 0;
             ZSTD_free(zcs->inBuff, zcs->customMem);
-            zcs->inBuff = (char*) ZSTD_malloc(neededInBuffSize, zcs->customMem);
+            zcs->inBuff = (char*)ZSTD_malloc(neededInBuffSize, zcs->customMem);
             if (zcs->inBuff == NULL) return ERROR(memory_allocation);
             zcs->inBuffSize = neededInBuffSize;
         }
-        zcs->blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, neededInBuffSize);
     }
     if (zcs->outBuffSize < ZSTD_compressBound(zcs->blockSize)+1) {
         size_t const outBuffSize = ZSTD_compressBound(zcs->blockSize)+1;
         zcs->outBuffSize = 0;
         ZSTD_free(zcs->outBuff, zcs->customMem);
-        zcs->outBuff = (char*) ZSTD_malloc(outBuffSize, zcs->customMem);
+        zcs->outBuff = (char*)ZSTD_malloc(outBuffSize, zcs->customMem);
         if (zcs->outBuff == NULL) return ERROR(memory_allocation);
         zcs->outBuffSize = outBuffSize;
     }
diff --git a/lib/zstd.h b/lib/zstd.h
index bb447ee17..e8614334b 100644
--- a/lib/zstd.h
+++ b/lib/zstd.h
@@ -460,8 +460,10 @@ ZSTDLIB_API unsigned long long ZSTD_findDecompressedSize(const void* src, size_t
 ***************************************/
 /*! ZSTD_estimateCCtxSize() :
  *  Gives the amount of memory allocated for a ZSTD_CCtx given a set of compression parameters.
- *  `frameContentSize` is an optional parameter, provide `0` if unknown */
-ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams);
+ *  Set streaming to 1 if the CCtx will be used for streaming (CStream)
+ *  Note : this function is currently unable to estimate additional memory allocation needed to create an internal CDict
+ *         which can only happen when starting with ZSTD_initCStream_usingDict() */
+ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned streaming);
 
 /*! ZSTD_createCCtx_advanced() :
  *  Create a ZSTD compression context using external alloc and free functions */
@@ -599,7 +601,7 @@ ZSTDLIB_API unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize);
 ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem);
 ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);   /**< same as ZSTD_sizeof_CCtx */
 ZSTDLIB_API size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize);   /**< pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
-ZSTDLIB_API size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
+ZSTDLIB_API size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8. This result in the creation of an internal CDict */
 ZSTDLIB_API size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize,
                                              ZSTD_parameters params, unsigned long long pledgedSrcSize);  /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */
 ZSTDLIB_API size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict);  /**< note : cdict will just be referenced, and must outlive compression session */
diff --git a/tests/paramgrill.c b/tests/paramgrill.c
index 1913b54d0..3cc916e01 100644
--- a/tests/paramgrill.c
+++ b/tests/paramgrill.c
@@ -388,8 +388,8 @@ static int BMK_seed(winnerInfo_t* winners, const ZSTD_compressionParameters para
             double W_DMemUsed_note = W_ratioNote * ( 40 + 9*cLevel) - log((double)W_DMemUsed);
             double O_DMemUsed_note = O_ratioNote * ( 40 + 9*cLevel) - log((double)O_DMemUsed);
 
-            size_t W_CMemUsed = (1 << params.windowLog) + ZSTD_estimateCCtxSize(params);
-            size_t O_CMemUsed = (1 << winners[cLevel].params.windowLog) + ZSTD_estimateCCtxSize(winners[cLevel].params);
+            size_t W_CMemUsed = (1 << params.windowLog) + ZSTD_estimateCCtxSize(params, 0/*streaming*/);
+            size_t O_CMemUsed = (1 << winners[cLevel].params.windowLog) + ZSTD_estimateCCtxSize(winners[cLevel].params, 0);
             double W_CMemUsed_note = W_ratioNote * ( 50 + 13*cLevel) - log((double)W_CMemUsed);
             double O_CMemUsed_note = O_ratioNote * ( 50 + 13*cLevel) - log((double)O_CMemUsed);
 

From 7855366598f3287974fba8a0ff4d32de56d9efcd Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Mon, 8 May 2017 17:15:00 -0700
Subject: [PATCH 294/305] Updated ZSTD_freeCCtx()

which can also contain streaming buffers now.
Redirected ZSTD_freeCStream() towards it.
---
 lib/compress/zstd_compress.c | 19 ++++++++-----------
 1 file changed, 8 insertions(+), 11 deletions(-)

diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 1248032bd..748aff930 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -158,6 +158,13 @@ size_t ZSTD_freeCCtx(ZSTD_CCtx* cctx)
 {
     if (cctx==NULL) return 0;   /* support free on NULL */
     ZSTD_free(cctx->workSpace, cctx->customMem);
+    cctx->workSpace = NULL;
+    ZSTD_freeCDict(cctx->cdictLocal);
+    cctx->cdictLocal = NULL;
+    ZSTD_free(cctx->inBuff, cctx->customMem);
+    cctx->inBuff = NULL;
+    ZSTD_free(cctx->outBuff, cctx->customMem);
+    cctx->outBuff = NULL;
     ZSTD_free(cctx, cctx->customMem);
     return 0;   /* reserved as a potential error code in the future */
 }
@@ -3140,17 +3147,7 @@ ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem)
 
 size_t ZSTD_freeCStream(ZSTD_CStream* zcs)
 {
-    if (zcs==NULL) return 0;   /* support free on NULL */
-    {   ZSTD_customMem const cMem = zcs->customMem;
-        ZSTD_freeCDict(zcs->cdictLocal);
-        zcs->cdictLocal = NULL;
-        ZSTD_free(zcs->inBuff, cMem);
-        zcs->inBuff = NULL;
-        ZSTD_free(zcs->outBuff, cMem);
-        zcs->outBuff = NULL;
-        ZSTD_free(zcs, cMem);
-        return 0;
-    }
+    return ZSTD_freeCCtx(zcs);   /* same object */
 }
 
 

From a1d6704d7f63fb9ad177a8364a40385206d21b53 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Mon, 8 May 2017 17:51:49 -0700
Subject: [PATCH 295/305] added ZSTD_estimateCDictSize() and
 ZSTD_estimateDDictSize()

it complements ZSTD_estimateCCtxSize()
for the special case of ZSTD_initCStream_usingDict()
---
 doc/zstd_manual.html             | 24 ++++++++++++++++++------
 lib/compress/zstd_compress.c     |  8 ++++++++
 lib/decompress/zstd_decompress.c |  8 ++++++++
 lib/zstd.h                       | 24 +++++++++++++++++-------
 tests/fuzzer.c                   |  6 ++++++
 5 files changed, 57 insertions(+), 13 deletions(-)

diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html
index d70915458..9cfb67100 100644
--- a/doc/zstd_manual.html
+++ b/doc/zstd_manual.html
@@ -372,15 +372,17 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v
 
 

Advanced compression functions


 
-
size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned streaming);
-

Gives the amount of memory allocated for a ZSTD_CCtx given a set of compression parameters. - `frameContentSize` is an optional parameter, provide `0` if unknown -


-
ZSTD_CCtx* ZSTD_createCCtx_advanced(ZSTD_customMem customMem);
 

Create a ZSTD compression context using external alloc and free functions


+
size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned streaming);
+

Provides amount of memory needed to allocate ZSTD_CCtx with a set of compression parameters. + Set streaming to 1 if the CCtx will be used for streaming (CStream). + Special case : when using ZSTD_initCStream_usingDict(), init will transparently create an internal CDict. + Use ZSTD_estimateCDictSize() and add this value to estimate total CCtx size +


+
size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx);
 

amount of used memory is variable, depending primarily on compression level


@@ -406,6 +408,11 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v

Create a ZSTD_CDict using external alloc and free, and customized compression parameters


+
size_t ZSTD_estimateCDictSize(ZSTD_compressionParameters cParams, size_t dictSize);
+

Estimate amount of memory that will be needed to create a dictionary with following arguments + Note : if dictionary is created "byReference", reduce this amount by dictSize +


+
size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict);
 

Gives the amount of memory used by a given ZSTD_sizeof_CDict


@@ -476,6 +483,11 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v

Create a ZSTD_DDict using external alloc and free, optionally by reference


+
size_t ZSTD_estimateDDictSize(size_t dictSize);
+

Estimate amount of memory that will be needed to create a dictionary for decompression. + Note : if dictionary is created "byReference", reduce this amount by dictSize +


+
size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict);
 

Gives the amount of memory used by a given ZSTD_DDict


@@ -509,7 +521,7 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v

Advanced Streaming compression functions

ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem);
 size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);   /**< same as ZSTD_sizeof_CCtx */
 size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize);   /**< pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
-size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
+size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8. This result in the creation of an internal CDict */
 size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize,
                                              ZSTD_parameters params, unsigned long long pledgedSrcSize);  /**< pledgedSrcSize is optional and can be 0 (meaning unknown). note: if the contentSizeFlag is set, pledgedSrcSize == 0 means the source size is actually 0 */
 size_t ZSTD_initCStream_usingCDict(ZSTD_CStream* zcs, const ZSTD_CDict* cdict);  /**< note : cdict will just be referenced, and must outlive compression session */
diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index 748aff930..c5687b21c 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -2984,6 +2984,14 @@ struct ZSTD_CDict_s {
     ZSTD_CCtx* refContext;
 };  /* typedef'd tp ZSTD_CDict within "zstd.h" */
 
+/*! ZSTD_estimateCDictSize() :
+ *  Estimate amount of memory that will be needed to create a dictionary with following arguments */
+size_t ZSTD_estimateCDictSize(ZSTD_compressionParameters cParams, size_t dictSize)
+{
+    cParams = ZSTD_adjustCParams(cParams, 0, dictSize);
+    return sizeof(ZSTD_CDict) + dictSize + ZSTD_estimateCCtxSize(cParams, 0);
+}
+
 size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict)
 {
     if (cdict==NULL) return 0;   /* support sizeof on NULL */
diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c
index 910f9ab78..88488b4b0 100644
--- a/lib/decompress/zstd_decompress.c
+++ b/lib/decompress/zstd_decompress.c
@@ -1969,6 +1969,14 @@ size_t ZSTD_freeDDict(ZSTD_DDict* ddict)
     }
 }
 
+/*! ZSTD_estimateDDictSize() :
+ *  Estimate amount of memory that will be needed to create a dictionary for decompression.
+ *  Note : if dictionary is created "byReference", reduce this amount by dictSize */
+size_t ZSTD_estimateDDictSize(size_t dictSize)
+{
+    return dictSize + sizeof(ZSTD_DDict);
+}
+
 size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict)
 {
     if (ddict==NULL) return 0;   /* support sizeof on NULL */
diff --git a/lib/zstd.h b/lib/zstd.h
index e8614334b..270943b30 100644
--- a/lib/zstd.h
+++ b/lib/zstd.h
@@ -458,17 +458,17 @@ ZSTDLIB_API unsigned long long ZSTD_findDecompressedSize(const void* src, size_t
 /***************************************
 *  Advanced compression functions
 ***************************************/
-/*! ZSTD_estimateCCtxSize() :
- *  Gives the amount of memory allocated for a ZSTD_CCtx given a set of compression parameters.
- *  Set streaming to 1 if the CCtx will be used for streaming (CStream)
- *  Note : this function is currently unable to estimate additional memory allocation needed to create an internal CDict
- *         which can only happen when starting with ZSTD_initCStream_usingDict() */
-ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned streaming);
-
 /*! ZSTD_createCCtx_advanced() :
  *  Create a ZSTD compression context using external alloc and free functions */
 ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx_advanced(ZSTD_customMem customMem);
 
+/*! ZSTD_estimateCCtxSize() :
+ *  Provides amount of memory needed to allocate ZSTD_CCtx with a set of compression parameters.
+ *  Set streaming to 1 if the CCtx will be used for streaming (CStream).
+ *  Special case : when using ZSTD_initCStream_usingDict(), init will transparently create an internal CDict.
+ *         Use ZSTD_estimateCDictSize() and add this value to estimate total CCtx size */
+ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned streaming);
+
 /*! ZSTD_sizeofCCtx() :
  *  amount of used memory is variable, depending primarily on compression level */
 ZSTDLIB_API size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx);
@@ -493,6 +493,11 @@ ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_byReference(const void* dictBuffer, siz
 ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_advanced(const void* dict, size_t dictSize, unsigned byReference,
                                                   ZSTD_compressionParameters cParams, ZSTD_customMem customMem);
 
+/*! ZSTD_estimateCDictSize() :
+ *  Estimate amount of memory that will be needed to create a dictionary with following arguments
+ *  Note : if dictionary is created "byReference", reduce this amount by dictSize */
+ZSTDLIB_API size_t ZSTD_estimateCDictSize(ZSTD_compressionParameters cParams, size_t dictSize);
+
 /*! ZSTD_sizeof_CDict() :
  *  Gives the amount of memory used by a given ZSTD_sizeof_CDict */
 ZSTDLIB_API size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict);
@@ -564,6 +569,11 @@ ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_byReference(const void* dictBuffer, siz
 ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize,
                                                   unsigned byReference, ZSTD_customMem customMem);
 
+/*! ZSTD_estimateDDictSize() :
+ *  Estimate amount of memory that will be needed to create a dictionary for decompression.
+ *  Note : if dictionary is created "byReference", reduce this amount by dictSize */
+ZSTDLIB_API size_t ZSTD_estimateDDictSize(size_t dictSize);
+
 /*! ZSTD_sizeof_DDict() :
  *  Gives the amount of memory used by a given ZSTD_DDict */
 ZSTDLIB_API size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict);
diff --git a/tests/fuzzer.c b/tests/fuzzer.c
index a9dcf12e0..7ffcff26e 100644
--- a/tests/fuzzer.c
+++ b/tests/fuzzer.c
@@ -404,6 +404,12 @@ static int basicUnitTests(U32 seed, double compressibility)
                   if (r != CNBuffSize) goto _output_error);
         DISPLAYLEVEL(4, "OK \n");
 
+        DISPLAYLEVEL(4, "test%3i : estimate CDict size : ", testNb++);
+        {   ZSTD_compressionParameters const cParams = ZSTD_getCParams(1, CNBuffSize, dictSize);
+            size_t const estimatedSize = ZSTD_estimateCDictSize(cParams, dictSize);
+            DISPLAYLEVEL(4, "OK : %u \n", (U32)estimatedSize);
+        }
+
         DISPLAYLEVEL(4, "test%3i : compress with preprocessed dictionary : ", testNb++);
         {   ZSTD_compressionParameters const cParams = ZSTD_getCParams(1, CNBuffSize, dictSize);
             ZSTD_customMem customMem = { NULL, NULL, NULL };

From 51652522a21b5897964019fc158dd86db1fafbd5 Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Mon, 8 May 2017 17:52:46 -0700
Subject: [PATCH 296/305] bumped version number

---
 lib/zstd.h | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/lib/zstd.h b/lib/zstd.h
index 270943b30..c3a8a52a3 100644
--- a/lib/zstd.h
+++ b/lib/zstd.h
@@ -55,7 +55,7 @@ extern "C" {
 
 /*------   Version   ------*/
 #define ZSTD_VERSION_MAJOR    1
-#define ZSTD_VERSION_MINOR    2
+#define ZSTD_VERSION_MINOR    3
 #define ZSTD_VERSION_RELEASE  0
 
 #define ZSTD_LIB_VERSION ZSTD_VERSION_MAJOR.ZSTD_VERSION_MINOR.ZSTD_VERSION_RELEASE

From fa8dadb294cbda271c3c0b645a5adabd208d998b Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Mon, 8 May 2017 18:24:16 -0700
Subject: [PATCH 297/305] separated ZSTD_estimateCStreamSize() from
 ZSTD_estimateCCtxSize()

for clarity
---
 doc/zstd_manual.html         | 16 +++++++++-------
 lib/compress/zstd_compress.c | 22 ++++++++++++++--------
 lib/zstd.h                   | 12 +++++++-----
 tests/paramgrill.c           |  4 ++--
 4 files changed, 32 insertions(+), 22 deletions(-)

diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html
index 9cfb67100..a5612c92c 100644
--- a/doc/zstd_manual.html
+++ b/doc/zstd_manual.html
@@ -1,10 +1,10 @@
 
 
 
-zstd 1.2.0 Manual
+zstd 1.3.0 Manual
 
 
-

zstd 1.2.0 Manual

+

zstd 1.3.0 Manual


Contents

    @@ -376,11 +376,8 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v

    Create a ZSTD compression context using external alloc and free functions


-
size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned streaming);
-

Provides amount of memory needed to allocate ZSTD_CCtx with a set of compression parameters. - Set streaming to 1 if the CCtx will be used for streaming (CStream). - Special case : when using ZSTD_initCStream_usingDict(), init will transparently create an internal CDict. - Use ZSTD_estimateCDictSize() and add this value to estimate total CCtx size +

size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams);
+

Provides amount of memory needed to allocate ZSTD_CCtx with a set of compression parameters.


size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx);
@@ -519,6 +516,11 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v
 

Advanced streaming functions


 
 

Advanced Streaming compression functions

ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem);
+/*! ZSTD_estimateCStreamSize() :
+ *  Provides amount of memory needed to allocate ZSTD_CStream with a set of compression parameters.
+ *  Special case : when using ZSTD_initCStream_usingDict(), init will transparently create an internal CDict.
+ *         Use ZSTD_estimateCDictSize() to estimate its size, and add for total CStream size */
+size_t ZSTD_estimateCStreamSize(ZSTD_compressionParameters cParams);
 size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);   /**< same as ZSTD_sizeof_CCtx */
 size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize);   /**< pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
 size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8. This result in the creation of an internal CDict */
diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
index c5687b21c..37590fca7 100644
--- a/lib/compress/zstd_compress.c
+++ b/lib/compress/zstd_compress.c
@@ -251,7 +251,7 @@ ZSTD_compressionParameters ZSTD_adjustCParams(ZSTD_compressionParameters cPar, u
 }
 
 
-size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned streaming)
+size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams)
 {
     size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << cParams.windowLog);
     U32    const divider = (cParams.searchLength==3) ? 3 : 4;
@@ -272,12 +272,7 @@ size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned stream
     size_t const optSpace = ((cParams.strategy == ZSTD_btopt) || (cParams.strategy == ZSTD_btopt2)) ? optBudget : 0;
     size_t const neededSpace = entropySpace + tableSpace + tokenSpace + optSpace;
 
-    size_t const inBuffSize = ((size_t)1 << cParams.windowLog) + blockSize;
-    size_t const outBuffSize = ZSTD_compressBound(blockSize) + 1;
-    size_t const streamingBudget = inBuffSize + outBuffSize;
-    size_t const streamingSize = streaming ? streamingBudget : 0;
-
-    return sizeof(ZSTD_CCtx) + neededSpace + streamingSize;
+    return sizeof(ZSTD_CCtx) + neededSpace;
 }
 
 
@@ -2989,7 +2984,7 @@ struct ZSTD_CDict_s {
 size_t ZSTD_estimateCDictSize(ZSTD_compressionParameters cParams, size_t dictSize)
 {
     cParams = ZSTD_adjustCParams(cParams, 0, dictSize);
-    return sizeof(ZSTD_CDict) + dictSize + ZSTD_estimateCCtxSize(cParams, 0);
+    return sizeof(ZSTD_CDict) + dictSize + ZSTD_estimateCCtxSize(cParams);
 }
 
 size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict)
@@ -3158,6 +3153,17 @@ size_t ZSTD_freeCStream(ZSTD_CStream* zcs)
     return ZSTD_freeCCtx(zcs);   /* same object */
 }
 
+size_t ZSTD_estimateCStreamSize(ZSTD_compressionParameters cParams)
+{
+    size_t const CCtxSize = ZSTD_estimateCCtxSize(cParams);
+    size_t const blockSize = MIN(ZSTD_BLOCKSIZE_ABSOLUTEMAX, (size_t)1 << cParams.windowLog);
+    size_t const inBuffSize = ((size_t)1 << cParams.windowLog) + blockSize;
+    size_t const outBuffSize = ZSTD_compressBound(blockSize) + 1;
+    size_t const streamingSize = inBuffSize + outBuffSize;
+
+    return sizeof(ZSTD_CCtx) + CCtxSize + streamingSize;
+}
+
 
 /*======   Initialization   ======*/
 
diff --git a/lib/zstd.h b/lib/zstd.h
index c3a8a52a3..86f1c4405 100644
--- a/lib/zstd.h
+++ b/lib/zstd.h
@@ -463,11 +463,8 @@ ZSTDLIB_API unsigned long long ZSTD_findDecompressedSize(const void* src, size_t
 ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx_advanced(ZSTD_customMem customMem);
 
 /*! ZSTD_estimateCCtxSize() :
- *  Provides amount of memory needed to allocate ZSTD_CCtx with a set of compression parameters.
- *  Set streaming to 1 if the CCtx will be used for streaming (CStream).
- *  Special case : when using ZSTD_initCStream_usingDict(), init will transparently create an internal CDict.
- *         Use ZSTD_estimateCDictSize() and add this value to estimate total CCtx size */
-ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams, unsigned streaming);
+ *  Provides amount of memory needed to allocate ZSTD_CCtx with a set of compression parameters. */
+ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams);
 
 /*! ZSTD_sizeofCCtx() :
  *  amount of used memory is variable, depending primarily on compression level */
@@ -609,6 +606,11 @@ ZSTDLIB_API unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize);
 
 /*=====   Advanced Streaming compression functions  =====*/
 ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem);
+/*! ZSTD_estimateCStreamSize() :
+ *  Provides amount of memory needed to allocate ZSTD_CStream with a set of compression parameters.
+ *  Special case : when using ZSTD_initCStream_usingDict(), init will transparently create an internal CDict.
+ *         Use ZSTD_estimateCDictSize() to estimate its size, and add for total CStream size */
+ZSTDLIB_API size_t ZSTD_estimateCStreamSize(ZSTD_compressionParameters cParams);
 ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);   /**< same as ZSTD_sizeof_CCtx */
 ZSTDLIB_API size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize);   /**< pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
 ZSTDLIB_API size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8. This result in the creation of an internal CDict */
diff --git a/tests/paramgrill.c b/tests/paramgrill.c
index 3cc916e01..1913b54d0 100644
--- a/tests/paramgrill.c
+++ b/tests/paramgrill.c
@@ -388,8 +388,8 @@ static int BMK_seed(winnerInfo_t* winners, const ZSTD_compressionParameters para
             double W_DMemUsed_note = W_ratioNote * ( 40 + 9*cLevel) - log((double)W_DMemUsed);
             double O_DMemUsed_note = O_ratioNote * ( 40 + 9*cLevel) - log((double)O_DMemUsed);
 
-            size_t W_CMemUsed = (1 << params.windowLog) + ZSTD_estimateCCtxSize(params, 0/*streaming*/);
-            size_t O_CMemUsed = (1 << winners[cLevel].params.windowLog) + ZSTD_estimateCCtxSize(winners[cLevel].params, 0);
+            size_t W_CMemUsed = (1 << params.windowLog) + ZSTD_estimateCCtxSize(params);
+            size_t O_CMemUsed = (1 << winners[cLevel].params.windowLog) + ZSTD_estimateCCtxSize(winners[cLevel].params);
             double W_CMemUsed_note = W_ratioNote * ( 50 + 13*cLevel) - log((double)W_CMemUsed);
             double O_CMemUsed_note = O_ratioNote * ( 50 + 13*cLevel) - log((double)O_CMemUsed);
 

From 5a36c069e73d7c2297e44201c5eb7d758435208e Mon Sep 17 00:00:00 2001
From: Yann Collet 
Date: Tue, 9 May 2017 15:11:30 -0700
Subject: [PATCH 298/305] regroup memory usage function declarations

in a single paragraph in zstd.h, for clarity
---
 doc/zstd_manual.html | 143 ++++++++++++++++++++---------------------
 lib/zstd.h           | 148 ++++++++++++++++++++-----------------------
 2 files changed, 136 insertions(+), 155 deletions(-)

diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html
index a5612c92c..c934da041 100644
--- a/doc/zstd_manual.html
+++ b/doc/zstd_manual.html
@@ -19,8 +19,8 @@
 
  • Streaming decompression - HowTo
  • START OF ADVANCED AND EXPERIMENTAL FUNCTIONS
  • Advanced types
  • -
  • Compressed size functions
  • -
  • Decompressed size functions
  • +
  • Frame size functions
  • +
  • Context memory usage
  • Advanced compression functions
  • Advanced decompression functions
  • Advanced streaming functions
  • @@ -325,49 +325,81 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB typedef void (*ZSTD_freeFunction) (void* opaque, void* address); typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; void* opaque; } ZSTD_customMem;

    -

    Compressed size functions

    
    +

    Frame size functions

    
     
     
    size_t ZSTD_findFrameCompressedSize(const void* src, size_t srcSize);
     

    `src` should point to the start of a ZSTD encoded frame or skippable frame `srcSize` must be at least as large as the frame - @return : the compressed size of the frame pointed to by `src`, suitable to pass to - `ZSTD_decompress` or similar, or an error code if given invalid input. + @return : the compressed size of the frame pointed to by `src`, + suitable to pass to `ZSTD_decompress` or similar, + or an error code if given invalid input.


    -

    Decompressed size functions

    
    -
    -
    unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize);
    -

    `src` should point to the start of a ZSTD encoded frame - `srcSize` must be at least as large as the frame header. A value greater than or equal - to `ZSTD_frameHeaderSize_max` is guaranteed to be large enough in all cases. - @return : decompressed size of the frame pointed to be `src` if known, otherwise - - ZSTD_CONTENTSIZE_UNKNOWN if the size cannot be determined - - ZSTD_CONTENTSIZE_ERROR if an error occurred (e.g. invalid magic number, srcSize too small) +

    #define ZSTD_CONTENTSIZE_UNKNOWN (0ULL - 1)
    +#define ZSTD_CONTENTSIZE_ERROR   (0ULL - 2)
    +unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize);
    +

    `src` should point to the start of a ZSTD encoded frame. + `srcSize` must be at least as large as the frame header. + A value >= `ZSTD_frameHeaderSize_max` is guaranteed to be large enough. + @return : - decompressed size of the frame pointed to be `src` if known + - ZSTD_CONTENTSIZE_UNKNOWN if the size cannot be determined + - ZSTD_CONTENTSIZE_ERROR if an error occurred (e.g. invalid magic number, srcSize too small)


    unsigned long long ZSTD_findDecompressedSize(const void* src, size_t srcSize);
    -

    `src` should point the start of a series of ZSTD encoded and/or skippable frames - `srcSize` must be the _exact_ size of this series +

    `src` should point the start of a series of ZSTD encoded and/or skippable frames + `srcSize` must be the _exact_ size of this series (i.e. there should be a frame boundary exactly `srcSize` bytes after `src`) - @return : the decompressed size of all data in the contained frames, as a 64-bit value _if known_ - - if the decompressed size cannot be determined: ZSTD_CONTENTSIZE_UNKNOWN - - if an error occurred: ZSTD_CONTENTSIZE_ERROR + @return : - decompressed size of all data in all successive frames + - if the decompressed size cannot be determined: ZSTD_CONTENTSIZE_UNKNOWN + - if an error occurred: ZSTD_CONTENTSIZE_ERROR - note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode. - When `return==ZSTD_CONTENTSIZE_UNKNOWN`, data to decompress could be any size. - In which case, it's necessary to use streaming mode to decompress data. - Optionally, application can still use ZSTD_decompress() while relying on implied limits. - (For example, data may be necessarily cut into blocks <= 16 KB). - note 2 : decompressed size is always present when compression is done with ZSTD_compress() - note 3 : decompressed size can be very large (64-bits value), - potentially larger than what local system can handle as a single memory segment. - In which case, it's necessary to use streaming mode to decompress data. - note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified. - Always ensure result fits within application's authorized limits. - Each application can set its own limits. - note 5 : ZSTD_findDecompressedSize handles multiple frames, and so it must traverse the input to - read each contained frame header. This is efficient as most of the data is skipped, - however it does mean that all frame data must be present and valid. + note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode. + When `return==ZSTD_CONTENTSIZE_UNKNOWN`, data to decompress could be any size. + In which case, it's necessary to use streaming mode to decompress data. + Optionally, application can still use ZSTD_decompress() while relying on implied limits. + (For example, data may be necessarily cut into blocks <= 16 KB). + note 2 : decompressed size is always present when compression is done with ZSTD_compress() + note 3 : decompressed size can be very large (64-bits value), + potentially larger than what local system can handle as a single memory segment. + In which case, it's necessary to use streaming mode to decompress data. + note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified. + Always ensure result fits within application's authorized limits. + Each application can set its own limits. + note 5 : ZSTD_findDecompressedSize handles multiple frames, and so it must traverse the input to + read each contained frame header. This is efficient as most of the data is skipped, + however it does mean that all frame data must be present and valid. +


    + +

    Context memory usage

    
    +
    +
    size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx);
    +size_t ZSTD_sizeof_DCtx(const ZSTD_DCtx* dctx);
    +size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);
    +size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds);
    +size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict);
    +size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict);
    +

    These functions give the current memory usage of selected object. + Object memory usage can evolve if it's re-used multiple times. +


    + +
    size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams);
    +size_t ZSTD_estimateDCtxSize(void);
    +

    These functions make it possible to estimate memory usage + of a future target object, before its allocation, + given a set of parameters, which vary depending on target object. + The objective is to guide decision before allocation. +


    + +
    size_t ZSTD_estimateCStreamSize(ZSTD_compressionParameters cParams);
    +

    Note : if streaming is init with function ZSTD_init?Stream_usingDict(), + an internal ?Dict will be created, which size is not estimated. + In this case, get additional size by using ZSTD_estimate?DictSize +


    + +
    size_t ZSTD_estimateCDictSize(ZSTD_compressionParameters cParams, size_t dictSize);
    +size_t ZSTD_estimateDDictSize(size_t dictSize);
    +

    Note : if dictionary is created "byReference", reduce estimation by dictSize


    Advanced compression functions

    
    @@ -376,14 +408,6 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v
     

    Create a ZSTD compression context using external alloc and free functions


    -
    size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams);
    -

    Provides amount of memory needed to allocate ZSTD_CCtx with a set of compression parameters. -


    - -
    size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx);
    -

    amount of used memory is variable, depending primarily on compression level -


    -
    typedef enum {
         ZSTD_p_forceWindow,   /* Force back-references to remain < windowSize, even when referencing Dictionary content (default:0) */
         ZSTD_p_forceRawDict   /* Force loading dictionary in "content-only" mode (no header analysis) */
    @@ -405,15 +429,6 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v
     

    Create a ZSTD_CDict using external alloc and free, and customized compression parameters


    -
    size_t ZSTD_estimateCDictSize(ZSTD_compressionParameters cParams, size_t dictSize);
    -

    Estimate amount of memory that will be needed to create a dictionary with following arguments - Note : if dictionary is created "byReference", reduce this amount by dictSize -


    - -
    size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict);
    -

    Gives the amount of memory used by a given ZSTD_sizeof_CDict -


    -
    ZSTD_compressionParameters ZSTD_getCParams(int compressionLevel, unsigned long long estimatedSrcSize, size_t dictSize);
     

    @return ZSTD_compressionParameters structure for a selected compression level and estimated srcSize. `estimatedSrcSize` value is optional, select 0 if not known @@ -457,18 +472,10 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v Note 3 : Skippable Frame Identifiers are considered valid.


    -
    size_t ZSTD_estimateDCtxSize(void);
    -

    Gives the potential amount of memory allocated to create a ZSTD_DCtx -


    -
    ZSTD_DCtx* ZSTD_createDCtx_advanced(ZSTD_customMem customMem);
     

    Create a ZSTD decompression context using external alloc and free functions


    -
    size_t ZSTD_sizeof_DCtx(const ZSTD_DCtx* dctx);
    -

    Gives the amount of memory used by a given ZSTD_DCtx -


    -
    ZSTD_DDict* ZSTD_createDDict_byReference(const void* dictBuffer, size_t dictSize);
     

    Create a digested dictionary, ready to start decompression operation without startup delay. Dictionary content is simply referenced, and therefore stays in dictBuffer. @@ -480,15 +487,6 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v

    Create a ZSTD_DDict using external alloc and free, optionally by reference


    -
    size_t ZSTD_estimateDDictSize(size_t dictSize);
    -

    Estimate amount of memory that will be needed to create a dictionary for decompression. - Note : if dictionary is created "byReference", reduce this amount by dictSize -


    - -
    size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict);
    -

    Gives the amount of memory used by a given ZSTD_DDict -


    -
    unsigned ZSTD_getDictID_fromDict(const void* dict, size_t dictSize);
     

    Provides the dictID stored within dictionary. if @return == 0, the dictionary is not conformant with Zstandard specification. @@ -516,12 +514,6 @@ typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; v

    Advanced streaming functions

    
     
     

    Advanced Streaming compression functions

    ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem);
    -/*! ZSTD_estimateCStreamSize() :
    - *  Provides amount of memory needed to allocate ZSTD_CStream with a set of compression parameters.
    - *  Special case : when using ZSTD_initCStream_usingDict(), init will transparently create an internal CDict.
    - *         Use ZSTD_estimateCDictSize() to estimate its size, and add for total CStream size */
    -size_t ZSTD_estimateCStreamSize(ZSTD_compressionParameters cParams);
    -size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);   /**< same as ZSTD_sizeof_CCtx */
     size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize);   /**< pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
     size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8. This result in the creation of an internal CDict */
     size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize,
    @@ -540,11 +532,10 @@ size_t ZSTD_initCStream_usingCDict_advanced(ZSTD_CStream* zcs, const ZSTD_CDict*
     
     

    Advanced Streaming decompression functions

    typedef enum { DStream_p_maxWindowSize } ZSTD_DStreamParameter_e;
     ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem);
    -size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
     size_t ZSTD_setDStreamParameter(ZSTD_DStream* zds, ZSTD_DStreamParameter_e paramType, unsigned paramValue);
    +size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
     size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, const ZSTD_DDict* ddict);  /**< note : ddict will just be referenced, and must outlive decompression session */
     size_t ZSTD_resetDStream(ZSTD_DStream* zds);  /**< re-use decompression parameters from previous init; saves dictionary loading */
    -size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds);
     

    Buffer-less and synchronous inner streaming functions

       This is an advanced API, giving full control over buffer management, for users which need direct control over memory.
    diff --git a/lib/zstd.h b/lib/zstd.h
    index 86f1c4405..690c851f1 100644
    --- a/lib/zstd.h
    +++ b/lib/zstd.h
    @@ -350,9 +350,6 @@ ZSTDLIB_API size_t ZSTD_DStreamOutSize(void);   /*!< recommended size for output
     #define ZSTD_MAGICNUMBER            0xFD2FB528   /* >= v0.8.0 */
     #define ZSTD_MAGIC_SKIPPABLE_START  0x184D2A50U
     
    -#define ZSTD_CONTENTSIZE_UNKNOWN (0ULL - 1)
    -#define ZSTD_CONTENTSIZE_ERROR   (0ULL - 2)
    -
     #define ZSTD_WINDOWLOG_MAX_32  27
     #define ZSTD_WINDOWLOG_MAX_64  27
     #define ZSTD_WINDOWLOG_MAX    ((unsigned)(sizeof(size_t) == 4 ? ZSTD_WINDOWLOG_MAX_32 : ZSTD_WINDOWLOG_MAX_64))
    @@ -407,54 +404,88 @@ typedef void  (*ZSTD_freeFunction) (void* opaque, void* address);
     typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; void* opaque; } ZSTD_customMem;
     
     /***************************************
    -*  Compressed size functions
    +*  Frame size functions
     ***************************************/
     
     /*! ZSTD_findFrameCompressedSize() :
      *  `src` should point to the start of a ZSTD encoded frame or skippable frame
      *  `srcSize` must be at least as large as the frame
    - *  @return : the compressed size of the frame pointed to by `src`, suitable to pass to
    - *      `ZSTD_decompress` or similar, or an error code if given invalid input. */
    + *  @return : the compressed size of the frame pointed to by `src`,
    + *            suitable to pass to `ZSTD_decompress` or similar,
    + *            or an error code if given invalid input. */
     ZSTDLIB_API size_t ZSTD_findFrameCompressedSize(const void* src, size_t srcSize);
     
    -/***************************************
    -*  Decompressed size functions
    -***************************************/
     /*! ZSTD_getFrameContentSize() :
    -*   `src` should point to the start of a ZSTD encoded frame
    -*   `srcSize` must be at least as large as the frame header.  A value greater than or equal
    -*       to `ZSTD_frameHeaderSize_max` is guaranteed to be large enough in all cases.
    -*   @return : decompressed size of the frame pointed to be `src` if known, otherwise
    -*             - ZSTD_CONTENTSIZE_UNKNOWN if the size cannot be determined
    -*             - ZSTD_CONTENTSIZE_ERROR if an error occurred (e.g. invalid magic number, srcSize too small) */
    + *  `src` should point to the start of a ZSTD encoded frame.
    + *  `srcSize` must be at least as large as the frame header.
    + *       A value >= `ZSTD_frameHeaderSize_max` is guaranteed to be large enough.
    + *  @return : - decompressed size of the frame pointed to be `src` if known
    + *            - ZSTD_CONTENTSIZE_UNKNOWN if the size cannot be determined
    + *            - ZSTD_CONTENTSIZE_ERROR if an error occurred (e.g. invalid magic number, srcSize too small) */
    +#define ZSTD_CONTENTSIZE_UNKNOWN (0ULL - 1)
    +#define ZSTD_CONTENTSIZE_ERROR   (0ULL - 2)
     ZSTDLIB_API unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize);
     
     /*! ZSTD_findDecompressedSize() :
    -*   `src` should point the start of a series of ZSTD encoded and/or skippable frames
    -*   `srcSize` must be the _exact_ size of this series
    -*       (i.e. there should be a frame boundary exactly `srcSize` bytes after `src`)
    -*   @return : the decompressed size of all data in the contained frames, as a 64-bit value _if known_
    -*             - if the decompressed size cannot be determined: ZSTD_CONTENTSIZE_UNKNOWN
    -*             - if an error occurred: ZSTD_CONTENTSIZE_ERROR
    -*
    -*    note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode.
    -*             When `return==ZSTD_CONTENTSIZE_UNKNOWN`, data to decompress could be any size.
    -*             In which case, it's necessary to use streaming mode to decompress data.
    -*             Optionally, application can still use ZSTD_decompress() while relying on implied limits.
    -*             (For example, data may be necessarily cut into blocks <= 16 KB).
    -*    note 2 : decompressed size is always present when compression is done with ZSTD_compress()
    -*    note 3 : decompressed size can be very large (64-bits value),
    -*             potentially larger than what local system can handle as a single memory segment.
    -*             In which case, it's necessary to use streaming mode to decompress data.
    -*    note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified.
    -*             Always ensure result fits within application's authorized limits.
    -*             Each application can set its own limits.
    -*    note 5 : ZSTD_findDecompressedSize handles multiple frames, and so it must traverse the input to
    -*             read each contained frame header.  This is efficient as most of the data is skipped,
    -*             however it does mean that all frame data must be present and valid. */
    + *  `src` should point the start of a series of ZSTD encoded and/or skippable frames
    + *  `srcSize` must be the _exact_ size of this series
    + *       (i.e. there should be a frame boundary exactly `srcSize` bytes after `src`)
    + *  @return : - decompressed size of all data in all successive frames
    + *            - if the decompressed size cannot be determined: ZSTD_CONTENTSIZE_UNKNOWN
    + *            - if an error occurred: ZSTD_CONTENTSIZE_ERROR
    + *
    + *   note 1 : decompressed size is an optional field, that may not be present, especially in streaming mode.
    + *            When `return==ZSTD_CONTENTSIZE_UNKNOWN`, data to decompress could be any size.
    + *            In which case, it's necessary to use streaming mode to decompress data.
    + *            Optionally, application can still use ZSTD_decompress() while relying on implied limits.
    + *            (For example, data may be necessarily cut into blocks <= 16 KB).
    + *   note 2 : decompressed size is always present when compression is done with ZSTD_compress()
    + *   note 3 : decompressed size can be very large (64-bits value),
    + *            potentially larger than what local system can handle as a single memory segment.
    + *            In which case, it's necessary to use streaming mode to decompress data.
    + *   note 4 : If source is untrusted, decompressed size could be wrong or intentionally modified.
    + *            Always ensure result fits within application's authorized limits.
    + *            Each application can set its own limits.
    + *   note 5 : ZSTD_findDecompressedSize handles multiple frames, and so it must traverse the input to
    + *            read each contained frame header.  This is efficient as most of the data is skipped,
    + *            however it does mean that all frame data must be present and valid. */
     ZSTDLIB_API unsigned long long ZSTD_findDecompressedSize(const void* src, size_t srcSize);
     
     
    +/***************************************
    +*  Context memory usage
    +***************************************/
    +
    +/*! ZSTD_sizeof_*() :
    + *  These functions give the current memory usage of selected object.
    + *  Object memory usage can evolve if it's re-used multiple times. */
    +ZSTDLIB_API size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx);
    +ZSTDLIB_API size_t ZSTD_sizeof_DCtx(const ZSTD_DCtx* dctx);
    +ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);
    +ZSTDLIB_API size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds);
    +ZSTDLIB_API size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict);
    +ZSTDLIB_API size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict);
    +
    +/*! ZSTD_estimate*() :
    + *  These functions make it possible to estimate memory usage
    + *  of a future target object, before its allocation,
    + *  given a set of parameters, which vary depending on target object.
    + *  The objective is to guide decision before allocation. */
    +ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams);
    +ZSTDLIB_API size_t ZSTD_estimateDCtxSize(void);
    +
    +/*! ZSTD_estimate?StreamSize() :
    + *  Note : if streaming is init with function ZSTD_init?Stream_usingDict(),
    + *         an internal ?Dict will be created, which size is not estimated.
    + *         In this case, get additional size by using ZSTD_estimate?DictSize */
    +ZSTDLIB_API size_t ZSTD_estimateCStreamSize(ZSTD_compressionParameters cParams);
    +
    +/*! ZSTD_estimate?DictSize() :
    + *  Note : if dictionary is created "byReference", reduce estimation by dictSize */
    +ZSTDLIB_API size_t ZSTD_estimateCDictSize(ZSTD_compressionParameters cParams, size_t dictSize);
    +ZSTDLIB_API size_t ZSTD_estimateDDictSize(size_t dictSize);
    +
    +
     /***************************************
     *  Advanced compression functions
     ***************************************/
    @@ -462,14 +493,6 @@ ZSTDLIB_API unsigned long long ZSTD_findDecompressedSize(const void* src, size_t
      *  Create a ZSTD compression context using external alloc and free functions */
     ZSTDLIB_API ZSTD_CCtx* ZSTD_createCCtx_advanced(ZSTD_customMem customMem);
     
    -/*! ZSTD_estimateCCtxSize() :
    - *  Provides amount of memory needed to allocate ZSTD_CCtx with a set of compression parameters. */
    -ZSTDLIB_API size_t ZSTD_estimateCCtxSize(ZSTD_compressionParameters cParams);
    -
    -/*! ZSTD_sizeofCCtx() :
    - *  amount of used memory is variable, depending primarily on compression level */
    -ZSTDLIB_API size_t ZSTD_sizeof_CCtx(const ZSTD_CCtx* cctx);
    -
     typedef enum {
         ZSTD_p_forceWindow,   /* Force back-references to remain < windowSize, even when referencing Dictionary content (default:0) */
         ZSTD_p_forceRawDict   /* Force loading dictionary in "content-only" mode (no header analysis) */
    @@ -490,15 +513,6 @@ ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_byReference(const void* dictBuffer, siz
     ZSTDLIB_API ZSTD_CDict* ZSTD_createCDict_advanced(const void* dict, size_t dictSize, unsigned byReference,
                                                       ZSTD_compressionParameters cParams, ZSTD_customMem customMem);
     
    -/*! ZSTD_estimateCDictSize() :
    - *  Estimate amount of memory that will be needed to create a dictionary with following arguments
    - *  Note : if dictionary is created "byReference", reduce this amount by dictSize */
    -ZSTDLIB_API size_t ZSTD_estimateCDictSize(ZSTD_compressionParameters cParams, size_t dictSize);
    -
    -/*! ZSTD_sizeof_CDict() :
    - *  Gives the amount of memory used by a given ZSTD_sizeof_CDict */
    -ZSTDLIB_API size_t ZSTD_sizeof_CDict(const ZSTD_CDict* cdict);
    -
     /*! ZSTD_getCParams() :
     *   @return ZSTD_compressionParameters structure for a selected compression level and estimated srcSize.
     *   `estimatedSrcSize` value is optional, select 0 if not known */
    @@ -543,18 +557,10 @@ ZSTDLIB_API size_t ZSTD_compress_usingCDict_advanced(ZSTD_CCtx* cctx,
      *  Note 3 : Skippable Frame Identifiers are considered valid. */
     ZSTDLIB_API unsigned ZSTD_isFrame(const void* buffer, size_t size);
     
    -/*! ZSTD_estimateDCtxSize() :
    - *  Gives the potential amount of memory allocated to create a ZSTD_DCtx */
    -ZSTDLIB_API size_t ZSTD_estimateDCtxSize(void);
    -
     /*! ZSTD_createDCtx_advanced() :
      *  Create a ZSTD decompression context using external alloc and free functions */
     ZSTDLIB_API ZSTD_DCtx* ZSTD_createDCtx_advanced(ZSTD_customMem customMem);
     
    -/*! ZSTD_sizeof_DCtx() :
    - *  Gives the amount of memory used by a given ZSTD_DCtx */
    -ZSTDLIB_API size_t ZSTD_sizeof_DCtx(const ZSTD_DCtx* dctx);
    -
     /*! ZSTD_createDDict_byReference() :
      *  Create a digested dictionary, ready to start decompression operation without startup delay.
      *  Dictionary content is simply referenced, and therefore stays in dictBuffer.
    @@ -566,15 +572,6 @@ ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_byReference(const void* dictBuffer, siz
     ZSTDLIB_API ZSTD_DDict* ZSTD_createDDict_advanced(const void* dict, size_t dictSize,
                                                       unsigned byReference, ZSTD_customMem customMem);
     
    -/*! ZSTD_estimateDDictSize() :
    - *  Estimate amount of memory that will be needed to create a dictionary for decompression.
    - *  Note : if dictionary is created "byReference", reduce this amount by dictSize */
    -ZSTDLIB_API size_t ZSTD_estimateDDictSize(size_t dictSize);
    -
    -/*! ZSTD_sizeof_DDict() :
    - *  Gives the amount of memory used by a given ZSTD_DDict */
    -ZSTDLIB_API size_t ZSTD_sizeof_DDict(const ZSTD_DDict* ddict);
    -
     /*! ZSTD_getDictID_fromDict() :
      *  Provides the dictID stored within dictionary.
      *  if @return == 0, the dictionary is not conformant with Zstandard specification.
    @@ -606,12 +603,6 @@ ZSTDLIB_API unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize);
     
     /*=====   Advanced Streaming compression functions  =====*/
     ZSTDLIB_API ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem);
    -/*! ZSTD_estimateCStreamSize() :
    - *  Provides amount of memory needed to allocate ZSTD_CStream with a set of compression parameters.
    - *  Special case : when using ZSTD_initCStream_usingDict(), init will transparently create an internal CDict.
    - *         Use ZSTD_estimateCDictSize() to estimate its size, and add for total CStream size */
    -ZSTDLIB_API size_t ZSTD_estimateCStreamSize(ZSTD_compressionParameters cParams);
    -ZSTDLIB_API size_t ZSTD_sizeof_CStream(const ZSTD_CStream* zcs);   /**< same as ZSTD_sizeof_CCtx */
     ZSTDLIB_API size_t ZSTD_initCStream_srcSize(ZSTD_CStream* zcs, int compressionLevel, unsigned long long pledgedSrcSize);   /**< pledgedSrcSize must be correct, a size of 0 means unknown.  for a frame size of 0 use initCStream_advanced */
     ZSTDLIB_API size_t ZSTD_initCStream_usingDict(ZSTD_CStream* zcs, const void* dict, size_t dictSize, int compressionLevel); /**< note: a dict will not be used if dict == NULL or dictSize < 8. This result in the creation of an internal CDict */
     ZSTDLIB_API size_t ZSTD_initCStream_advanced(ZSTD_CStream* zcs, const void* dict, size_t dictSize,
    @@ -632,11 +623,10 @@ ZSTDLIB_API size_t ZSTD_resetCStream(ZSTD_CStream* zcs, unsigned long long pledg
     /*=====   Advanced Streaming decompression functions  =====*/
     typedef enum { DStream_p_maxWindowSize } ZSTD_DStreamParameter_e;
     ZSTDLIB_API ZSTD_DStream* ZSTD_createDStream_advanced(ZSTD_customMem customMem);
    -ZSTDLIB_API size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
     ZSTDLIB_API size_t ZSTD_setDStreamParameter(ZSTD_DStream* zds, ZSTD_DStreamParameter_e paramType, unsigned paramValue);
    +ZSTDLIB_API size_t ZSTD_initDStream_usingDict(ZSTD_DStream* zds, const void* dict, size_t dictSize); /**< note: a dict will not be used if dict == NULL or dictSize < 8 */
     ZSTDLIB_API size_t ZSTD_initDStream_usingDDict(ZSTD_DStream* zds, const ZSTD_DDict* ddict);  /**< note : ddict will just be referenced, and must outlive decompression session */
     ZSTDLIB_API size_t ZSTD_resetDStream(ZSTD_DStream* zds);  /**< re-use decompression parameters from previous init; saves dictionary loading */
    -ZSTDLIB_API size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds);
     
     
     /*********************************************************************
    
    From 542c9dfcf8002fc6b5fe338abbb3316770833e40 Mon Sep 17 00:00:00 2001
    From: Yann Collet 
    Date: Tue, 9 May 2017 15:46:07 -0700
    Subject: [PATCH 299/305] changed name frameParams into frameHeader
    
    ZSTD_frameParams => ZSTD_frameHeader
    ZSTD_getFrameParams() -> ZSTD_getFrameHeader()
    
    The new naming is more distinctive from ZSTD_frameParameters,
    which is used during compression.
    
    ZSTD_frameHeader is clearer in its intention to described frame header content.
    It also implies we are decoding a ZSTD frame, hence we are at decoding stage.
    ---
     doc/zstd_manual.html             |  4 ++--
     lib/decompress/zstd_decompress.c | 27 +++++++++++++--------------
     lib/zstd.h                       |  4 ++--
     tests/fullbench.c                |  8 ++++----
     tests/fuzzer.c                   |  8 ++++----
     5 files changed, 25 insertions(+), 26 deletions(-)
    
    diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html
    index c934da041..d24a6b892 100644
    --- a/doc/zstd_manual.html
    +++ b/doc/zstd_manual.html
    @@ -643,9 +643,9 @@ size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned lo
         unsigned windowSize;
         unsigned dictID;
         unsigned checksumFlag;
    -} ZSTD_frameParams;
    +} ZSTD_frameHeader;
     

    -

    Buffer-less streaming decompression functions

    size_t ZSTD_getFrameParams(ZSTD_frameParams* fparamsPtr, const void* src, size_t srcSize);   /**< doesn't consume input, see details below */
    +

    Buffer-less streaming decompression functions

    size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fparamsPtr, const void* src, size_t srcSize);   /**< doesn't consume input, see details below */
     size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx);
     size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx* dctx, const void* dict, size_t dictSize);
     void   ZSTD_copyDCtx(ZSTD_DCtx* dctx, const ZSTD_DCtx* preparedDCtx);
    diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c
    index 88488b4b0..3c231a941 100644
    --- a/lib/decompress/zstd_decompress.c
    +++ b/lib/decompress/zstd_decompress.c
    @@ -105,7 +105,7 @@ struct ZSTD_DCtx_s
         const void* vBase;            /* virtual start of previous segment if it was just before current one */
         const void* dictEnd;          /* end of previous segment */
         size_t expected;
    -    ZSTD_frameParams fParams;
    +    ZSTD_frameHeader fParams;
         blockType_e bType;   /* used in ZSTD_decompressContinue(), to transfer blockType between header decoding and block decoding stages */
         ZSTD_dStage stage;
         U32 litEntropy;
    @@ -219,12 +219,12 @@ static size_t ZSTD_frameHeaderSize(const void* src, size_t srcSize)
     }
     
     
    -/** ZSTD_getFrameParams() :
    +/** ZSTD_getFrameHeader() :
     *   decode Frame Header, or require larger `srcSize`.
     *   @return : 0, `fparamsPtr` is correctly filled,
     *            >0, `srcSize` is too small, result is expected `srcSize`,
     *             or an error code, which can be tested using ZSTD_isError() */
    -size_t ZSTD_getFrameParams(ZSTD_frameParams* fparamsPtr, const void* src, size_t srcSize)
    +size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fparamsPtr, const void* src, size_t srcSize)
     {
         const BYTE* ip = (const BYTE*)src;
     
    @@ -302,9 +302,8 @@ unsigned long long ZSTD_getFrameContentSize(const void *src, size_t srcSize)
             return ret == 0 ? ZSTD_CONTENTSIZE_UNKNOWN : ret;
         }
     #endif
    -    {
    -        ZSTD_frameParams fParams;
    -        if (ZSTD_getFrameParams(&fParams, src, srcSize) != 0) return ZSTD_CONTENTSIZE_ERROR;
    +    {   ZSTD_frameHeader fParams;
    +        if (ZSTD_getFrameHeader(&fParams, src, srcSize) != 0) return ZSTD_CONTENTSIZE_ERROR;
             if (fParams.windowSize == 0) {
                 /* Either skippable or empty frame, size == 0 either way */
                 return 0;
    @@ -389,7 +388,7 @@ unsigned long long ZSTD_getDecompressedSize(const void* src, size_t srcSize)
     *   @return : 0 if success, or an error code, which can be tested using ZSTD_isError() */
     static size_t ZSTD_decodeFrameHeader(ZSTD_DCtx* dctx, const void* src, size_t headerSize)
     {
    -    size_t const result = ZSTD_getFrameParams(&(dctx->fParams), src, headerSize);
    +    size_t const result = ZSTD_getFrameHeader(&(dctx->fParams), src, headerSize);
         if (ZSTD_isError(result)) return result;  /* invalid header */
         if (result>0) return ERROR(srcSize_wrong);   /* headerSize too small */
         if (dctx->fParams.dictID && (dctx->dictID != dctx->fParams.dictID)) return ERROR(dictionary_wrong);
    @@ -1364,13 +1363,13 @@ size_t ZSTD_findFrameCompressedSize(const void *src, size_t srcSize)
             const BYTE* ip = (const BYTE*)src;
             const BYTE* const ipstart = ip;
             size_t remainingSize = srcSize;
    -        ZSTD_frameParams fParams;
    +        ZSTD_frameHeader fParams;
     
             size_t const headerSize = ZSTD_frameHeaderSize(ip, remainingSize);
             if (ZSTD_isError(headerSize)) return headerSize;
     
             /* Frame Header */
    -        {   size_t const ret = ZSTD_getFrameParams(&fParams, ip, remainingSize);
    +        {   size_t const ret = ZSTD_getFrameHeader(&fParams, ip, remainingSize);
                 if (ZSTD_isError(ret)) return ret;
                 if (ret > 0) return ERROR(srcSize_wrong);
             }
    @@ -2016,11 +2015,11 @@ unsigned ZSTD_getDictID_fromDDict(const ZSTD_DDict* ddict)
      *    Note : possible if `srcSize < ZSTD_FRAMEHEADERSIZE_MAX`.
      *  - This is not a Zstandard frame.
      *  When identifying the exact failure cause, it's possible to use
    - *  ZSTD_getFrameParams(), which will provide a more precise error code. */
    + *  ZSTD_getFrameHeader(), which will provide a more precise error code. */
     unsigned ZSTD_getDictID_fromFrame(const void* src, size_t srcSize)
     {
    -    ZSTD_frameParams zfp = { 0 , 0 , 0 , 0 };
    -    size_t const hError = ZSTD_getFrameParams(&zfp, src, srcSize);
    +    ZSTD_frameHeader zfp = { 0 , 0 , 0 , 0 };
    +    size_t const hError = ZSTD_getFrameHeader(&zfp, src, srcSize);
         if (ZSTD_isError(hError)) return 0;
         return zfp.dictID;
     }
    @@ -2053,7 +2052,7 @@ struct ZSTD_DStream_s {
         ZSTD_DCtx* dctx;
         ZSTD_DDict* ddictLocal;
         const ZSTD_DDict* ddict;
    -    ZSTD_frameParams fParams;
    +    ZSTD_frameHeader fParams;
         ZSTD_dStreamStage stage;
         char*  inBuff;
         size_t inBuffSize;
    @@ -2217,7 +2216,7 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
                 /* fall-through */
     
             case zdss_loadHeader :
    -            {   size_t const hSize = ZSTD_getFrameParams(&zds->fParams, zds->headerBuffer, zds->lhSize);
    +            {   size_t const hSize = ZSTD_getFrameHeader(&zds->fParams, zds->headerBuffer, zds->lhSize);
                     if (ZSTD_isError(hSize))
     #if defined(ZSTD_LEGACY_SUPPORT) && (ZSTD_LEGACY_SUPPORT>=1)
                     {   U32 const legacyVersion = ZSTD_isLegacy(istart, iend-istart);
    diff --git a/lib/zstd.h b/lib/zstd.h
    index 690c851f1..ddcaaf8bd 100644
    --- a/lib/zstd.h
    +++ b/lib/zstd.h
    @@ -746,10 +746,10 @@ typedef struct {
         unsigned windowSize;
         unsigned dictID;
         unsigned checksumFlag;
    -} ZSTD_frameParams;
    +} ZSTD_frameHeader;
     
     /*=====   Buffer-less streaming decompression functions  =====*/
    -ZSTDLIB_API size_t ZSTD_getFrameParams(ZSTD_frameParams* fparamsPtr, const void* src, size_t srcSize);   /**< doesn't consume input, see details below */
    +ZSTDLIB_API size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fparamsPtr, const void* src, size_t srcSize);   /**< doesn't consume input, see details below */
     ZSTDLIB_API size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx);
     ZSTDLIB_API size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx* dctx, const void* dict, size_t dictSize);
     ZSTDLIB_API void   ZSTD_copyDCtx(ZSTD_DCtx* dctx, const ZSTD_DCtx* preparedDCtx);
    diff --git a/tests/fullbench.c b/tests/fullbench.c
    index 38cf22fa7..13323aec1 100644
    --- a/tests/fullbench.c
    +++ b/tests/fullbench.c
    @@ -297,10 +297,10 @@ static size_t benchMem(const void* src, size_t srcSize, U32 benchNb)
         case 31:  /* ZSTD_decodeLiteralsBlock */
             if (g_zdc==NULL) g_zdc = ZSTD_createDCtx();
             {   blockProperties_t bp;
    -            ZSTD_frameParams zfp;
    +            ZSTD_frameHeader zfp;
                 size_t frameHeaderSize, skippedSize;
                 g_cSize = ZSTD_compress(dstBuff, dstBuffSize, src, srcSize, 1);
    -            frameHeaderSize = ZSTD_getFrameParams(&zfp, dstBuff, ZSTD_frameHeaderSize_min);
    +            frameHeaderSize = ZSTD_getFrameHeader(&zfp, dstBuff, ZSTD_frameHeaderSize_min);
                 if (frameHeaderSize==0) frameHeaderSize = ZSTD_frameHeaderSize_min;
                 ZSTD_getcBlockSize(dstBuff+frameHeaderSize, dstBuffSize, &bp);  /* Get 1st block type */
                 if (bp.blockType != bt_compressed) {
    @@ -315,13 +315,13 @@ static size_t benchMem(const void* src, size_t srcSize, U32 benchNb)
         case 32:   /* ZSTD_decodeSeqHeaders */
             if (g_zdc==NULL) g_zdc = ZSTD_createDCtx();
             {   blockProperties_t bp;
    -            ZSTD_frameParams zfp;
    +            ZSTD_frameHeader zfp;
                 const BYTE* ip = dstBuff;
                 const BYTE* iend;
                 size_t frameHeaderSize, cBlockSize;
                 ZSTD_compress(dstBuff, dstBuffSize, src, srcSize, 1);   /* it would be better to use direct block compression here */
                 g_cSize = ZSTD_compress(dstBuff, dstBuffSize, src, srcSize, 1);
    -            frameHeaderSize = ZSTD_getFrameParams(&zfp, dstBuff, ZSTD_frameHeaderSize_min);
    +            frameHeaderSize = ZSTD_getFrameHeader(&zfp, dstBuff, ZSTD_frameHeaderSize_min);
                 if (frameHeaderSize==0) frameHeaderSize = ZSTD_frameHeaderSize_min;
                 ip += frameHeaderSize;   /* Skip frame Header */
                 cBlockSize = ZSTD_getcBlockSize(ip, dstBuffSize, &bp);   /* Get 1st block type */
    diff --git a/tests/fuzzer.c b/tests/fuzzer.c
    index 7ffcff26e..4b9cd97a5 100644
    --- a/tests/fuzzer.c
    +++ b/tests/fuzzer.c
    @@ -339,8 +339,8 @@ static int basicUnitTests(U32 seed, double compressibility)
                 CHECKPLUS(r, ZSTD_compressEnd(ctxDuplicated, compressedBuffer, ZSTD_compressBound(testSize),
                                               (const char*)CNBuffer + dictSize, testSize),
                           cSize = r);
    -            {   ZSTD_frameParams fp;
    -                if (ZSTD_getFrameParams(&fp, compressedBuffer, cSize)) goto _output_error;
    +            {   ZSTD_frameHeader fp;
    +                if (ZSTD_getFrameHeader(&fp, compressedBuffer, cSize)) goto _output_error;
                     if ((fp.frameContentSize != testSize) && (fp.frameContentSize != 0)) goto _output_error;
             }   }
             DISPLAYLEVEL(4, "OK \n");
    @@ -862,8 +862,8 @@ static int fuzzerTests(U32 seed, U32 nbTests, unsigned startTest, U32 const maxD
             }
     
             /* frame header decompression test */
    -        {   ZSTD_frameParams dParams;
    -            size_t const check = ZSTD_getFrameParams(&dParams, cBuffer, cSize);
    +        {   ZSTD_frameHeader dParams;
    +            size_t const check = ZSTD_getFrameHeader(&dParams, cBuffer, cSize);
                 CHECK(ZSTD_isError(check), "Frame Parameters extraction failed");
                 CHECK(dParams.frameContentSize != sampleSize, "Frame content size incorrect");
             }
    
    From f16f4497ca121f8a0ec82e2af35a4a00d82acf13 Mon Sep 17 00:00:00 2001
    From: Yann Collet 
    Date: Tue, 9 May 2017 16:18:17 -0700
    Subject: [PATCH 300/305] added ZSTD_estimateDStreamSize()
    
    ---
     doc/zstd_manual.html             | 15 ++++++++-------
     lib/decompress/zstd_decompress.c | 30 ++++++++++++++++++++----------
     lib/zstd.h                       | 15 ++++++++-------
     tests/zstreamtest.c              | 15 ++++++++++++++-
     4 files changed, 50 insertions(+), 25 deletions(-)
    
    diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html
    index d24a6b892..9cfb1b68b 100644
    --- a/doc/zstd_manual.html
    +++ b/doc/zstd_manual.html
    @@ -321,6 +321,13 @@ size_t ZSTD_decompressStream(ZSTD_DStream* zds, ZSTD_outBuffer* output, ZSTD_inB
         ZSTD_frameParameters fParams;
     } ZSTD_parameters;
     

    +
    typedef struct {
    +    unsigned long long frameContentSize;
    +    unsigned windowSize;
    +    unsigned dictID;
    +    unsigned checksumFlag;
    +} ZSTD_frameHeader;
    +

    Custom memory allocation functions

    typedef void* (*ZSTD_allocFunction) (void* opaque, size_t size);
     typedef void  (*ZSTD_freeFunction) (void* opaque, void* address);
     typedef struct { ZSTD_allocFunction customAlloc; ZSTD_freeFunction customFree; void* opaque; } ZSTD_customMem;
    @@ -392,6 +399,7 @@ size_t ZSTD_estimateDCtxSize(void);
     


    size_t ZSTD_estimateCStreamSize(ZSTD_compressionParameters cParams);
    +size_t ZSTD_estimateDStreamSize(ZSTD_frameHeader fHeader);
     

    Note : if streaming is init with function ZSTD_init?Stream_usingDict(), an internal ?Dict will be created, which size is not estimated. In this case, get additional size by using ZSTD_estimate?DictSize @@ -638,13 +646,6 @@ size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned lo It also returns Frame Size as fparamsPtr->frameContentSize.

    -
    typedef struct {
    -    unsigned long long frameContentSize;
    -    unsigned windowSize;
    -    unsigned dictID;
    -    unsigned checksumFlag;
    -} ZSTD_frameHeader;
    -

    Buffer-less streaming decompression functions

    size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fparamsPtr, const void* src, size_t srcSize);   /**< doesn't consume input, see details below */
     size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx);
     size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx* dctx, const void* dict, size_t dictSize);
    diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c
    index 3c231a941..267aa94d0 100644
    --- a/lib/decompress/zstd_decompress.c
    +++ b/lib/decompress/zstd_decompress.c
    @@ -221,20 +221,21 @@ static size_t ZSTD_frameHeaderSize(const void* src, size_t srcSize)
     
     /** ZSTD_getFrameHeader() :
     *   decode Frame Header, or require larger `srcSize`.
    -*   @return : 0, `fparamsPtr` is correctly filled,
    +*   @return : 0, `fhiPtr` is correctly filled,
     *            >0, `srcSize` is too small, result is expected `srcSize`,
     *             or an error code, which can be tested using ZSTD_isError() */
    -size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fparamsPtr, const void* src, size_t srcSize)
    +size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fhiPtr, const void* src, size_t srcSize)
     {
         const BYTE* ip = (const BYTE*)src;
    -
         if (srcSize < ZSTD_frameHeaderSize_prefix) return ZSTD_frameHeaderSize_prefix;
    +
         if (MEM_readLE32(src) != ZSTD_MAGICNUMBER) {
             if ((MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) {
    +            /* skippable frame */
                 if (srcSize < ZSTD_skippableHeaderSize) return ZSTD_skippableHeaderSize; /* magic number + skippable frame length */
    -            memset(fparamsPtr, 0, sizeof(*fparamsPtr));
    -            fparamsPtr->frameContentSize = MEM_readLE32((const char *)src + 4);
    -            fparamsPtr->windowSize = 0; /* windowSize==0 means a frame is skippable */
    +            memset(fhiPtr, 0, sizeof(*fhiPtr));
    +            fhiPtr->frameContentSize = MEM_readLE32((const char *)src + 4);
    +            fhiPtr->windowSize = 0; /* windowSize==0 means a frame is skippable */
                 return 0;
             }
             return ERROR(prefix_unknown);
    @@ -281,10 +282,10 @@ size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fparamsPtr, const void* src, size_t
             }
             if (!windowSize) windowSize = (U32)frameContentSize;
             if (windowSize > windowSizeMax) return ERROR(frameParameter_windowTooLarge);
    -        fparamsPtr->frameContentSize = frameContentSize;
    -        fparamsPtr->windowSize = windowSize;
    -        fparamsPtr->dictID = dictID;
    -        fparamsPtr->checksumFlag = checksumFlag;
    +        fhiPtr->frameContentSize = frameContentSize;
    +        fhiPtr->windowSize = windowSize;
    +        fhiPtr->dictID = dictID;
    +        fhiPtr->checksumFlag = checksumFlag;
         }
         return 0;
     }
    @@ -2182,6 +2183,15 @@ size_t ZSTD_sizeof_DStream(const ZSTD_DStream* zds)
                + zds->inBuffSize + zds->outBuffSize;
     }
     
    +size_t ZSTD_estimateDStreamSize(ZSTD_frameHeader fHeader)
    +{
    +    size_t const windowSize = fHeader.windowSize;
    +    size_t const blockSize = MIN(windowSize, ZSTD_BLOCKSIZE_ABSOLUTEMAX);
    +    size_t const inBuffSize = blockSize;  /* no block can be larger */
    +    size_t const outBuffSize = windowSize + blockSize + (WILDCOPY_OVERLENGTH * 2);
    +    return sizeof(ZSTD_DStream) + ZSTD_estimateDCtxSize() + inBuffSize + outBuffSize;
    +}
    +
     
     /* *****   Decompression   ***** */
     
    diff --git a/lib/zstd.h b/lib/zstd.h
    index ddcaaf8bd..08754cb2a 100644
    --- a/lib/zstd.h
    +++ b/lib/zstd.h
    @@ -398,6 +398,13 @@ typedef struct {
         ZSTD_frameParameters fParams;
     } ZSTD_parameters;
     
    +typedef struct {
    +    unsigned long long frameContentSize;
    +    unsigned windowSize;
    +    unsigned dictID;
    +    unsigned checksumFlag;
    +} ZSTD_frameHeader;
    +
     /*= Custom memory allocation functions */
     typedef void* (*ZSTD_allocFunction) (void* opaque, size_t size);
     typedef void  (*ZSTD_freeFunction) (void* opaque, void* address);
    @@ -479,6 +486,7 @@ ZSTDLIB_API size_t ZSTD_estimateDCtxSize(void);
      *         an internal ?Dict will be created, which size is not estimated.
      *         In this case, get additional size by using ZSTD_estimate?DictSize */
     ZSTDLIB_API size_t ZSTD_estimateCStreamSize(ZSTD_compressionParameters cParams);
    +ZSTDLIB_API size_t ZSTD_estimateDStreamSize(ZSTD_frameHeader fHeader);
     
     /*! ZSTD_estimate?DictSize() :
      *  Note : if dictionary is created "byReference", reduce estimation by dictSize */
    @@ -741,13 +749,6 @@ ZSTDLIB_API size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapaci
       It also returns Frame Size as fparamsPtr->frameContentSize.
     */
     
    -typedef struct {
    -    unsigned long long frameContentSize;
    -    unsigned windowSize;
    -    unsigned dictID;
    -    unsigned checksumFlag;
    -} ZSTD_frameHeader;
    -
     /*=====   Buffer-less streaming decompression functions  =====*/
     ZSTDLIB_API size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fparamsPtr, const void* src, size_t srcSize);   /**< doesn't consume input, see details below */
     ZSTDLIB_API size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx);
    diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c
    index 6c210f2ff..6ce309c55 100644
    --- a/tests/zstreamtest.c
    +++ b/tests/zstreamtest.c
    @@ -262,7 +262,20 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
         }   }
         DISPLAYLEVEL(3, "OK \n");
     
    -    DISPLAYLEVEL(3, "test%3i : check DStream size : ", testNb++);
    +    /* context size functions */
    +    DISPLAYLEVEL(3, "test%3i : estimate DStream size : ", testNb++);
    +    {   ZSTD_frameHeader fhi;
    +        const void* cStart = (char*)compressedBuffer + (skippableFrameSize + 8);
    +        size_t const gfhError = ZSTD_getFrameHeader(&fhi, cStart, cSize);
    +        if (gfhError!=0) goto _output_error;
    +        DISPLAYLEVEL(5, " (windowSize : %u) ", fhi.windowSize);
    +        {   size_t const s = ZSTD_estimateDStreamSize(fhi)
    +                           + ZSTD_estimateDDictSize(128 KB);  /* uses ZSTD_initDStream_usingDict() */
    +            if (ZSTD_isError(s)) goto _output_error;
    +            DISPLAYLEVEL(3, "OK (%u bytes) \n", (U32)s);
    +    }   }
    +
    +    DISPLAYLEVEL(3, "test%3i : check actual DStream size : ", testNb++);
         { size_t const s = ZSTD_sizeof_DStream(zd);
           if (ZSTD_isError(s)) goto _output_error;
           DISPLAYLEVEL(3, "OK (%u bytes) \n", (U32)s);
    
    From 461a9cc7c62f9d6fc5bcb2725601e2e073a0f392 Mon Sep 17 00:00:00 2001
    From: Yann Collet 
    Date: Tue, 9 May 2017 16:53:09 -0700
    Subject: [PATCH 301/305] fixed symbols test
    
    ---
     tests/symbols.c | 2 +-
     1 file changed, 1 insertion(+), 1 deletion(-)
    
    diff --git a/tests/symbols.c b/tests/symbols.c
    index 7dacfc058..ade3aa02c 100644
    --- a/tests/symbols.c
    +++ b/tests/symbols.c
    @@ -88,7 +88,7 @@ static const void *symbols[] = {
       &ZSTD_copyCCtx,
       &ZSTD_compressContinue,
       &ZSTD_compressEnd,
    -  &ZSTD_getFrameParams,
    +  &ZSTD_getFrameHeader,
       &ZSTD_decompressBegin,
       &ZSTD_decompressBegin_usingDict,
       &ZSTD_copyDCtx,
    
    From 6fb2f241322ec6f66ce53132f4226bed20f5f1bc Mon Sep 17 00:00:00 2001
    From: Yann Collet 
    Date: Wed, 10 May 2017 11:06:06 -0700
    Subject: [PATCH 302/305] shortened ZSTD_createCStream_Advanced()
    
    https://github.com/facebook/zstd/pull/689#discussion_r115637613
    ---
     lib/compress/zstd_compress.c | 12 ++----------
     1 file changed, 2 insertions(+), 10 deletions(-)
    
    diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
    index 37590fca7..b1d799cdc 100644
    --- a/lib/compress/zstd_compress.c
    +++ b/lib/compress/zstd_compress.c
    @@ -3136,16 +3136,8 @@ ZSTD_CStream* ZSTD_createCStream(void)
     
     ZSTD_CStream* ZSTD_createCStream_advanced(ZSTD_customMem customMem)
     {
    -    ZSTD_CStream* zcs;
    -
    -    if (!customMem.customAlloc && !customMem.customFree) customMem = defaultCustomMem;
    -    if (!customMem.customAlloc || !customMem.customFree) return NULL;
    -
    -    zcs = (ZSTD_CStream*)ZSTD_malloc(sizeof(ZSTD_CStream), customMem);
    -    if (zcs==NULL) return NULL;
    -    memset(zcs, 0, sizeof(ZSTD_CStream));
    -    memcpy(&zcs->customMem, &customMem, sizeof(ZSTD_customMem));
    -    return zcs;
    +    /* CStream and CCtx are now same object */
    +    return ZSTD_createCCtx_advanced(customMem);
     }
     
     size_t ZSTD_freeCStream(ZSTD_CStream* zcs)
    
    From 669346fe8b015c26586c8e3ca75751fffd161bb0 Mon Sep 17 00:00:00 2001
    From: Yann Collet 
    Date: Wed, 10 May 2017 11:08:00 -0700
    Subject: [PATCH 303/305] fixed ZSTD_estimateCStreamSize()
    
    https://github.com/facebook/zstd/pull/689#discussion_r115637721
    ---
     lib/compress/zstd_compress.c | 2 +-
     1 file changed, 1 insertion(+), 1 deletion(-)
    
    diff --git a/lib/compress/zstd_compress.c b/lib/compress/zstd_compress.c
    index b1d799cdc..848472ec4 100644
    --- a/lib/compress/zstd_compress.c
    +++ b/lib/compress/zstd_compress.c
    @@ -3153,7 +3153,7 @@ size_t ZSTD_estimateCStreamSize(ZSTD_compressionParameters cParams)
         size_t const outBuffSize = ZSTD_compressBound(blockSize) + 1;
         size_t const streamingSize = inBuffSize + outBuffSize;
     
    -    return sizeof(ZSTD_CCtx) + CCtxSize + streamingSize;
    +    return CCtxSize + streamingSize;
     }
     
     
    
    From cef02d9317743fba04673fa2da038670cd07f8b6 Mon Sep 17 00:00:00 2001
    From: Yann Collet 
    Date: Wed, 10 May 2017 11:14:08 -0700
    Subject: [PATCH 304/305] changed variable name, for clarity
    
    fhiPtr -> zfhPtr
    https://github.com/facebook/zstd/pull/689#discussion_r115638676
    ---
     doc/zstd_manual.html             |  2 +-
     lib/decompress/zstd_decompress.c | 18 +++++++++---------
     lib/zstd.h                       |  2 +-
     3 files changed, 11 insertions(+), 11 deletions(-)
    
    diff --git a/doc/zstd_manual.html b/doc/zstd_manual.html
    index 9cfb1b68b..cb376741f 100644
    --- a/doc/zstd_manual.html
    +++ b/doc/zstd_manual.html
    @@ -646,7 +646,7 @@ size_t ZSTD_copyCCtx(ZSTD_CCtx* cctx, const ZSTD_CCtx* preparedCCtx, unsigned lo
       It also returns Frame Size as fparamsPtr->frameContentSize.
     
    -

    Buffer-less streaming decompression functions

    size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fparamsPtr, const void* src, size_t srcSize);   /**< doesn't consume input, see details below */
    +

    Buffer-less streaming decompression functions

    size_t ZSTD_getFrameHeader(ZSTD_frameHeader* zfhPtr, const void* src, size_t srcSize);   /**< doesn't consume input, see details below */
     size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx);
     size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx* dctx, const void* dict, size_t dictSize);
     void   ZSTD_copyDCtx(ZSTD_DCtx* dctx, const ZSTD_DCtx* preparedDCtx);
    diff --git a/lib/decompress/zstd_decompress.c b/lib/decompress/zstd_decompress.c
    index 267aa94d0..379842b57 100644
    --- a/lib/decompress/zstd_decompress.c
    +++ b/lib/decompress/zstd_decompress.c
    @@ -221,10 +221,10 @@ static size_t ZSTD_frameHeaderSize(const void* src, size_t srcSize)
     
     /** ZSTD_getFrameHeader() :
     *   decode Frame Header, or require larger `srcSize`.
    -*   @return : 0, `fhiPtr` is correctly filled,
    +*   @return : 0, `zfhPtr` is correctly filled,
     *            >0, `srcSize` is too small, result is expected `srcSize`,
     *             or an error code, which can be tested using ZSTD_isError() */
    -size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fhiPtr, const void* src, size_t srcSize)
    +size_t ZSTD_getFrameHeader(ZSTD_frameHeader* zfhPtr, const void* src, size_t srcSize)
     {
         const BYTE* ip = (const BYTE*)src;
         if (srcSize < ZSTD_frameHeaderSize_prefix) return ZSTD_frameHeaderSize_prefix;
    @@ -233,9 +233,9 @@ size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fhiPtr, const void* src, size_t src
             if ((MEM_readLE32(src) & 0xFFFFFFF0U) == ZSTD_MAGIC_SKIPPABLE_START) {
                 /* skippable frame */
                 if (srcSize < ZSTD_skippableHeaderSize) return ZSTD_skippableHeaderSize; /* magic number + skippable frame length */
    -            memset(fhiPtr, 0, sizeof(*fhiPtr));
    -            fhiPtr->frameContentSize = MEM_readLE32((const char *)src + 4);
    -            fhiPtr->windowSize = 0; /* windowSize==0 means a frame is skippable */
    +            memset(zfhPtr, 0, sizeof(*zfhPtr));
    +            zfhPtr->frameContentSize = MEM_readLE32((const char *)src + 4);
    +            zfhPtr->windowSize = 0; /* windowSize==0 means a frame is skippable */
                 return 0;
             }
             return ERROR(prefix_unknown);
    @@ -282,10 +282,10 @@ size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fhiPtr, const void* src, size_t src
             }
             if (!windowSize) windowSize = (U32)frameContentSize;
             if (windowSize > windowSizeMax) return ERROR(frameParameter_windowTooLarge);
    -        fhiPtr->frameContentSize = frameContentSize;
    -        fhiPtr->windowSize = windowSize;
    -        fhiPtr->dictID = dictID;
    -        fhiPtr->checksumFlag = checksumFlag;
    +        zfhPtr->frameContentSize = frameContentSize;
    +        zfhPtr->windowSize = windowSize;
    +        zfhPtr->dictID = dictID;
    +        zfhPtr->checksumFlag = checksumFlag;
         }
         return 0;
     }
    diff --git a/lib/zstd.h b/lib/zstd.h
    index 08754cb2a..c956860ab 100644
    --- a/lib/zstd.h
    +++ b/lib/zstd.h
    @@ -750,7 +750,7 @@ ZSTDLIB_API size_t ZSTD_compressEnd(ZSTD_CCtx* cctx, void* dst, size_t dstCapaci
     */
     
     /*=====   Buffer-less streaming decompression functions  =====*/
    -ZSTDLIB_API size_t ZSTD_getFrameHeader(ZSTD_frameHeader* fparamsPtr, const void* src, size_t srcSize);   /**< doesn't consume input, see details below */
    +ZSTDLIB_API size_t ZSTD_getFrameHeader(ZSTD_frameHeader* zfhPtr, const void* src, size_t srcSize);   /**< doesn't consume input, see details below */
     ZSTDLIB_API size_t ZSTD_decompressBegin(ZSTD_DCtx* dctx);
     ZSTDLIB_API size_t ZSTD_decompressBegin_usingDict(ZSTD_DCtx* dctx, const void* dict, size_t dictSize);
     ZSTDLIB_API void   ZSTD_copyDCtx(ZSTD_DCtx* dctx, const ZSTD_DCtx* preparedDCtx);
    
    From 30ab64e21d09f8b493c2ae7968c7139b5d197e23 Mon Sep 17 00:00:00 2001
    From: Yann Collet 
    Date: Wed, 10 May 2017 11:30:19 -0700
    Subject: [PATCH 305/305] added test for ZSTD_estimateCStreamSize()
    
    ---
     tests/zstreamtest.c | 26 ++++++++++++++++++--------
     1 file changed, 18 insertions(+), 8 deletions(-)
    
    diff --git a/tests/zstreamtest.c b/tests/zstreamtest.c
    index 6ce309c55..5c166bfb8 100644
    --- a/tests/zstreamtest.c
    +++ b/tests/zstreamtest.c
    @@ -163,6 +163,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
         ZSTD_inBuffer  inBuff, inBuff2;
         ZSTD_outBuffer outBuff;
         buffer_t dictionary = g_nullBuffer;
    +    size_t const dictSize = 128 KB;
         unsigned dictID = 0;
     
         /* Create compressible test buffer */
    @@ -188,7 +189,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
     
         /* Basic compression test */
         DISPLAYLEVEL(3, "test%3i : compress %u bytes : ", testNb++, COMPRESSIBLE_NOISE_LENGTH);
    -    ZSTD_initCStream_usingDict(zc, CNBuffer, 128 KB, 1);
    +    ZSTD_initCStream_usingDict(zc, CNBuffer, dictSize, 1);
         outBuff.dst = (char*)(compressedBuffer)+cSize;
         outBuff.size = compressedBufferSize;
         outBuff.pos = 0;
    @@ -203,7 +204,16 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
         cSize += outBuff.pos;
         DISPLAYLEVEL(3, "OK (%u bytes : %.2f%%)\n", (U32)cSize, (double)cSize/COMPRESSIBLE_NOISE_LENGTH*100);
     
    -    DISPLAYLEVEL(3, "test%3i : check CStream size : ", testNb++);
    +    /* context size functions */
    +    DISPLAYLEVEL(3, "test%3i : estimate CStream size : ", testNb++);
    +    {   ZSTD_compressionParameters const cParams = ZSTD_getCParams(1, CNBufferSize, dictSize);
    +        size_t const s = ZSTD_estimateCStreamSize(cParams)
    +                       + ZSTD_estimateCDictSize(cParams, dictSize);  /* uses ZSTD_initCStream_usingDict() */
    +            if (ZSTD_isError(s)) goto _output_error;
    +            DISPLAYLEVEL(3, "OK (%u bytes) \n", (U32)s);
    +    }
    +
    +    DISPLAYLEVEL(3, "test%3i : check actual CStream size : ", testNb++);
         { size_t const s = ZSTD_sizeof_CStream(zc);
           if (ZSTD_isError(s)) goto _output_error;
           DISPLAYLEVEL(3, "OK (%u bytes) \n", (U32)s);
    @@ -221,7 +231,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
     
         /* skippable frame test */
         DISPLAYLEVEL(3, "test%3i : decompress skippable frame : ", testNb++);
    -    ZSTD_initDStream_usingDict(zd, CNBuffer, 128 KB);
    +    ZSTD_initDStream_usingDict(zd, CNBuffer, dictSize);
         inBuff.src = compressedBuffer;
         inBuff.size = cSize;
         inBuff.pos = 0;
    @@ -236,7 +246,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
         /* Basic decompression test */
         inBuff2 = inBuff;
         DISPLAYLEVEL(3, "test%3i : decompress %u bytes : ", testNb++, COMPRESSIBLE_NOISE_LENGTH);
    -    ZSTD_initDStream_usingDict(zd, CNBuffer, 128 KB);
    +    ZSTD_initDStream_usingDict(zd, CNBuffer, dictSize);
         { size_t const r = ZSTD_setDStreamParameter(zd, DStream_p_maxWindowSize, 1000000000);  /* large limit */
           if (ZSTD_isError(r)) goto _output_error; }
         { size_t const remaining = ZSTD_decompressStream(zd, &outBuff, &inBuff);
    @@ -270,7 +280,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
             if (gfhError!=0) goto _output_error;
             DISPLAYLEVEL(5, " (windowSize : %u) ", fhi.windowSize);
             {   size_t const s = ZSTD_estimateDStreamSize(fhi)
    -                           + ZSTD_estimateDDictSize(128 KB);  /* uses ZSTD_initDStream_usingDict() */
    +                           + ZSTD_estimateDDictSize(dictSize);  /* uses ZSTD_initDStream_usingDict() */
                 if (ZSTD_isError(s)) goto _output_error;
                 DISPLAYLEVEL(3, "OK (%u bytes) \n", (U32)s);
         }   }
    @@ -285,7 +295,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
         DISPLAYLEVEL(3, "test%3i : decompress byte-by-byte : ", testNb++);
         {   /* skippable frame */
             size_t r = 1;
    -        ZSTD_initDStream_usingDict(zd, CNBuffer, 128 KB);
    +        ZSTD_initDStream_usingDict(zd, CNBuffer, dictSize);
             inBuff.src = compressedBuffer;
             outBuff.dst = decodedBuffer;
             inBuff.pos = 0;
    @@ -297,7 +307,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
                 if (ZSTD_isError(r)) goto _output_error;
             }
             /* normal frame */
    -        ZSTD_initDStream_usingDict(zd, CNBuffer, 128 KB);
    +        ZSTD_initDStream_usingDict(zd, CNBuffer, dictSize);
             r=1;
             while (r) {
                 inBuff.size = inBuff.pos + 1;
    @@ -456,7 +466,7 @@ static int basicUnitTests(U32 seed, double compressibility, ZSTD_customMem custo
     
         /* Memory restriction */
         DISPLAYLEVEL(3, "test%3i : maxWindowSize < frame requirement : ", testNb++);
    -    ZSTD_initDStream_usingDict(zd, CNBuffer, 128 KB);
    +    ZSTD_initDStream_usingDict(zd, CNBuffer, dictSize);
         { size_t const r = ZSTD_setDStreamParameter(zd, DStream_p_maxWindowSize, 1000);  /* too small limit */
           if (ZSTD_isError(r)) goto _output_error; }
         inBuff.src = compressedBuffer;