Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
574 lines
21 KiB
574 lines
21 KiB
package brotli |
|
|
|
import ( |
|
"sync" |
|
) |
|
|
|
/* Copyright 2014 Google Inc. All Rights Reserved. |
|
|
|
Distributed under MIT license. |
|
See file LICENSE for detail or copy at https://opensource.org/licenses/MIT |
|
*/ |
|
|
|
/* Algorithms for distributing the literals and commands of a metablock between |
|
block types and contexts. */ |
|
|
|
type metaBlockSplit struct { |
|
literal_split blockSplit |
|
command_split blockSplit |
|
distance_split blockSplit |
|
literal_context_map []uint32 |
|
literal_context_map_size uint |
|
distance_context_map []uint32 |
|
distance_context_map_size uint |
|
literal_histograms []histogramLiteral |
|
literal_histograms_size uint |
|
command_histograms []histogramCommand |
|
command_histograms_size uint |
|
distance_histograms []histogramDistance |
|
distance_histograms_size uint |
|
} |
|
|
|
var metaBlockPool sync.Pool |
|
|
|
func getMetaBlockSplit() *metaBlockSplit { |
|
mb, _ := metaBlockPool.Get().(*metaBlockSplit) |
|
|
|
if mb == nil { |
|
mb = &metaBlockSplit{} |
|
} else { |
|
initBlockSplit(&mb.literal_split) |
|
initBlockSplit(&mb.command_split) |
|
initBlockSplit(&mb.distance_split) |
|
mb.literal_context_map = mb.literal_context_map[:0] |
|
mb.literal_context_map_size = 0 |
|
mb.distance_context_map = mb.distance_context_map[:0] |
|
mb.distance_context_map_size = 0 |
|
mb.literal_histograms = mb.literal_histograms[:0] |
|
mb.command_histograms = mb.command_histograms[:0] |
|
mb.distance_histograms = mb.distance_histograms[:0] |
|
} |
|
return mb |
|
} |
|
|
|
func freeMetaBlockSplit(mb *metaBlockSplit) { |
|
metaBlockPool.Put(mb) |
|
} |
|
|
|
func initDistanceParams(params *encoderParams, npostfix uint32, ndirect uint32) { |
|
var dist_params *distanceParams = ¶ms.dist |
|
var alphabet_size uint32 |
|
var max_distance uint32 |
|
|
|
dist_params.distance_postfix_bits = npostfix |
|
dist_params.num_direct_distance_codes = ndirect |
|
|
|
alphabet_size = uint32(distanceAlphabetSize(uint(npostfix), uint(ndirect), maxDistanceBits)) |
|
max_distance = ndirect + (1 << (maxDistanceBits + npostfix + 2)) - (1 << (npostfix + 2)) |
|
|
|
if params.large_window { |
|
var bound = [maxNpostfix + 1]uint32{0, 4, 12, 28} |
|
var postfix uint32 = 1 << npostfix |
|
alphabet_size = uint32(distanceAlphabetSize(uint(npostfix), uint(ndirect), largeMaxDistanceBits)) |
|
|
|
/* The maximum distance is set so that no distance symbol used can encode |
|
a distance larger than BROTLI_MAX_ALLOWED_DISTANCE with all |
|
its extra bits set. */ |
|
if ndirect < bound[npostfix] { |
|
max_distance = maxAllowedDistance - (bound[npostfix] - ndirect) |
|
} else if ndirect >= bound[npostfix]+postfix { |
|
max_distance = (3 << 29) - 4 + (ndirect - bound[npostfix]) |
|
} else { |
|
max_distance = maxAllowedDistance |
|
} |
|
} |
|
|
|
dist_params.alphabet_size = alphabet_size |
|
dist_params.max_distance = uint(max_distance) |
|
} |
|
|
|
func recomputeDistancePrefixes(cmds []command, orig_params *distanceParams, new_params *distanceParams) { |
|
if orig_params.distance_postfix_bits == new_params.distance_postfix_bits && orig_params.num_direct_distance_codes == new_params.num_direct_distance_codes { |
|
return |
|
} |
|
|
|
for i := range cmds { |
|
var cmd *command = &cmds[i] |
|
if commandCopyLen(cmd) != 0 && cmd.cmd_prefix_ >= 128 { |
|
prefixEncodeCopyDistance(uint(commandRestoreDistanceCode(cmd, orig_params)), uint(new_params.num_direct_distance_codes), uint(new_params.distance_postfix_bits), &cmd.dist_prefix_, &cmd.dist_extra_) |
|
} |
|
} |
|
} |
|
|
|
func computeDistanceCost(cmds []command, orig_params *distanceParams, new_params *distanceParams, cost *float64) bool { |
|
var equal_params bool = false |
|
var dist_prefix uint16 |
|
var dist_extra uint32 |
|
var extra_bits float64 = 0.0 |
|
var histo histogramDistance |
|
histogramClearDistance(&histo) |
|
|
|
if orig_params.distance_postfix_bits == new_params.distance_postfix_bits && orig_params.num_direct_distance_codes == new_params.num_direct_distance_codes { |
|
equal_params = true |
|
} |
|
|
|
for i := range cmds { |
|
cmd := &cmds[i] |
|
if commandCopyLen(cmd) != 0 && cmd.cmd_prefix_ >= 128 { |
|
if equal_params { |
|
dist_prefix = cmd.dist_prefix_ |
|
} else { |
|
var distance uint32 = commandRestoreDistanceCode(cmd, orig_params) |
|
if distance > uint32(new_params.max_distance) { |
|
return false |
|
} |
|
|
|
prefixEncodeCopyDistance(uint(distance), uint(new_params.num_direct_distance_codes), uint(new_params.distance_postfix_bits), &dist_prefix, &dist_extra) |
|
} |
|
|
|
histogramAddDistance(&histo, uint(dist_prefix)&0x3FF) |
|
extra_bits += float64(dist_prefix >> 10) |
|
} |
|
} |
|
|
|
*cost = populationCostDistance(&histo) + extra_bits |
|
return true |
|
} |
|
|
|
var buildMetaBlock_kMaxNumberOfHistograms uint = 256 |
|
|
|
func buildMetaBlock(ringbuffer []byte, pos uint, mask uint, params *encoderParams, prev_byte byte, prev_byte2 byte, cmds []command, literal_context_mode int, mb *metaBlockSplit) { |
|
var distance_histograms []histogramDistance |
|
var literal_histograms []histogramLiteral |
|
var literal_context_modes []int = nil |
|
var literal_histograms_size uint |
|
var distance_histograms_size uint |
|
var i uint |
|
var literal_context_multiplier uint = 1 |
|
var npostfix uint32 |
|
var ndirect_msb uint32 = 0 |
|
var check_orig bool = true |
|
var best_dist_cost float64 = 1e99 |
|
var orig_params encoderParams = *params |
|
/* Histogram ids need to fit in one byte. */ |
|
|
|
var new_params encoderParams = *params |
|
|
|
for npostfix = 0; npostfix <= maxNpostfix; npostfix++ { |
|
for ; ndirect_msb < 16; ndirect_msb++ { |
|
var ndirect uint32 = ndirect_msb << npostfix |
|
var skip bool |
|
var dist_cost float64 |
|
initDistanceParams(&new_params, npostfix, ndirect) |
|
if npostfix == orig_params.dist.distance_postfix_bits && ndirect == orig_params.dist.num_direct_distance_codes { |
|
check_orig = false |
|
} |
|
|
|
skip = !computeDistanceCost(cmds, &orig_params.dist, &new_params.dist, &dist_cost) |
|
if skip || (dist_cost > best_dist_cost) { |
|
break |
|
} |
|
|
|
best_dist_cost = dist_cost |
|
params.dist = new_params.dist |
|
} |
|
|
|
if ndirect_msb > 0 { |
|
ndirect_msb-- |
|
} |
|
ndirect_msb /= 2 |
|
} |
|
|
|
if check_orig { |
|
var dist_cost float64 |
|
computeDistanceCost(cmds, &orig_params.dist, &orig_params.dist, &dist_cost) |
|
if dist_cost < best_dist_cost { |
|
/* NB: currently unused; uncomment when more param tuning is added. */ |
|
/* best_dist_cost = dist_cost; */ |
|
params.dist = orig_params.dist |
|
} |
|
} |
|
|
|
recomputeDistancePrefixes(cmds, &orig_params.dist, ¶ms.dist) |
|
|
|
splitBlock(cmds, ringbuffer, pos, mask, params, &mb.literal_split, &mb.command_split, &mb.distance_split) |
|
|
|
if !params.disable_literal_context_modeling { |
|
literal_context_multiplier = 1 << literalContextBits |
|
literal_context_modes = make([]int, (mb.literal_split.num_types)) |
|
for i = 0; i < mb.literal_split.num_types; i++ { |
|
literal_context_modes[i] = literal_context_mode |
|
} |
|
} |
|
|
|
literal_histograms_size = mb.literal_split.num_types * literal_context_multiplier |
|
literal_histograms = make([]histogramLiteral, literal_histograms_size) |
|
clearHistogramsLiteral(literal_histograms, literal_histograms_size) |
|
|
|
distance_histograms_size = mb.distance_split.num_types << distanceContextBits |
|
distance_histograms = make([]histogramDistance, distance_histograms_size) |
|
clearHistogramsDistance(distance_histograms, distance_histograms_size) |
|
|
|
mb.command_histograms_size = mb.command_split.num_types |
|
if cap(mb.command_histograms) < int(mb.command_histograms_size) { |
|
mb.command_histograms = make([]histogramCommand, (mb.command_histograms_size)) |
|
} else { |
|
mb.command_histograms = mb.command_histograms[:mb.command_histograms_size] |
|
} |
|
clearHistogramsCommand(mb.command_histograms, mb.command_histograms_size) |
|
|
|
buildHistogramsWithContext(cmds, &mb.literal_split, &mb.command_split, &mb.distance_split, ringbuffer, pos, mask, prev_byte, prev_byte2, literal_context_modes, literal_histograms, mb.command_histograms, distance_histograms) |
|
literal_context_modes = nil |
|
|
|
mb.literal_context_map_size = mb.literal_split.num_types << literalContextBits |
|
if cap(mb.literal_context_map) < int(mb.literal_context_map_size) { |
|
mb.literal_context_map = make([]uint32, (mb.literal_context_map_size)) |
|
} else { |
|
mb.literal_context_map = mb.literal_context_map[:mb.literal_context_map_size] |
|
} |
|
|
|
mb.literal_histograms_size = mb.literal_context_map_size |
|
if cap(mb.literal_histograms) < int(mb.literal_histograms_size) { |
|
mb.literal_histograms = make([]histogramLiteral, (mb.literal_histograms_size)) |
|
} else { |
|
mb.literal_histograms = mb.literal_histograms[:mb.literal_histograms_size] |
|
} |
|
|
|
clusterHistogramsLiteral(literal_histograms, literal_histograms_size, buildMetaBlock_kMaxNumberOfHistograms, mb.literal_histograms, &mb.literal_histograms_size, mb.literal_context_map) |
|
literal_histograms = nil |
|
|
|
if params.disable_literal_context_modeling { |
|
/* Distribute assignment to all contexts. */ |
|
for i = mb.literal_split.num_types; i != 0; { |
|
var j uint = 0 |
|
i-- |
|
for ; j < 1<<literalContextBits; j++ { |
|
mb.literal_context_map[(i<<literalContextBits)+j] = mb.literal_context_map[i] |
|
} |
|
} |
|
} |
|
|
|
mb.distance_context_map_size = mb.distance_split.num_types << distanceContextBits |
|
if cap(mb.distance_context_map) < int(mb.distance_context_map_size) { |
|
mb.distance_context_map = make([]uint32, (mb.distance_context_map_size)) |
|
} else { |
|
mb.distance_context_map = mb.distance_context_map[:mb.distance_context_map_size] |
|
} |
|
|
|
mb.distance_histograms_size = mb.distance_context_map_size |
|
if cap(mb.distance_histograms) < int(mb.distance_histograms_size) { |
|
mb.distance_histograms = make([]histogramDistance, (mb.distance_histograms_size)) |
|
} else { |
|
mb.distance_histograms = mb.distance_histograms[:mb.distance_histograms_size] |
|
} |
|
|
|
clusterHistogramsDistance(distance_histograms, mb.distance_context_map_size, buildMetaBlock_kMaxNumberOfHistograms, mb.distance_histograms, &mb.distance_histograms_size, mb.distance_context_map) |
|
distance_histograms = nil |
|
} |
|
|
|
const maxStaticContexts = 13 |
|
|
|
/* Greedy block splitter for one block category (literal, command or distance). |
|
Gathers histograms for all context buckets. */ |
|
type contextBlockSplitter struct { |
|
alphabet_size_ uint |
|
num_contexts_ uint |
|
max_block_types_ uint |
|
min_block_size_ uint |
|
split_threshold_ float64 |
|
num_blocks_ uint |
|
split_ *blockSplit |
|
histograms_ []histogramLiteral |
|
histograms_size_ *uint |
|
target_block_size_ uint |
|
block_size_ uint |
|
curr_histogram_ix_ uint |
|
last_histogram_ix_ [2]uint |
|
last_entropy_ [2 * maxStaticContexts]float64 |
|
merge_last_count_ uint |
|
} |
|
|
|
func initContextBlockSplitter(self *contextBlockSplitter, alphabet_size uint, num_contexts uint, min_block_size uint, split_threshold float64, num_symbols uint, split *blockSplit, histograms *[]histogramLiteral, histograms_size *uint) { |
|
var max_num_blocks uint = num_symbols/min_block_size + 1 |
|
var max_num_types uint |
|
assert(num_contexts <= maxStaticContexts) |
|
|
|
self.alphabet_size_ = alphabet_size |
|
self.num_contexts_ = num_contexts |
|
self.max_block_types_ = maxNumberOfBlockTypes / num_contexts |
|
self.min_block_size_ = min_block_size |
|
self.split_threshold_ = split_threshold |
|
self.num_blocks_ = 0 |
|
self.split_ = split |
|
self.histograms_size_ = histograms_size |
|
self.target_block_size_ = min_block_size |
|
self.block_size_ = 0 |
|
self.curr_histogram_ix_ = 0 |
|
self.merge_last_count_ = 0 |
|
|
|
/* We have to allocate one more histogram than the maximum number of block |
|
types for the current histogram when the meta-block is too big. */ |
|
max_num_types = brotli_min_size_t(max_num_blocks, self.max_block_types_+1) |
|
|
|
brotli_ensure_capacity_uint8_t(&split.types, &split.types_alloc_size, max_num_blocks) |
|
brotli_ensure_capacity_uint32_t(&split.lengths, &split.lengths_alloc_size, max_num_blocks) |
|
split.num_blocks = max_num_blocks |
|
*histograms_size = max_num_types * num_contexts |
|
if histograms == nil || cap(*histograms) < int(*histograms_size) { |
|
*histograms = make([]histogramLiteral, (*histograms_size)) |
|
} else { |
|
*histograms = (*histograms)[:*histograms_size] |
|
} |
|
self.histograms_ = *histograms |
|
|
|
/* Clear only current histogram. */ |
|
clearHistogramsLiteral(self.histograms_[0:], num_contexts) |
|
|
|
self.last_histogram_ix_[1] = 0 |
|
self.last_histogram_ix_[0] = self.last_histogram_ix_[1] |
|
} |
|
|
|
/* Does either of three things: |
|
(1) emits the current block with a new block type; |
|
(2) emits the current block with the type of the second last block; |
|
(3) merges the current block with the last block. */ |
|
func contextBlockSplitterFinishBlock(self *contextBlockSplitter, is_final bool) { |
|
var split *blockSplit = self.split_ |
|
var num_contexts uint = self.num_contexts_ |
|
var last_entropy []float64 = self.last_entropy_[:] |
|
var histograms []histogramLiteral = self.histograms_ |
|
|
|
if self.block_size_ < self.min_block_size_ { |
|
self.block_size_ = self.min_block_size_ |
|
} |
|
|
|
if self.num_blocks_ == 0 { |
|
var i uint |
|
|
|
/* Create first block. */ |
|
split.lengths[0] = uint32(self.block_size_) |
|
|
|
split.types[0] = 0 |
|
|
|
for i = 0; i < num_contexts; i++ { |
|
last_entropy[i] = bitsEntropy(histograms[i].data_[:], self.alphabet_size_) |
|
last_entropy[num_contexts+i] = last_entropy[i] |
|
} |
|
|
|
self.num_blocks_++ |
|
split.num_types++ |
|
self.curr_histogram_ix_ += num_contexts |
|
if self.curr_histogram_ix_ < *self.histograms_size_ { |
|
clearHistogramsLiteral(self.histograms_[self.curr_histogram_ix_:], self.num_contexts_) |
|
} |
|
|
|
self.block_size_ = 0 |
|
} else if self.block_size_ > 0 { |
|
var entropy [maxStaticContexts]float64 |
|
var combined_histo []histogramLiteral = make([]histogramLiteral, (2 * num_contexts)) |
|
var combined_entropy [2 * maxStaticContexts]float64 |
|
var diff = [2]float64{0.0} |
|
/* Try merging the set of histograms for the current block type with the |
|
respective set of histograms for the last and second last block types. |
|
Decide over the split based on the total reduction of entropy across |
|
all contexts. */ |
|
|
|
var i uint |
|
for i = 0; i < num_contexts; i++ { |
|
var curr_histo_ix uint = self.curr_histogram_ix_ + i |
|
var j uint |
|
entropy[i] = bitsEntropy(histograms[curr_histo_ix].data_[:], self.alphabet_size_) |
|
for j = 0; j < 2; j++ { |
|
var jx uint = j*num_contexts + i |
|
var last_histogram_ix uint = self.last_histogram_ix_[j] + i |
|
combined_histo[jx] = histograms[curr_histo_ix] |
|
histogramAddHistogramLiteral(&combined_histo[jx], &histograms[last_histogram_ix]) |
|
combined_entropy[jx] = bitsEntropy(combined_histo[jx].data_[0:], self.alphabet_size_) |
|
diff[j] += combined_entropy[jx] - entropy[i] - last_entropy[jx] |
|
} |
|
} |
|
|
|
if split.num_types < self.max_block_types_ && diff[0] > self.split_threshold_ && diff[1] > self.split_threshold_ { |
|
/* Create new block. */ |
|
split.lengths[self.num_blocks_] = uint32(self.block_size_) |
|
|
|
split.types[self.num_blocks_] = byte(split.num_types) |
|
self.last_histogram_ix_[1] = self.last_histogram_ix_[0] |
|
self.last_histogram_ix_[0] = split.num_types * num_contexts |
|
for i = 0; i < num_contexts; i++ { |
|
last_entropy[num_contexts+i] = last_entropy[i] |
|
last_entropy[i] = entropy[i] |
|
} |
|
|
|
self.num_blocks_++ |
|
split.num_types++ |
|
self.curr_histogram_ix_ += num_contexts |
|
if self.curr_histogram_ix_ < *self.histograms_size_ { |
|
clearHistogramsLiteral(self.histograms_[self.curr_histogram_ix_:], self.num_contexts_) |
|
} |
|
|
|
self.block_size_ = 0 |
|
self.merge_last_count_ = 0 |
|
self.target_block_size_ = self.min_block_size_ |
|
} else if diff[1] < diff[0]-20.0 { |
|
split.lengths[self.num_blocks_] = uint32(self.block_size_) |
|
split.types[self.num_blocks_] = split.types[self.num_blocks_-2] |
|
/* Combine this block with second last block. */ |
|
|
|
var tmp uint = self.last_histogram_ix_[0] |
|
self.last_histogram_ix_[0] = self.last_histogram_ix_[1] |
|
self.last_histogram_ix_[1] = tmp |
|
for i = 0; i < num_contexts; i++ { |
|
histograms[self.last_histogram_ix_[0]+i] = combined_histo[num_contexts+i] |
|
last_entropy[num_contexts+i] = last_entropy[i] |
|
last_entropy[i] = combined_entropy[num_contexts+i] |
|
histogramClearLiteral(&histograms[self.curr_histogram_ix_+i]) |
|
} |
|
|
|
self.num_blocks_++ |
|
self.block_size_ = 0 |
|
self.merge_last_count_ = 0 |
|
self.target_block_size_ = self.min_block_size_ |
|
} else { |
|
/* Combine this block with last block. */ |
|
split.lengths[self.num_blocks_-1] += uint32(self.block_size_) |
|
|
|
for i = 0; i < num_contexts; i++ { |
|
histograms[self.last_histogram_ix_[0]+i] = combined_histo[i] |
|
last_entropy[i] = combined_entropy[i] |
|
if split.num_types == 1 { |
|
last_entropy[num_contexts+i] = last_entropy[i] |
|
} |
|
|
|
histogramClearLiteral(&histograms[self.curr_histogram_ix_+i]) |
|
} |
|
|
|
self.block_size_ = 0 |
|
self.merge_last_count_++ |
|
if self.merge_last_count_ > 1 { |
|
self.target_block_size_ += self.min_block_size_ |
|
} |
|
} |
|
|
|
combined_histo = nil |
|
} |
|
|
|
if is_final { |
|
*self.histograms_size_ = split.num_types * num_contexts |
|
split.num_blocks = self.num_blocks_ |
|
} |
|
} |
|
|
|
/* Adds the next symbol to the current block type and context. When the |
|
current block reaches the target size, decides on merging the block. */ |
|
func contextBlockSplitterAddSymbol(self *contextBlockSplitter, symbol uint, context uint) { |
|
histogramAddLiteral(&self.histograms_[self.curr_histogram_ix_+context], symbol) |
|
self.block_size_++ |
|
if self.block_size_ == self.target_block_size_ { |
|
contextBlockSplitterFinishBlock(self, false) /* is_final = */ |
|
} |
|
} |
|
|
|
func mapStaticContexts(num_contexts uint, static_context_map []uint32, mb *metaBlockSplit) { |
|
var i uint |
|
mb.literal_context_map_size = mb.literal_split.num_types << literalContextBits |
|
if cap(mb.literal_context_map) < int(mb.literal_context_map_size) { |
|
mb.literal_context_map = make([]uint32, (mb.literal_context_map_size)) |
|
} else { |
|
mb.literal_context_map = mb.literal_context_map[:mb.literal_context_map_size] |
|
} |
|
|
|
for i = 0; i < mb.literal_split.num_types; i++ { |
|
var offset uint32 = uint32(i * num_contexts) |
|
var j uint |
|
for j = 0; j < 1<<literalContextBits; j++ { |
|
mb.literal_context_map[(i<<literalContextBits)+j] = offset + static_context_map[j] |
|
} |
|
} |
|
} |
|
|
|
func buildMetaBlockGreedyInternal(ringbuffer []byte, pos uint, mask uint, prev_byte byte, prev_byte2 byte, literal_context_lut contextLUT, num_contexts uint, static_context_map []uint32, commands []command, mb *metaBlockSplit) { |
|
var lit_blocks struct { |
|
plain blockSplitterLiteral |
|
ctx contextBlockSplitter |
|
} |
|
var cmd_blocks blockSplitterCommand |
|
var dist_blocks blockSplitterDistance |
|
var num_literals uint = 0 |
|
for i := range commands { |
|
num_literals += uint(commands[i].insert_len_) |
|
} |
|
|
|
if num_contexts == 1 { |
|
initBlockSplitterLiteral(&lit_blocks.plain, 256, 512, 400.0, num_literals, &mb.literal_split, &mb.literal_histograms, &mb.literal_histograms_size) |
|
} else { |
|
initContextBlockSplitter(&lit_blocks.ctx, 256, num_contexts, 512, 400.0, num_literals, &mb.literal_split, &mb.literal_histograms, &mb.literal_histograms_size) |
|
} |
|
|
|
initBlockSplitterCommand(&cmd_blocks, numCommandSymbols, 1024, 500.0, uint(len(commands)), &mb.command_split, &mb.command_histograms, &mb.command_histograms_size) |
|
initBlockSplitterDistance(&dist_blocks, 64, 512, 100.0, uint(len(commands)), &mb.distance_split, &mb.distance_histograms, &mb.distance_histograms_size) |
|
|
|
for _, cmd := range commands { |
|
var j uint |
|
blockSplitterAddSymbolCommand(&cmd_blocks, uint(cmd.cmd_prefix_)) |
|
for j = uint(cmd.insert_len_); j != 0; j-- { |
|
var literal byte = ringbuffer[pos&mask] |
|
if num_contexts == 1 { |
|
blockSplitterAddSymbolLiteral(&lit_blocks.plain, uint(literal)) |
|
} else { |
|
var context uint = uint(getContext(prev_byte, prev_byte2, literal_context_lut)) |
|
contextBlockSplitterAddSymbol(&lit_blocks.ctx, uint(literal), uint(static_context_map[context])) |
|
} |
|
|
|
prev_byte2 = prev_byte |
|
prev_byte = literal |
|
pos++ |
|
} |
|
|
|
pos += uint(commandCopyLen(&cmd)) |
|
if commandCopyLen(&cmd) != 0 { |
|
prev_byte2 = ringbuffer[(pos-2)&mask] |
|
prev_byte = ringbuffer[(pos-1)&mask] |
|
if cmd.cmd_prefix_ >= 128 { |
|
blockSplitterAddSymbolDistance(&dist_blocks, uint(cmd.dist_prefix_)&0x3FF) |
|
} |
|
} |
|
} |
|
|
|
if num_contexts == 1 { |
|
blockSplitterFinishBlockLiteral(&lit_blocks.plain, true) /* is_final = */ |
|
} else { |
|
contextBlockSplitterFinishBlock(&lit_blocks.ctx, true) /* is_final = */ |
|
} |
|
|
|
blockSplitterFinishBlockCommand(&cmd_blocks, true) /* is_final = */ |
|
blockSplitterFinishBlockDistance(&dist_blocks, true) /* is_final = */ |
|
|
|
if num_contexts > 1 { |
|
mapStaticContexts(num_contexts, static_context_map, mb) |
|
} |
|
} |
|
|
|
func buildMetaBlockGreedy(ringbuffer []byte, pos uint, mask uint, prev_byte byte, prev_byte2 byte, literal_context_lut contextLUT, num_contexts uint, static_context_map []uint32, commands []command, mb *metaBlockSplit) { |
|
if num_contexts == 1 { |
|
buildMetaBlockGreedyInternal(ringbuffer, pos, mask, prev_byte, prev_byte2, literal_context_lut, 1, nil, commands, mb) |
|
} else { |
|
buildMetaBlockGreedyInternal(ringbuffer, pos, mask, prev_byte, prev_byte2, literal_context_lut, num_contexts, static_context_map, commands, mb) |
|
} |
|
} |
|
|
|
func optimizeHistograms(num_distance_codes uint32, mb *metaBlockSplit) { |
|
var good_for_rle [numCommandSymbols]byte |
|
var i uint |
|
for i = 0; i < mb.literal_histograms_size; i++ { |
|
optimizeHuffmanCountsForRLE(256, mb.literal_histograms[i].data_[:], good_for_rle[:]) |
|
} |
|
|
|
for i = 0; i < mb.command_histograms_size; i++ { |
|
optimizeHuffmanCountsForRLE(numCommandSymbols, mb.command_histograms[i].data_[:], good_for_rle[:]) |
|
} |
|
|
|
for i = 0; i < mb.distance_histograms_size; i++ { |
|
optimizeHuffmanCountsForRLE(uint(num_distance_codes), mb.distance_histograms[i].data_[:], good_for_rle[:]) |
|
} |
|
}
|
|
|