Update QuickLZ to 1.5.1 beta 7

git-svn-id: https://svn.eduke32.com/eduke32@2662 1a8010ca-5511-0410-912e-c29ae57300e0
This commit is contained in:
terminx 2012-05-17 21:36:50 +00:00
parent 265ff9dbfd
commit ad2f6a6560
2 changed files with 224 additions and 206 deletions

View file

@ -2,27 +2,20 @@
#define QLZ_HEADER #define QLZ_HEADER
// Fast data compression library // Fast data compression library
// Copyright (C) 2006-2009 Lasse Mikkel Reinhold // Copyright (C) 2006-2011 Lasse Mikkel Reinhold
// lar@quicklz.com // lar@quicklz.com
// //
// QuickLZ can be used for free under the GPL-1 or GPL-2 license (where anything // QuickLZ can be used for free under the GPL 1, 2 or 3 license (where anything
// released into public must be open source) or under a commercial license if such // released into public must be open source) or under a commercial license if such
// has been acquired (see http://www.quicklz.com/order.html). The commercial license // has been acquired (see http://www.quicklz.com/order.html). The commercial license
// does not cover derived or ported versions created by third parties under GPL. // does not cover derived or ported versions created by third parties under GPL.
// You can edit following user settings. Data must be decompressed with the same // You can edit following user settings. Data must be decompressed with the same
// setting of QLZ_COMPRESSION_LEVEL and QLZ_STREAMING_BUFFER as it was compressed // setting of QLZ_COMPRESSION_LEVEL and QLZ_STREAMING_BUFFER as it was compressed
// (see manual). If QLZ_STREAMING_BUFFER > 0, scratch buffers must be initially // (see manual). First #ifndef makes it possible to define settings from
// zeroed out (see manual). First #ifndef makes it possible to define settings from
// the outside like the compiler command line. // the outside like the compiler command line.
// BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION // 1.5.1 BETA 7
// BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION
// BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION
// BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION
// BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION
// 1.5.0 BETA 2
#ifndef QLZ_COMPRESSION_LEVEL #ifndef QLZ_COMPRESSION_LEVEL
//#define QLZ_COMPRESSION_LEVEL 1 //#define QLZ_COMPRESSION_LEVEL 1
@ -38,7 +31,7 @@
#define QLZ_VERSION_MAJOR 1 #define QLZ_VERSION_MAJOR 1
#define QLZ_VERSION_MINOR 5 #define QLZ_VERSION_MINOR 5
#define QLZ_VERSION_REVISION 0 #define QLZ_VERSION_REVISION 1
// Using size_t, memset() and memcpy() // Using size_t, memset() and memcpy()
#include <string.h> #include <string.h>
@ -122,7 +115,9 @@ typedef struct
#if QLZ_STREAMING_BUFFER > 0 #if QLZ_STREAMING_BUFFER > 0
unsigned char stream_buffer[QLZ_STREAMING_BUFFER]; unsigned char stream_buffer[QLZ_STREAMING_BUFFER];
#endif #endif
#if QLZ_COMPRESSION_LEVEL <= 2
qlz_hash_decompress hash[QLZ_HASH_VALUES]; qlz_hash_decompress hash[QLZ_HASH_VALUES];
#endif
size_t stream_counter; size_t stream_counter;
} qlz_state_decompress; } qlz_state_decompress;
#endif #endif

View file

@ -1,23 +1,20 @@
// Fast data compression library // Fast data compression library
// Copyright (C) 2006-2009 Lasse Mikkel Reinhold // Copyright (C) 2006-2011 Lasse Mikkel Reinhold
// lar@quicklz.com // lar@quicklz.com
// //
// QuickLZ can be used for free under the GPL-1 or GPL-2 license (where anything // QuickLZ can be used for free under the GPL 1, 2 or 3 license (where anything
// released into public must be open source) or under a commercial license if such // released into public must be open source) or under a commercial license if such
// has been acquired (see http://www.quicklz.com/order.html). The commercial license // has been acquired (see http://www.quicklz.com/order.html). The commercial license
// does not cover derived or ported versions created by third parties under GPL. // does not cover derived or ported versions created by third parties under GPL.
// BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION // 1.5.1 BETA 7
// BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION
// BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION
// BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION
// BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION BETA VERSION
// 1.5.0 BETA 2
#include "quicklz.h" #include "quicklz.h"
#if defined _MSC_VER
#include <intrin.h>
#endif
#if QLZ_VERSION_MAJOR != 1 || QLZ_VERSION_MINOR != 5 || QLZ_VERSION_REVISION != 0 #if QLZ_VERSION_MAJOR != 1 || QLZ_VERSION_MINOR != 5 || QLZ_VERSION_REVISION != 1
#error quicklz.c and quicklz.h have different versions #error quicklz.c and quicklz.h have different versions
#endif #endif
@ -26,7 +23,8 @@
#endif #endif
#define MINOFFSET 2 #define MINOFFSET 2
#define UNCONDITIONAL_MATCHLEN 6 #define UNCONDITIONAL_MATCHLEN_COMPRESSOR 12
#define UNCONDITIONAL_MATCHLEN_DECOMPRESSOR 6
#define UNCOMPRESSED_END 4 #define UNCOMPRESSED_END 4
#define CWORD_LEN 4 #define CWORD_LEN 4
@ -38,6 +36,14 @@
#define CAST #define CAST
#endif #endif
#if defined(X86X64) && (defined(__GNUC__) || defined(__INTEL_COMPILER))
#define qlz_likely(x) __builtin_expect (x, 1)
#define qlz_unlikely(x) __builtin_expect (x, 0)
#else
#define qlz_likely(x) (x)
#define qlz_unlikely(x) (x)
#endif
int qlz_get_setting(int setting) int qlz_get_setting(int setting)
{ {
switch (setting) switch (setting)
@ -265,7 +271,7 @@ static size_t qlz_compress_core(const unsigned char *source, unsigned char *dest
unsigned char *cword_ptr = destination; unsigned char *cword_ptr = destination;
unsigned char *dst = destination + CWORD_LEN; unsigned char *dst = destination + CWORD_LEN;
ui32 cword_val = 1U << 31; ui32 cword_val = 1U << 31;
const unsigned char *last_matchstart = last_byte - UNCONDITIONAL_MATCHLEN - UNCOMPRESSED_END; const unsigned char *last_matchstart = last_byte - UNCONDITIONAL_MATCHLEN_COMPRESSOR - UNCOMPRESSED_END;
ui32 fetch = 0; ui32 fetch = 0;
unsigned int lits = 0; unsigned int lits = 0;
@ -274,9 +280,9 @@ static size_t qlz_compress_core(const unsigned char *source, unsigned char *dest
if (src <= last_matchstart) if (src <= last_matchstart)
fetch = fast_read(src, 3); fetch = fast_read(src, 3);
while (src <= last_matchstart) while (qlz_likely(src <= last_matchstart))
{ {
if ((cword_val & 1) == 1) if (qlz_unlikely((cword_val & 1) == 1))
{ {
// store uncompressed if compression ratio is too low // store uncompressed if compression ratio is too low
if (src > source + (size >> 1) && dst - destination > src - source - ((src - source) >> 5)) if (src > source + (size >> 1) && dst - destination > src - source - ((src - source) >> 5))
@ -300,48 +306,60 @@ static size_t qlz_compress_core(const unsigned char *source, unsigned char *dest
o = state->hash[hash].offset + OFFSET_BASE; o = state->hash[hash].offset + OFFSET_BASE;
state->hash[hash].offset = CAST(src - OFFSET_BASE); state->hash[hash].offset = CAST(src - OFFSET_BASE);
#ifdef X86X64 #ifdef X86X64
if ((cached & 0xffffff) == 0 && o != OFFSET_BASE && (src - o > MINOFFSET || (src == o + 1 && lits >= 3 && src > source + 3 && same(src - 3, 6)))) if ((cached & 0xffffff) == 0 && o != OFFSET_BASE && (src - o > MINOFFSET || (src == o + 1 && lits >= 3 && src > source + 3 && same(src - 3, 6))))
{ {
if (cached != 0)
{
#else #else
if (cached == 0 && o != OFFSET_BASE && (src - o > MINOFFSET || (src == o + 1 && lits >= 3 && src > source + 3 && same(src - 3, 6)))) if (cached == 0 && o != OFFSET_BASE && (src - o > MINOFFSET || (src == o + 1 && lits >= 3 && src > source + 3 && same(src - 3, 6))))
{ {
if (*(o + 3) != *(src + 3))
{
#endif #endif
size_t matchlen = 3;
hash <<= 4; hash <<= 4;
cword_val = (cword_val >> 1) | (1U << 31); cword_val = (cword_val >> 1) | (1U << 31);
fast_write((3 - 2) | hash, dst, 2);
src += 3; #if defined X86X64 && defined QLZ_PTR_64
dst += 2; {
#ifdef __GNUC__
unsigned long long a = *(unsigned long long *)(src + matchlen);
unsigned long long b = *(unsigned long long *)(o + matchlen);
unsigned long long c = a^b;
#else
unsigned int a = *(unsigned int *)(src + matchlen);
unsigned int b = *(unsigned int *)(o + matchlen);
unsigned int c = a^b;
#endif
if (qlz_unlikely(c == 0))
{
size_t q = last_byte - UNCOMPRESSED_END - src + 1;
size_t remaining = q > 255 ? 255 : q;
matchlen += sizeof(c);
while (src[matchlen] == o[matchlen] && matchlen < remaining)
matchlen++;
} }
else else
{ {
const unsigned char *old_src = src; #if defined _MSC_VER || defined __INTEL_COMPILER
size_t matchlen; unsigned int index = 0;
hash <<= 4; _BitScanForward((unsigned long *)&index, c);
matchlen += index >> 3;
cword_val = (cword_val >> 1) | (1U << 31); #else
src += 4; matchlen += __builtin_ctzll(c) >> 3;
#endif
if (*(o + (src - old_src)) == *src) }
}
#else
if (src[matchlen] == o[matchlen])
{ {
src++; size_t q = last_byte - UNCOMPRESSED_END - src + 1;
if (*(o + (src - old_src)) == *src)
{
size_t q = last_byte - UNCOMPRESSED_END - (src - 5) + 1;
size_t remaining = q > 255 ? 255 : q; size_t remaining = q > 255 ? 255 : q;
src++; matchlen ++;
while (*(o + (src - old_src)) == *src && (size_t)(src - old_src) < remaining) while (src[matchlen] == o[matchlen] && matchlen < remaining)
src++; matchlen++;
}
} }
#endif
src += matchlen;
matchlen = src - old_src; if (qlz_likely(matchlen < 18))
if (matchlen < 18)
{ {
fast_write((ui32)(matchlen - 2) | hash, dst, 2); fast_write((ui32)(matchlen - 2) | hash, dst, 2);
dst += 2; dst += 2;
@ -351,7 +369,7 @@ static size_t qlz_compress_core(const unsigned char *source, unsigned char *dest
fast_write((ui32)(matchlen << 16) | hash, dst, 3); fast_write((ui32)(matchlen << 16) | hash, dst, 3);
dst += 3; dst += 3;
} }
}
fetch = fast_read(src, 3); fetch = fast_read(src, 3);
lits = 0; lits = 0;
} }
@ -377,8 +395,6 @@ static size_t qlz_compress_core(const unsigned char *source, unsigned char *dest
size_t remaining = (last_byte - UNCOMPRESSED_END - src + 1) > 255 ? 255 : (last_byte - UNCOMPRESSED_END - src + 1); size_t remaining = (last_byte - UNCOMPRESSED_END - src + 1) > 255 ? 255 : (last_byte - UNCOMPRESSED_END - src + 1);
(void)best_k; (void)best_k;
//hash = hashat(src);
fetch = fast_read(src, 3); fetch = fast_read(src, 3);
hash = hash_func(fetch); hash = hash_func(fetch);
@ -561,7 +577,7 @@ static size_t qlz_decompress_core(const unsigned char *source, unsigned char *de
unsigned char *dst = destination; unsigned char *dst = destination;
const unsigned char *last_destination_byte = destination + size - 1; const unsigned char *last_destination_byte = destination + size - 1;
ui32 cword_val = 1; ui32 cword_val = 1;
const unsigned char *last_matchstart = last_destination_byte - UNCONDITIONAL_MATCHLEN - UNCOMPRESSED_END; const unsigned char *last_matchstart = last_destination_byte - UNCONDITIONAL_MATCHLEN_DECOMPRESSOR - UNCOMPRESSED_END;
unsigned char *last_hashed = destination - 1; unsigned char *last_hashed = destination - 1;
const unsigned char *last_source_byte = source + qlz_size_compressed((const char *)source) - 1; const unsigned char *last_source_byte = source + qlz_size_compressed((const char *)source) - 1;
static const ui32 bitlut[16] = {4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0}; static const ui32 bitlut[16] = {4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0};
@ -815,6 +831,7 @@ size_t qlz_compress(const void *source, char *destination, size_t size, qlz_stat
size_t qlz_decompress(const char *source, void *destination, qlz_state_decompress *state) size_t qlz_decompress(const char *source, void *destination, qlz_state_decompress *state)
{ {
size_t dsiz = qlz_size_decompressed(source); size_t dsiz = qlz_size_decompressed(source);
size_t csiz = qlz_size_compressed(source);
#if QLZ_STREAMING_BUFFER > 0 #if QLZ_STREAMING_BUFFER > 0
if (state->stream_counter + qlz_size_decompressed(source) - 1 >= QLZ_STREAMING_BUFFER) if (state->stream_counter + qlz_size_decompressed(source) - 1 >= QLZ_STREAMING_BUFFER)
@ -827,6 +844,9 @@ size_t qlz_decompress(const char *source, void *destination, qlz_state_decompres
} }
else else
{ {
if (csiz != dsiz + qlz_size_header(source))
return 0;
memcpy(destination, source + qlz_size_header(source), dsiz); memcpy(destination, source + qlz_size_header(source), dsiz);
} }
state->stream_counter = 0; state->stream_counter = 0;
@ -842,6 +862,9 @@ size_t qlz_decompress(const char *source, void *destination, qlz_state_decompres
} }
else else
{ {
// if(csiz != dsiz + qlz_size_header(source))
// return 0;
memcpy(dst, source + qlz_size_header(source), dsiz); memcpy(dst, source + qlz_size_header(source), dsiz);
reset_table_decompress(state); reset_table_decompress(state);
} }