2025-02-20 21:31:06 +01:00
|
|
|
/* Lzlib - Compression library for the lzip format
|
2025-02-20 21:35:19 +01:00
|
|
|
Copyright (C) 2009-2025 Antonio Diaz Diaz.
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:31:06 +01:00
|
|
|
This library is free software. Redistribution and use in source and
|
|
|
|
binary forms, with or without modification, are permitted provided
|
|
|
|
that the following conditions are met:
|
2025-02-20 21:23:49 +01:00
|
|
|
|
2025-02-20 21:31:06 +01:00
|
|
|
1. Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions, and the following disclaimer.
|
2025-02-20 21:23:49 +01:00
|
|
|
|
2025-02-20 21:31:06 +01:00
|
|
|
2. 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.
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:31:06 +01:00
|
|
|
This library 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.
|
2025-02-20 19:59:10 +01:00
|
|
|
*/
|
|
|
|
|
2025-02-20 21:34:58 +01:00
|
|
|
static int LZe_get_match_pairs( LZ_encoder * const e, Pair * pairs )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
int32_t * ptr0 = e->eb.mb.pos_array + ( e->eb.mb.cyclic_pos << 1 );
|
2025-02-20 19:59:10 +01:00
|
|
|
int32_t * ptr1 = ptr0 + 1;
|
2025-02-20 20:52:54 +01:00
|
|
|
int len_limit = e->match_len_limit;
|
|
|
|
if( len_limit > Mb_available_bytes( &e->eb.mb ) )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
e->been_flushed = true;
|
|
|
|
len_limit = Mb_available_bytes( &e->eb.mb );
|
2025-02-20 20:21:13 +01:00
|
|
|
if( len_limit < 4 ) { *ptr0 = *ptr1 = 0; return 0; }
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
int maxlen = 3; /* only used if pairs != 0 */
|
|
|
|
int num_pairs = 0;
|
2025-02-20 21:34:58 +01:00
|
|
|
const int min_pos = (e->eb.mb.pos > e->eb.mb.dictionary_size) ?
|
|
|
|
e->eb.mb.pos - e->eb.mb.dictionary_size : 0;
|
2025-02-20 21:32:02 +01:00
|
|
|
const uint8_t * const data = Mb_ptr_to_current_pos( &e->eb.mb );
|
|
|
|
|
|
|
|
unsigned tmp = crc32[data[0]] ^ data[1];
|
|
|
|
const int key2 = tmp & ( num_prev_positions2 - 1 );
|
2025-02-20 20:21:13 +01:00
|
|
|
tmp ^= (unsigned)data[2] << 8;
|
2025-02-20 21:32:02 +01:00
|
|
|
const int key3 = num_prev_positions2 + ( tmp & ( num_prev_positions3 - 1 ) );
|
|
|
|
const int key4 = num_prev_positions2 + num_prev_positions3 +
|
|
|
|
( ( tmp ^ ( crc32[data[3]] << 5 ) ) & e->eb.mb.key4_mask );
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 20:13:15 +01:00
|
|
|
if( pairs )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 21:31:06 +01:00
|
|
|
const int np2 = e->eb.mb.prev_positions[key2];
|
|
|
|
const int np3 = e->eb.mb.prev_positions[key3];
|
2025-02-20 20:52:54 +01:00
|
|
|
if( np2 > min_pos && e->eb.mb.buffer[np2-1] == data[0] )
|
2025-02-20 20:13:15 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
pairs[0].dis = e->eb.mb.pos - np2;
|
2025-02-20 21:32:02 +01:00
|
|
|
pairs[0].len = maxlen = 2 + ( np2 == np3 );
|
2025-02-20 20:13:15 +01:00
|
|
|
num_pairs = 1;
|
|
|
|
}
|
2025-02-20 20:52:54 +01:00
|
|
|
if( np2 != np3 && np3 > min_pos && e->eb.mb.buffer[np3-1] == data[0] )
|
2025-02-20 20:13:15 +01:00
|
|
|
{
|
|
|
|
maxlen = 3;
|
2025-02-20 21:31:06 +01:00
|
|
|
pairs[num_pairs++].dis = e->eb.mb.pos - np3;
|
2025-02-20 20:13:15 +01:00
|
|
|
}
|
|
|
|
if( num_pairs > 0 )
|
|
|
|
{
|
2025-02-20 21:31:06 +01:00
|
|
|
const int delta = pairs[num_pairs-1].dis + 1;
|
2025-02-20 20:13:15 +01:00
|
|
|
while( maxlen < len_limit && data[maxlen-delta] == data[maxlen] )
|
|
|
|
++maxlen;
|
|
|
|
pairs[num_pairs-1].len = maxlen;
|
2025-02-20 21:31:06 +01:00
|
|
|
if( maxlen < 3 ) maxlen = 3;
|
2025-02-20 20:21:13 +01:00
|
|
|
if( maxlen >= len_limit ) pairs = 0; /* done. now just skip */
|
2025-02-20 20:13:15 +01:00
|
|
|
}
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
const int pos1 = e->eb.mb.pos + 1;
|
2025-02-20 20:52:54 +01:00
|
|
|
e->eb.mb.prev_positions[key2] = pos1;
|
|
|
|
e->eb.mb.prev_positions[key3] = pos1;
|
2025-02-20 21:32:02 +01:00
|
|
|
int newpos1 = e->eb.mb.prev_positions[key4];
|
2025-02-20 20:52:54 +01:00
|
|
|
e->eb.mb.prev_positions[key4] = pos1;
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
int len = 0, len0 = 0, len1 = 0;
|
|
|
|
|
|
|
|
int count;
|
2025-02-20 20:52:54 +01:00
|
|
|
for( count = e->cycles; ; )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 21:23:49 +01:00
|
|
|
if( newpos1 <= min_pos || --count < 0 ) { *ptr0 = *ptr1 = 0; break; }
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 20:52:54 +01:00
|
|
|
if( e->been_flushed ) len = 0;
|
2025-02-20 21:32:02 +01:00
|
|
|
const int delta = pos1 - newpos1;
|
|
|
|
int32_t * const newptr = e->eb.mb.pos_array +
|
2025-02-20 20:52:54 +01:00
|
|
|
( ( e->eb.mb.cyclic_pos - delta +
|
|
|
|
( (e->eb.mb.cyclic_pos >= delta) ? 0 : e->eb.mb.dictionary_size + 1 ) ) << 1 );
|
2025-02-20 20:13:15 +01:00
|
|
|
if( data[len-delta] == data[len] )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:13:15 +01:00
|
|
|
while( ++len < len_limit && data[len-delta] == data[len] ) {}
|
|
|
|
if( pairs && maxlen < len )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:13:15 +01:00
|
|
|
pairs[num_pairs].dis = delta - 1;
|
|
|
|
pairs[num_pairs].len = maxlen = len;
|
|
|
|
++num_pairs;
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
2025-02-20 20:13:15 +01:00
|
|
|
if( len >= len_limit )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:13:15 +01:00
|
|
|
*ptr0 = newptr[0];
|
|
|
|
*ptr1 = newptr[1];
|
|
|
|
break;
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
}
|
2025-02-20 20:13:15 +01:00
|
|
|
if( data[len-delta] < data[len] )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 21:23:49 +01:00
|
|
|
*ptr0 = newpos1;
|
2025-02-20 20:13:15 +01:00
|
|
|
ptr0 = newptr + 1;
|
2025-02-20 21:23:49 +01:00
|
|
|
newpos1 = *ptr0;
|
2025-02-20 20:13:15 +01:00
|
|
|
len0 = len; if( len1 < len ) len = len1;
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
2025-02-20 20:13:15 +01:00
|
|
|
else
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 21:23:49 +01:00
|
|
|
*ptr1 = newpos1;
|
2025-02-20 20:13:15 +01:00
|
|
|
ptr1 = newptr;
|
2025-02-20 21:23:49 +01:00
|
|
|
newpos1 = *ptr1;
|
2025-02-20 20:13:15 +01:00
|
|
|
len1 = len; if( len0 < len ) len = len0;
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
}
|
2025-02-20 20:13:15 +01:00
|
|
|
return num_pairs;
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2025-02-20 21:34:58 +01:00
|
|
|
static void LZe_update_distance_prices( LZ_encoder * const e )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:20:27 +01:00
|
|
|
int dis, len_state;
|
2025-02-20 19:59:10 +01:00
|
|
|
for( dis = start_dis_model; dis < modeled_distances; ++dis )
|
|
|
|
{
|
|
|
|
const int dis_slot = dis_slots[dis];
|
|
|
|
const int direct_bits = ( dis_slot >> 1 ) - 1;
|
|
|
|
const int base = ( 2 | ( dis_slot & 1 ) ) << direct_bits;
|
2025-02-20 21:23:49 +01:00
|
|
|
const int price = price_symbol_reversed( e->eb.bm_dis + ( base - dis_slot ),
|
2025-02-20 20:21:13 +01:00
|
|
|
dis - base, direct_bits );
|
2025-02-20 20:20:27 +01:00
|
|
|
for( len_state = 0; len_state < len_states; ++len_state )
|
2025-02-20 20:21:13 +01:00
|
|
|
e->dis_prices[len_state][dis] = price;
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
|
2025-02-20 20:20:27 +01:00
|
|
|
for( len_state = 0; len_state < len_states; ++len_state )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:21:13 +01:00
|
|
|
int * const dsp = e->dis_slot_prices[len_state];
|
2025-02-20 20:52:54 +01:00
|
|
|
const Bit_model * const bmds = e->eb.bm_dis_slot[len_state];
|
2025-02-20 19:59:10 +01:00
|
|
|
int slot = 0;
|
2025-02-20 20:21:13 +01:00
|
|
|
for( ; slot < end_dis_model; ++slot )
|
2025-02-20 21:23:49 +01:00
|
|
|
dsp[slot] = price_symbol6( bmds, slot );
|
2025-02-20 20:21:13 +01:00
|
|
|
for( ; slot < e->num_dis_slots; ++slot )
|
2025-02-20 21:23:49 +01:00
|
|
|
dsp[slot] = price_symbol6( bmds, slot ) +
|
2025-02-20 20:13:15 +01:00
|
|
|
(((( slot >> 1 ) - 1 ) - dis_align_bits ) << price_shift_bits );
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
int * const dp = e->dis_prices[len_state];
|
2025-02-20 19:59:10 +01:00
|
|
|
for( dis = 0; dis < start_dis_model; ++dis )
|
|
|
|
dp[dis] = dsp[dis];
|
|
|
|
for( ; dis < modeled_distances; ++dis )
|
|
|
|
dp[dis] += dsp[dis_slots[dis]];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
/* Return the number of bytes advanced (ahead).
|
2025-02-20 20:18:02 +01:00
|
|
|
trials[0]..trials[ahead-1] contain the steps to encode.
|
2025-02-20 21:23:49 +01:00
|
|
|
( trials[0].dis4 == -1 ) means literal.
|
2025-02-20 20:23:05 +01:00
|
|
|
A match/rep longer or equal than match_len_limit finishes the sequence.
|
2025-02-20 19:59:10 +01:00
|
|
|
*/
|
2025-02-20 21:34:58 +01:00
|
|
|
static int LZe_sequence_optimizer( LZ_encoder * const e,
|
2025-02-20 19:59:10 +01:00
|
|
|
const int reps[num_rep_distances],
|
|
|
|
const State state )
|
|
|
|
{
|
2025-02-20 21:32:02 +01:00
|
|
|
int num_pairs, num_trials;
|
|
|
|
int i, rep, len;
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 20:21:13 +01:00
|
|
|
if( e->pending_num_pairs > 0 ) /* from previous call */
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:21:13 +01:00
|
|
|
num_pairs = e->pending_num_pairs;
|
|
|
|
e->pending_num_pairs = 0;
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
2025-02-20 20:13:15 +01:00
|
|
|
else
|
2025-02-20 20:21:13 +01:00
|
|
|
num_pairs = LZe_read_match_distances( e );
|
2025-02-20 21:34:58 +01:00
|
|
|
const int main_len = (num_pairs > 0) ? e->pairs[num_pairs-1].len : 0;
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
int replens[num_rep_distances];
|
|
|
|
int rep_index = 0;
|
2025-02-20 19:59:10 +01:00
|
|
|
for( i = 0; i < num_rep_distances; ++i )
|
|
|
|
{
|
2025-02-20 21:23:49 +01:00
|
|
|
replens[i] = Mb_true_match_len( &e->eb.mb, 0, reps[i] + 1 );
|
2025-02-20 19:59:10 +01:00
|
|
|
if( replens[i] > replens[rep_index] ) rep_index = i;
|
|
|
|
}
|
2025-02-20 20:52:54 +01:00
|
|
|
if( replens[rep_index] >= e->match_len_limit )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:21:13 +01:00
|
|
|
e->trials[0].price = replens[rep_index];
|
2025-02-20 21:23:49 +01:00
|
|
|
e->trials[0].dis4 = rep_index;
|
2025-02-20 20:52:54 +01:00
|
|
|
if( !LZe_move_and_update( e, replens[rep_index] ) ) return 0;
|
2025-02-20 19:59:10 +01:00
|
|
|
return replens[rep_index];
|
|
|
|
}
|
|
|
|
|
2025-02-20 20:52:54 +01:00
|
|
|
if( main_len >= e->match_len_limit )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:21:13 +01:00
|
|
|
e->trials[0].price = main_len;
|
2025-02-20 21:23:49 +01:00
|
|
|
e->trials[0].dis4 = e->pairs[num_pairs-1].dis + num_rep_distances;
|
2025-02-20 20:52:54 +01:00
|
|
|
if( !LZe_move_and_update( e, main_len ) ) return 0;
|
2025-02-20 19:59:10 +01:00
|
|
|
return main_len;
|
|
|
|
}
|
|
|
|
|
2025-02-20 20:52:54 +01:00
|
|
|
const int pos_state = Mb_data_position( &e->eb.mb ) & pos_state_mask;
|
|
|
|
const uint8_t prev_byte = Mb_peek( &e->eb.mb, 1 );
|
|
|
|
const uint8_t cur_byte = Mb_peek( &e->eb.mb, 0 );
|
|
|
|
const uint8_t match_byte = Mb_peek( &e->eb.mb, reps[0] + 1 );
|
2025-02-20 20:21:13 +01:00
|
|
|
|
2025-02-20 20:52:54 +01:00
|
|
|
e->trials[1].price = price0( e->eb.bm_match[state][pos_state] );
|
2025-02-20 19:59:10 +01:00
|
|
|
if( St_is_char( state ) )
|
2025-02-20 20:52:54 +01:00
|
|
|
e->trials[1].price += LZeb_price_literal( &e->eb, prev_byte, cur_byte );
|
2025-02-20 19:59:10 +01:00
|
|
|
else
|
2025-02-20 20:52:54 +01:00
|
|
|
e->trials[1].price += LZeb_price_matched( &e->eb, prev_byte, cur_byte, match_byte );
|
2025-02-20 21:23:49 +01:00
|
|
|
e->trials[1].dis4 = -1; /* literal */
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:33:45 +01:00
|
|
|
const int match_price = price1( e->eb.bm_match[state][pos_state] );
|
|
|
|
const int rep_match_price = match_price + price1( e->eb.bm_rep[state] );
|
|
|
|
|
2025-02-20 19:59:10 +01:00
|
|
|
if( match_byte == cur_byte )
|
2025-02-20 20:21:13 +01:00
|
|
|
Tr_update( &e->trials[1], rep_match_price +
|
2025-02-20 20:52:54 +01:00
|
|
|
LZeb_price_shortrep( &e->eb, state, pos_state ), 0, 0 );
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 20:13:15 +01:00
|
|
|
num_trials = max( main_len, replens[rep_index] );
|
|
|
|
|
|
|
|
if( num_trials < min_match_len )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:21:13 +01:00
|
|
|
e->trials[0].price = 1;
|
2025-02-20 21:23:49 +01:00
|
|
|
e->trials[0].dis4 = e->trials[1].dis4;
|
2025-02-20 20:52:54 +01:00
|
|
|
if( !Mb_move_pos( &e->eb.mb ) ) return 0;
|
2025-02-20 19:59:10 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2025-02-20 21:21:14 +01:00
|
|
|
e->trials[0].state = state;
|
2025-02-20 20:13:15 +01:00
|
|
|
for( i = 0; i < num_rep_distances; ++i )
|
2025-02-20 20:21:13 +01:00
|
|
|
e->trials[0].reps[i] = reps[i];
|
2025-02-20 20:13:15 +01:00
|
|
|
|
|
|
|
for( len = min_match_len; len <= num_trials; ++len )
|
2025-02-20 20:21:13 +01:00
|
|
|
e->trials[len].price = infinite_price;
|
2025-02-20 20:13:15 +01:00
|
|
|
|
|
|
|
for( rep = 0; rep < num_rep_distances; ++rep )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:13:15 +01:00
|
|
|
if( replens[rep] < min_match_len ) continue;
|
2025-02-20 21:32:02 +01:00
|
|
|
const int price = rep_match_price + LZeb_price_rep( &e->eb, rep, state, pos_state );
|
2025-02-20 20:13:15 +01:00
|
|
|
for( len = min_match_len; len <= replens[rep]; ++len )
|
2025-02-20 20:21:13 +01:00
|
|
|
Tr_update( &e->trials[len], price +
|
2025-02-20 20:23:05 +01:00
|
|
|
Lp_price( &e->rep_len_prices, len, pos_state ), rep, 0 );
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
2025-02-20 20:13:15 +01:00
|
|
|
|
|
|
|
if( main_len > replens[0] )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
const int normal_match_price = match_price + price0( e->eb.bm_rep[state] );
|
2025-02-20 21:23:49 +01:00
|
|
|
int i = 0, len = max( replens[0] + 1, min_match_len );
|
2025-02-20 20:21:13 +01:00
|
|
|
while( len > e->pairs[i].len ) ++i;
|
2025-02-20 20:13:15 +01:00
|
|
|
while( true )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:21:13 +01:00
|
|
|
const int dis = e->pairs[i].dis;
|
|
|
|
Tr_update( &e->trials[len], normal_match_price +
|
|
|
|
LZe_price_pair( e, dis, len, pos_state ),
|
2025-02-20 20:13:15 +01:00
|
|
|
dis + num_rep_distances, 0 );
|
2025-02-20 20:21:13 +01:00
|
|
|
if( ++len > e->pairs[i].len && ++i >= num_pairs ) break;
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
int cur = 0;
|
2025-02-20 20:13:15 +01:00
|
|
|
while( true ) /* price optimization loop */
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
if( !Mb_move_pos( &e->eb.mb ) ) return 0;
|
2025-02-20 19:59:10 +01:00
|
|
|
if( ++cur >= num_trials ) /* no more initialized trials */
|
|
|
|
{
|
2025-02-20 20:21:13 +01:00
|
|
|
LZe_backward( e, cur );
|
2025-02-20 19:59:10 +01:00
|
|
|
return cur;
|
|
|
|
}
|
2025-02-20 20:13:15 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
const int num_pairs = LZe_read_match_distances( e );
|
2025-02-20 21:34:58 +01:00
|
|
|
const int newlen = (num_pairs > 0) ? e->pairs[num_pairs-1].len : 0;
|
2025-02-20 20:52:54 +01:00
|
|
|
if( newlen >= e->match_len_limit )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:21:13 +01:00
|
|
|
e->pending_num_pairs = num_pairs;
|
|
|
|
LZe_backward( e, cur );
|
2025-02-20 19:59:10 +01:00
|
|
|
return cur;
|
|
|
|
}
|
|
|
|
|
2025-02-20 20:13:15 +01:00
|
|
|
/* give final values to current trial */
|
2025-02-20 21:34:58 +01:00
|
|
|
Trial * cur_trial = &e->trials[cur];
|
2025-02-20 21:32:02 +01:00
|
|
|
State cur_state;
|
2025-02-20 20:21:13 +01:00
|
|
|
{
|
2025-02-20 21:23:49 +01:00
|
|
|
const int dis4 = cur_trial->dis4;
|
2025-02-20 20:21:13 +01:00
|
|
|
int prev_index = cur_trial->prev_index;
|
|
|
|
const int prev_index2 = cur_trial->prev_index2;
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 20:21:13 +01:00
|
|
|
if( prev_index2 == single_step_trial )
|
2025-02-20 20:13:15 +01:00
|
|
|
{
|
2025-02-20 20:21:13 +01:00
|
|
|
cur_state = e->trials[prev_index].state;
|
|
|
|
if( prev_index + 1 == cur ) /* len == 1 */
|
2025-02-20 20:13:15 +01:00
|
|
|
{
|
2025-02-20 21:35:19 +01:00
|
|
|
if( dis4 == 0 ) cur_state = St_set_shortrep( cur_state );
|
2025-02-20 20:21:13 +01:00
|
|
|
else cur_state = St_set_char( cur_state ); /* literal */
|
2025-02-20 20:13:15 +01:00
|
|
|
}
|
2025-02-20 21:23:49 +01:00
|
|
|
else if( dis4 < num_rep_distances ) cur_state = St_set_rep( cur_state );
|
2025-02-20 20:21:13 +01:00
|
|
|
else cur_state = St_set_match( cur_state );
|
2025-02-20 20:13:15 +01:00
|
|
|
}
|
2025-02-20 21:23:49 +01:00
|
|
|
else
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 21:23:49 +01:00
|
|
|
if( prev_index2 == dual_step_trial ) /* dis4 == 0 (rep0) */
|
|
|
|
--prev_index;
|
|
|
|
else /* prev_index2 >= 0 */
|
|
|
|
prev_index = prev_index2;
|
2025-02-20 21:28:08 +01:00
|
|
|
cur_state = St_set_char_rep();
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
2025-02-20 20:13:15 +01:00
|
|
|
cur_trial->state = cur_state;
|
2025-02-20 20:21:13 +01:00
|
|
|
for( i = 0; i < num_rep_distances; ++i )
|
|
|
|
cur_trial->reps[i] = e->trials[prev_index].reps[i];
|
2025-02-20 21:23:49 +01:00
|
|
|
mtf_reps( dis4, cur_trial->reps ); /* literal is ignored */
|
2025-02-20 20:21:13 +01:00
|
|
|
}
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
const int pos_state = Mb_data_position( &e->eb.mb ) & pos_state_mask;
|
|
|
|
const uint8_t prev_byte = Mb_peek( &e->eb.mb, 1 );
|
|
|
|
const uint8_t cur_byte = Mb_peek( &e->eb.mb, 0 );
|
|
|
|
const uint8_t match_byte = Mb_peek( &e->eb.mb, cur_trial->reps[0] + 1 );
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
int next_price = cur_trial->price +
|
|
|
|
price0( e->eb.bm_match[cur_state][pos_state] );
|
2025-02-20 20:13:15 +01:00
|
|
|
if( St_is_char( cur_state ) )
|
2025-02-20 20:52:54 +01:00
|
|
|
next_price += LZeb_price_literal( &e->eb, prev_byte, cur_byte );
|
2025-02-20 19:59:10 +01:00
|
|
|
else
|
2025-02-20 20:52:54 +01:00
|
|
|
next_price += LZeb_price_matched( &e->eb, prev_byte, cur_byte, match_byte );
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 20:13:15 +01:00
|
|
|
/* try last updates to next trial */
|
2025-02-20 21:34:58 +01:00
|
|
|
Trial * next_trial = &e->trials[cur+1];
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 20:21:13 +01:00
|
|
|
Tr_update( next_trial, next_price, -1, cur ); /* literal */
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
const int match_price = cur_trial->price + price1( e->eb.bm_match[cur_state][pos_state] );
|
|
|
|
const int rep_match_price = match_price + price1( e->eb.bm_rep[cur_state] );
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:23:49 +01:00
|
|
|
if( match_byte == cur_byte && next_trial->dis4 != 0 &&
|
2025-02-20 20:21:13 +01:00
|
|
|
next_trial->prev_index2 == single_step_trial )
|
2025-02-20 20:13:15 +01:00
|
|
|
{
|
|
|
|
const int price = rep_match_price +
|
2025-02-20 20:52:54 +01:00
|
|
|
LZeb_price_shortrep( &e->eb, cur_state, pos_state );
|
2025-02-20 20:13:15 +01:00
|
|
|
if( price <= next_trial->price )
|
|
|
|
{
|
|
|
|
next_trial->price = price;
|
2025-02-20 21:23:49 +01:00
|
|
|
next_trial->dis4 = 0; /* rep0 */
|
2025-02-20 20:13:15 +01:00
|
|
|
next_trial->prev_index = cur;
|
|
|
|
}
|
|
|
|
}
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
const int triable_bytes =
|
2025-02-20 20:52:54 +01:00
|
|
|
min( Mb_available_bytes( &e->eb.mb ), max_num_trials - 1 - cur );
|
|
|
|
if( triable_bytes < min_match_len ) continue;
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
const int len_limit = min( e->match_len_limit, triable_bytes );
|
2025-02-20 20:13:15 +01:00
|
|
|
|
|
|
|
/* try literal + rep0 */
|
|
|
|
if( match_byte != cur_byte && next_trial->prev_index != cur )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
const uint8_t * const data = Mb_ptr_to_current_pos( &e->eb.mb );
|
2025-02-20 20:13:15 +01:00
|
|
|
const int dis = cur_trial->reps[0] + 1;
|
2025-02-20 20:52:54 +01:00
|
|
|
const int limit = min( e->match_len_limit + 1, triable_bytes );
|
2025-02-20 21:23:49 +01:00
|
|
|
int len = 1;
|
2025-02-20 20:13:15 +01:00
|
|
|
while( len < limit && data[len-dis] == data[len] ) ++len;
|
|
|
|
if( --len >= min_match_len )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:13:15 +01:00
|
|
|
const int pos_state2 = ( pos_state + 1 ) & pos_state_mask;
|
|
|
|
const State state2 = St_set_char( cur_state );
|
|
|
|
const int price = next_price +
|
2025-02-20 21:23:49 +01:00
|
|
|
price1( e->eb.bm_match[state2][pos_state2] ) +
|
|
|
|
price1( e->eb.bm_rep[state2] ) +
|
|
|
|
LZe_price_rep0_len( e, len, state2, pos_state2 );
|
2025-02-20 20:13:15 +01:00
|
|
|
while( num_trials < cur + 1 + len )
|
2025-02-20 20:21:13 +01:00
|
|
|
e->trials[++num_trials].price = infinite_price;
|
|
|
|
Tr_update2( &e->trials[cur+1+len], price, cur + 1 );
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
int start_len = min_match_len;
|
|
|
|
|
2025-02-20 20:13:15 +01:00
|
|
|
/* try rep distances */
|
|
|
|
for( rep = 0; rep < num_rep_distances; ++rep )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
const uint8_t * const data = Mb_ptr_to_current_pos( &e->eb.mb );
|
2025-02-20 20:13:15 +01:00
|
|
|
const int dis = cur_trial->reps[rep] + 1;
|
|
|
|
|
2025-02-20 20:21:13 +01:00
|
|
|
if( data[0-dis] != data[0] || data[1-dis] != data[1] ) continue;
|
2025-02-20 20:13:15 +01:00
|
|
|
for( len = min_match_len; len < len_limit; ++len )
|
|
|
|
if( data[len-dis] != data[len] ) break;
|
|
|
|
while( num_trials < cur + len )
|
2025-02-20 20:21:13 +01:00
|
|
|
e->trials[++num_trials].price = infinite_price;
|
2025-02-20 21:32:02 +01:00
|
|
|
int price = rep_match_price + LZeb_price_rep( &e->eb, rep, cur_state, pos_state );
|
2025-02-20 20:13:15 +01:00
|
|
|
for( i = min_match_len; i <= len; ++i )
|
2025-02-20 20:21:13 +01:00
|
|
|
Tr_update( &e->trials[cur+i], price +
|
2025-02-20 20:23:05 +01:00
|
|
|
Lp_price( &e->rep_len_prices, i, pos_state ), rep, cur );
|
2025-02-20 20:13:15 +01:00
|
|
|
|
|
|
|
if( rep == 0 ) start_len = len + 1; /* discard shorter matches */
|
|
|
|
|
|
|
|
/* try rep + literal + rep0 */
|
2025-02-20 20:52:54 +01:00
|
|
|
int len2 = len + 1;
|
|
|
|
const int limit = min( e->match_len_limit + len2, triable_bytes );
|
2025-02-20 20:13:15 +01:00
|
|
|
while( len2 < limit && data[len2-dis] == data[len2] ) ++len2;
|
|
|
|
len2 -= len + 1;
|
|
|
|
if( len2 < min_match_len ) continue;
|
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
int pos_state2 = ( pos_state + len ) & pos_state_mask;
|
|
|
|
State state2 = St_set_rep( cur_state );
|
2025-02-20 20:23:05 +01:00
|
|
|
price += Lp_price( &e->rep_len_prices, len, pos_state ) +
|
2025-02-20 20:52:54 +01:00
|
|
|
price0( e->eb.bm_match[state2][pos_state2] ) +
|
|
|
|
LZeb_price_matched( &e->eb, data[len-1], data[len], data[len-dis] );
|
2025-02-20 20:13:15 +01:00
|
|
|
pos_state2 = ( pos_state2 + 1 ) & pos_state_mask;
|
|
|
|
state2 = St_set_char( state2 );
|
2025-02-20 20:52:54 +01:00
|
|
|
price += price1( e->eb.bm_match[state2][pos_state2] ) +
|
|
|
|
price1( e->eb.bm_rep[state2] ) +
|
2025-02-20 20:21:13 +01:00
|
|
|
LZe_price_rep0_len( e, len2, state2, pos_state2 );
|
2025-02-20 20:13:15 +01:00
|
|
|
while( num_trials < cur + len + 1 + len2 )
|
2025-02-20 20:21:13 +01:00
|
|
|
e->trials[++num_trials].price = infinite_price;
|
|
|
|
Tr_update3( &e->trials[cur+len+1+len2], price, rep, cur + len + 1, cur );
|
2025-02-20 20:13:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* try matches */
|
|
|
|
if( newlen >= start_len && newlen <= len_limit )
|
|
|
|
{
|
2025-02-20 19:59:10 +01:00
|
|
|
const int normal_match_price = match_price +
|
2025-02-20 20:52:54 +01:00
|
|
|
price0( e->eb.bm_rep[cur_state] );
|
2025-02-20 19:59:10 +01:00
|
|
|
|
|
|
|
while( num_trials < cur + newlen )
|
2025-02-20 20:21:13 +01:00
|
|
|
e->trials[++num_trials].price = infinite_price;
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
int i = 0;
|
2025-02-20 20:54:52 +01:00
|
|
|
while( e->pairs[i].len < start_len ) ++i;
|
2025-02-20 21:32:02 +01:00
|
|
|
int dis = e->pairs[i].dis;
|
2025-02-20 20:13:15 +01:00
|
|
|
for( len = start_len; ; ++len )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:21:13 +01:00
|
|
|
int price = normal_match_price + LZe_price_pair( e, dis, len, pos_state );
|
|
|
|
Tr_update( &e->trials[cur+len], price, dis + num_rep_distances, cur );
|
2025-02-20 20:13:15 +01:00
|
|
|
|
|
|
|
/* try match + literal + rep0 */
|
2025-02-20 20:21:13 +01:00
|
|
|
if( len == e->pairs[i].len )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
const uint8_t * const data = Mb_ptr_to_current_pos( &e->eb.mb );
|
2025-02-20 20:13:15 +01:00
|
|
|
const int dis2 = dis + 1;
|
|
|
|
int len2 = len + 1;
|
2025-02-20 20:52:54 +01:00
|
|
|
const int limit = min( e->match_len_limit + len2, triable_bytes );
|
2025-02-20 20:13:15 +01:00
|
|
|
while( len2 < limit && data[len2-dis2] == data[len2] ) ++len2;
|
|
|
|
len2 -= len + 1;
|
|
|
|
if( len2 >= min_match_len )
|
|
|
|
{
|
|
|
|
int pos_state2 = ( pos_state + len ) & pos_state_mask;
|
|
|
|
State state2 = St_set_match( cur_state );
|
2025-02-20 20:52:54 +01:00
|
|
|
price += price0( e->eb.bm_match[state2][pos_state2] ) +
|
|
|
|
LZeb_price_matched( &e->eb, data[len-1], data[len], data[len-dis2] );
|
2025-02-20 20:13:15 +01:00
|
|
|
pos_state2 = ( pos_state2 + 1 ) & pos_state_mask;
|
|
|
|
state2 = St_set_char( state2 );
|
2025-02-20 20:52:54 +01:00
|
|
|
price += price1( e->eb.bm_match[state2][pos_state2] ) +
|
|
|
|
price1( e->eb.bm_rep[state2] ) +
|
2025-02-20 20:21:13 +01:00
|
|
|
LZe_price_rep0_len( e, len2, state2, pos_state2 );
|
2025-02-20 20:13:15 +01:00
|
|
|
|
|
|
|
while( num_trials < cur + len + 1 + len2 )
|
2025-02-20 20:21:13 +01:00
|
|
|
e->trials[++num_trials].price = infinite_price;
|
|
|
|
Tr_update3( &e->trials[cur+len+1+len2], price,
|
|
|
|
dis + num_rep_distances, cur + len + 1, cur );
|
2025-02-20 20:13:15 +01:00
|
|
|
}
|
|
|
|
if( ++i >= num_pairs ) break;
|
2025-02-20 20:21:13 +01:00
|
|
|
dis = e->pairs[i].dis;
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2025-02-20 21:34:58 +01:00
|
|
|
static bool LZe_encode_member( LZ_encoder * const e )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 21:34:58 +01:00
|
|
|
const bool best = e->match_len_limit > 12;
|
2025-02-20 20:23:05 +01:00
|
|
|
const int dis_price_count = best ? 1 : 512;
|
|
|
|
const int align_price_count = best ? 1 : dis_align_size;
|
2025-02-20 21:34:58 +01:00
|
|
|
const int price_count = (e->match_len_limit > 36) ? 1013 : 4093;
|
2025-02-20 21:32:02 +01:00
|
|
|
int i;
|
2025-02-20 20:52:54 +01:00
|
|
|
State * const state = &e->eb.state;
|
2025-02-20 20:18:02 +01:00
|
|
|
|
2025-02-20 20:52:54 +01:00
|
|
|
if( e->eb.member_finished ) return true;
|
|
|
|
if( Re_member_position( &e->eb.renc ) >= e->eb.member_size_limit )
|
2025-02-20 21:23:49 +01:00
|
|
|
{ LZeb_try_full_flush( &e->eb ); return true; }
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 20:52:54 +01:00
|
|
|
if( Mb_data_position( &e->eb.mb ) == 0 &&
|
|
|
|
!Mb_data_finished( &e->eb.mb ) ) /* encode first byte */
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
if( !Mb_enough_available_bytes( &e->eb.mb ) ||
|
|
|
|
!Re_enough_free_bytes( &e->eb.renc ) ) return true;
|
2025-02-20 21:32:02 +01:00
|
|
|
const uint8_t prev_byte = 0;
|
|
|
|
const uint8_t cur_byte = Mb_peek( &e->eb.mb, 0 );
|
2025-02-20 20:52:54 +01:00
|
|
|
Re_encode_bit( &e->eb.renc, &e->eb.bm_match[*state][0], 0 );
|
|
|
|
LZeb_encode_literal( &e->eb, prev_byte, cur_byte );
|
|
|
|
CRC32_update_byte( &e->eb.crc, cur_byte );
|
|
|
|
LZe_get_match_pairs( e, 0 );
|
|
|
|
if( !Mb_move_pos( &e->eb.mb ) ) return false;
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
|
2025-02-20 20:52:54 +01:00
|
|
|
while( !Mb_data_finished( &e->eb.mb ) )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
if( !Mb_enough_available_bytes( &e->eb.mb ) ||
|
|
|
|
!Re_enough_free_bytes( &e->eb.renc ) ) return true;
|
2025-02-20 20:23:05 +01:00
|
|
|
if( e->price_counter <= 0 && e->pending_num_pairs == 0 )
|
2025-02-20 20:13:15 +01:00
|
|
|
{
|
2025-02-20 20:23:05 +01:00
|
|
|
e->price_counter = price_count; /* recalculate prices every these bytes */
|
|
|
|
if( e->dis_price_counter <= 0 )
|
|
|
|
{ e->dis_price_counter = dis_price_count; LZe_update_distance_prices( e ); }
|
|
|
|
if( e->align_price_counter <= 0 )
|
|
|
|
{
|
|
|
|
e->align_price_counter = align_price_count;
|
|
|
|
for( i = 0; i < dis_align_size; ++i )
|
2025-02-20 20:52:54 +01:00
|
|
|
e->align_prices[i] = price_symbol_reversed( e->eb.bm_align, i, dis_align_bits );
|
2025-02-20 20:23:05 +01:00
|
|
|
}
|
|
|
|
Lp_update_prices( &e->match_len_prices );
|
|
|
|
Lp_update_prices( &e->rep_len_prices );
|
2025-02-20 20:13:15 +01:00
|
|
|
}
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:32:02 +01:00
|
|
|
int ahead = LZe_sequence_optimizer( e, e->eb.reps, *state );
|
2025-02-20 20:23:05 +01:00
|
|
|
e->price_counter -= ahead;
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 20:21:13 +01:00
|
|
|
for( i = 0; ahead > 0; )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
|
|
|
const int pos_state =
|
2025-02-20 20:52:54 +01:00
|
|
|
( Mb_data_position( &e->eb.mb ) - ahead ) & pos_state_mask;
|
2025-02-20 20:21:13 +01:00
|
|
|
const int len = e->trials[i].price;
|
2025-02-20 21:23:49 +01:00
|
|
|
int dis = e->trials[i].dis4;
|
2025-02-20 19:59:10 +01:00
|
|
|
|
2025-02-20 21:34:58 +01:00
|
|
|
bool bit = dis < 0;
|
2025-02-20 20:52:54 +01:00
|
|
|
Re_encode_bit( &e->eb.renc, &e->eb.bm_match[*state][pos_state], !bit );
|
2025-02-20 19:59:10 +01:00
|
|
|
if( bit ) /* literal byte */
|
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
const uint8_t prev_byte = Mb_peek( &e->eb.mb, ahead + 1 );
|
|
|
|
const uint8_t cur_byte = Mb_peek( &e->eb.mb, ahead );
|
|
|
|
CRC32_update_byte( &e->eb.crc, cur_byte );
|
2025-02-20 21:32:02 +01:00
|
|
|
if( ( *state = St_set_char( *state ) ) < 4 )
|
2025-02-20 20:52:54 +01:00
|
|
|
LZeb_encode_literal( &e->eb, prev_byte, cur_byte );
|
2025-02-20 19:59:10 +01:00
|
|
|
else
|
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
const uint8_t match_byte = Mb_peek( &e->eb.mb, ahead + e->eb.reps[0] + 1 );
|
|
|
|
LZeb_encode_matched( &e->eb, prev_byte, cur_byte, match_byte );
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
}
|
2025-02-20 20:18:02 +01:00
|
|
|
else /* match or repeated match */
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
CRC32_update_buf( &e->eb.crc, Mb_ptr_to_current_pos( &e->eb.mb ) - ahead, len );
|
|
|
|
mtf_reps( dis, e->eb.reps );
|
2025-02-20 21:34:58 +01:00
|
|
|
bit = dis < num_rep_distances;
|
2025-02-20 20:52:54 +01:00
|
|
|
Re_encode_bit( &e->eb.renc, &e->eb.bm_rep[*state], bit );
|
2025-02-20 20:21:13 +01:00
|
|
|
if( bit ) /* repeated match */
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 21:34:58 +01:00
|
|
|
bit = dis == 0;
|
2025-02-20 20:52:54 +01:00
|
|
|
Re_encode_bit( &e->eb.renc, &e->eb.bm_rep0[*state], !bit );
|
2025-02-20 19:59:10 +01:00
|
|
|
if( bit )
|
2025-02-20 20:52:54 +01:00
|
|
|
Re_encode_bit( &e->eb.renc, &e->eb.bm_len[*state][pos_state], len > 1 );
|
2025-02-20 19:59:10 +01:00
|
|
|
else
|
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
Re_encode_bit( &e->eb.renc, &e->eb.bm_rep1[*state], dis > 1 );
|
2025-02-20 19:59:10 +01:00
|
|
|
if( dis > 1 )
|
2025-02-20 20:52:54 +01:00
|
|
|
Re_encode_bit( &e->eb.renc, &e->eb.bm_rep2[*state], dis > 2 );
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
2025-02-20 21:35:19 +01:00
|
|
|
if( len == 1 ) *state = St_set_shortrep( *state );
|
2025-02-20 19:59:10 +01:00
|
|
|
else
|
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
Re_encode_len( &e->eb.renc, &e->eb.rep_len_model, len, pos_state );
|
2025-02-20 20:23:05 +01:00
|
|
|
Lp_decrement_counter( &e->rep_len_prices, pos_state );
|
2025-02-20 20:13:15 +01:00
|
|
|
*state = St_set_rep( *state );
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
}
|
2025-02-20 20:21:13 +01:00
|
|
|
else /* match */
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 21:23:49 +01:00
|
|
|
dis -= num_rep_distances;
|
|
|
|
LZeb_encode_pair( &e->eb, dis, len, pos_state );
|
|
|
|
if( dis >= modeled_distances ) --e->align_price_counter;
|
2025-02-20 20:23:05 +01:00
|
|
|
--e->dis_price_counter;
|
|
|
|
Lp_decrement_counter( &e->match_len_prices, pos_state );
|
2025-02-20 20:13:15 +01:00
|
|
|
*state = St_set_match( *state );
|
2025-02-20 19:59:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ahead -= len; i += len;
|
2025-02-20 20:52:54 +01:00
|
|
|
if( Re_member_position( &e->eb.renc ) >= e->eb.member_size_limit )
|
2025-02-20 19:59:10 +01:00
|
|
|
{
|
2025-02-20 20:52:54 +01:00
|
|
|
if( !Mb_dec_pos( &e->eb.mb, ahead ) ) return false;
|
2025-02-20 21:23:49 +01:00
|
|
|
LZeb_try_full_flush( &e->eb );
|
2025-02-20 19:59:10 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2025-02-20 21:23:49 +01:00
|
|
|
LZeb_try_full_flush( &e->eb );
|
2025-02-20 19:59:10 +01:00
|
|
|
return true;
|
|
|
|
}
|