blob: ca29d99c93eba46a4e2d3ed94ced08cf217f75b8 [file] [log] [blame] [edit]
// Copyright 2006 The open-vcdiff Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
// Implementation of the Bentley/McIlroy algorithm for finding differences.
// Bentley, McIlroy. DCC 1999. Data Compression Using Long Common Strings.
// http://citeseer.ist.psu.edu/555557.html
#ifndef OPEN_VCDIFF_BLOCKHASH_H_
#define OPEN_VCDIFF_BLOCKHASH_H_
#include <config.h>
#include <stddef.h> // size_t
#include <stdint.h> // uint32_t
#include <vector>
namespace open_vcdiff {
// A generic hash table which will be used to keep track of byte runs
// of size kBlockSize in both the incrementally processed target data
// and the preprocessed source dictionary.
//
// A custom hash table implementation is used instead of the standard
// hash_map template because we know that there will be exactly one
// entry in the BlockHash corresponding to each kBlockSize bytes
// in the source data, which makes certain optimizations possible:
// * The memory for the hash table and for all hash entries can be allocated
// in one step rather than incrementally for each insert operation.
// * A single integer can be used to represent both
// the index of the next hash entry in the chain
// and the position of the entry within the source data
// (== kBlockSize * block_number). This greatly reduces the size
// of a hash entry.
//
class BlockHash {
public:
// Block size as per Bentley/McIlroy; must be a power of two.
//
// Using (for example) kBlockSize = 4 guarantees that no match smaller
// than size 4 will be identified, that some matches having sizes
// 4, 5, or 6 may be identified, and that all matches
// having size 7 or greater will be identified (because any string of
// 7 bytes must contain a complete aligned block of 4 bytes.)
//
// Increasing kBlockSize by a factor of two will halve the amount of
// memory needed for the next block table, and will halve the setup time
// for a new BlockHash. However, it also doubles the minimum
// match length that is guaranteed to be found in FindBestMatch(),
// so that function will be less effective in finding matches.
//
// Computational effort in FindBestMatch (which is the inner loop of
// the encoding algorithm) will be proportional to the number of
// matches found, and a low value of kBlockSize will waste time
// tracking down small matches. On the other hand, if this value
// is set too high, no matches will be found at all.
//
// It is suggested that different values of kBlockSize be tried against
// a representative data set to find the best tradeoff between
// memory/CPU and the effectiveness of FindBestMatch().
//
// If you change kBlockSize to a smaller value, please increase
// kMaxMatchesToCheck accordingly.
static const int kBlockSize = 16;
// This class is used to store the best match found by FindBestMatch()
// and return it to the caller.
class Match {
public:
Match() : size_(0), source_offset_(-1), target_offset_(-1) { }
void ReplaceIfBetterMatch(size_t candidate_size,
int candidate_source_offset,
int candidate_target_offset) {
if (candidate_size > size_) {
size_ = candidate_size;
source_offset_ = candidate_source_offset;
target_offset_ = candidate_target_offset;
}
}
size_t size() const { return size_; }
int source_offset() const { return source_offset_; }
int target_offset() const { return target_offset_; }
private:
// The size of the best (longest) match passed to ReplaceIfBetterMatch().
size_t size_;
// The source offset of the match, including the starting_offset_
// of the BlockHash for which the match was found.
int source_offset_;
// The target offset of the match. An offset of 0 corresponds to the
// data at target_start, which is an argument of FindBestMatch().
int target_offset_;
// Making these private avoids implicit copy constructor
// & assignment operator
Match(const Match&); // NOLINT
void operator=(const Match&);
};
// A BlockHash is created using a buffer of source data. The hash table
// will contain one entry for each kBlockSize-byte block in the
// source data.
//
// See the comments for starting_offset_, below, for a description of
// the starting_offset argument. For a hash of source (dictionary) data,
// starting_offset_ will be zero; for a hash of previously encoded
// target data, starting_offset_ will be equal to the dictionary size.
//
BlockHash(const char* source_data, size_t source_size, int starting_offset);
~BlockHash();
// Initializes the object before use.
// This method must be called after constructing a BlockHash object,
// and before any other method may be called. This is because
// Init() dynamically allocates hash_table_ and next_block_table_.
// Returns true if initialization succeeded, or false if an error occurred,
// in which case no other method except the destructor may then be used
// on the object.
//
// If populate_hash_table is true, then AddAllBlocks() will be called
// to populate the hash table. If populate_hash_table is false, then
// classes that inherit from BlockHash are expected to call AddBlock()
// to incrementally populate individual blocks of data.
//
bool Init(bool populate_hash_table);
// In the context of the open-vcdiff encoder, BlockHash is used for two
// purposes: to hash the source (dictionary) data, and to hash
// the previously encoded target data. The main differences between
// a dictionary BlockHash and a target BlockHash are as follows:
//
// 1. The best_match->source_offset() returned from FindBestMatch()
// for a target BlockHash is computed in the following manner:
// the starting offset of the first byte in the target data
// is equal to the dictionary size. FindBestMatch() will add
// starting_offset_ to any best_match->source_offset() value it returns,
// in order to produce the correct offset value for a target BlockHash.
// 2. For a dictionary BlockHash, the entire data set is hashed at once
// when Init() is called with the parameter populate_hash_table = true.
// For a target BlockHash, because the previously encoded target data
// includes only the data seen up to the current encoding position,
// the data blocks are hashed incrementally as the encoding position
// advances, using AddOneIndexHash() and AddAllBlocksThroughIndex().
//
// The following two factory functions can be used to create BlockHash
// objects for each of these two purposes. Each factory function calls
// the object constructor and also calls Init(). If an error occurs,
// NULL is returned; otherwise a valid BlockHash object is returned.
// Since a dictionary BlockHash is not expected to be modified after
// initialization, a const object is returned.
// The caller is responsible for deleting the returned object
// (using the C++ delete operator) once it is no longer needed.
static const BlockHash* CreateDictionaryHash(const char* dictionary_data,
size_t dictionary_size);
static BlockHash* CreateTargetHash(const char* target_data,
size_t target_size,
size_t dictionary_size);
// This function will be called to add blocks incrementally to the target hash
// as the encoding position advances through the target data. It will be
// called for every kBlockSize-byte block in the target data, regardless
// of whether the block is aligned evenly on a block boundary. The
// BlockHash will only store hash entries for the evenly-aligned blocks.
//
void AddOneIndexHash(int index, uint32_t hash_value) {
if (index == NextIndexToAdd()) {
AddBlock(hash_value);
}
}
// Calls AddBlock() for each kBlockSize-byte block in the range
// (last_block_added_ * kBlockSize, end_index), exclusive of the endpoints.
// If end_index <= the last index added (last_block_added_ * kBlockSize),
// this function does nothing.
//
// A partial block beginning anywhere up to (end_index - 1) is also added,
// unless it extends outside the end of the source data. Like AddAllBlocks(),
// this function computes the hash value for each of the blocks in question
// from scratch, so it is not a good option if the hash values have already
// been computed for some other purpose.
//
// Example: assume kBlockSize = 4, last_block_added_ = 1, and there are
// 14 bytes of source data.
// If AddAllBlocksThroughIndex(9) is invoked, then it will call AddBlock()
// only for block number 2 (at index 8).
// If, after that, AddAllBlocksThroughIndex(14) is invoked, it will not call
// AddBlock() at all, because block 3 (beginning at index 12) would
// fall outside the range of source data.
//
// VCDiffEngine::Encode (in vcdiffengine.cc) uses this function to
// add a whole range of data to a target hash when a COPY instruction
// is generated.
void AddAllBlocksThroughIndex(int end_index);
// FindBestMatch takes a position within the unencoded target data
// (target_candidate_start) and the hash value of the kBlockSize bytes
// beginning at that position (hash_value). It attempts to find a matching
// set of bytes within the source (== dictionary) data, expanding
// the match both below and above the target block. It cannot expand
// the match outside the bounds of the source data, or below
// target_start within the target data, or past
// the end limit of (target_start + target_length).
//
// target_candidate_start is the start of the candidate block within the
// target data for which a match will be sought, while
// target_start (which is <= target_candidate_start)
// is the start of the target data that has yet to be encoded.
//
// If a match is found whose size is greater than the size
// of best_match, this function populates *best_match with the
// size, source_offset, and target_offset of the match found.
// best_match->source_offset() will contain the index of the start of the
// matching source data, plus starting_offset_
// (see description of starting_offset_ for details);
// best_match->target_offset() will contain the offset of the match
// beginning with target_start = offset 0, such that
// 0 <= best_match->target_offset()
// <= (target_candidate_start - target_start);
// and best_match->size() will contain the size of the match.
// If no such match is found, this function leaves *best_match unmodified.
//
// On calling FindBestMatch(), best_match must
// point to a valid Match object, and cannot be NULL.
// The same Match object can be passed
// when calling FindBestMatch() on a different BlockHash object
// for the same candidate data block, in order to find
// the best match possible across both objects. For example:
//
// open_vcdiff::BlockHash::Match best_match;
// uint32_t hash_value =
// RollingHash<BlockHash::kBlockSize>::Hash(target_candidate_start);
// bh1.FindBestMatch(hash_value,
// target_candidate_start,
// target_start,
// target_length,
// &best_match);
// bh2.FindBestMatch(hash_value,
// target_candidate_start,
// target_start,
// target_length,
// &best_match);
// if (best_size >= 0) {
// // a match was found; its size, source offset, and target offset
// // can be found in best_match
// }
//
// hash_value is passed as a separate parameter from target_candidate_start,
// (rather than calculated within FindBestMatch) in order to take
// advantage of the rolling hash, which quickly calculates the hash value
// of the block starting at target_candidate_start based on
// the known hash value of the block starting at (target_candidate_start - 1).
// See vcdiffengine.cc for more details.
//
// Example:
// kBlockSize: 4
// target text: "ANDREW LLOYD WEBBER"
// 1^ 5^2^ 3^
// dictionary: "INSURANCE : LLOYDS OF LONDON"
// 4^
// hashed dictionary blocks:
// "INSU", "RANC", "E : ", "LLOY", "DS O", "F LON"
//
// 1: target_start (beginning of unencoded data)
// 2: target_candidate_start (for the block "LLOY")
// 3: target_length (points one byte beyond the last byte of data.)
// 4: best_match->source_offset() (after calling FindBestMatch)
// 5: best_match->target_offset() (after calling FindBestMatch)
//
// Under these conditions, FindBestMatch will find a matching
// hashed dictionary block for "LLOY", and will extend the beginning of
// this match backwards by one byte, and the end of the match forwards
// by one byte, finding that the best match is " LLOYD"
// with best_match->source_offset() = 10
// (offset of " LLOYD" in the source string),
// best_match->target_offset() = 6
// (offset of " LLOYD" in the target string),
// and best_match->size() = 6.
//
void FindBestMatch(uint32_t hash_value,
const char* target_candidate_start,
const char* target_start,
size_t target_size,
Match* best_match) const;
protected:
// FindBestMatch() will not process more than this number
// of matching hash entries.
//
// It is necessary to have a limit on the maximum number of matches
// that will be checked in order to avoid the worst-case performance
// possible if, for example, all the blocks in the dictionary have
// the same hash value. See the unit test SearchStringFindsTooManyMatches
// for an example of such a case. The encoder uses a loop in
// VCDiffEngine::Encode over each target byte, containing a loop in
// BlockHash::FindBestMatch over the number of matches (up to a maximum
// of the number of source blocks), containing two loops that extend
// the match forwards and backwards up to the number of source bytes.
// Total complexity in the worst case is
// O([target size] * source_size_ * source_size_)
// Placing a limit on the possible number of matches checked changes this to
// O([target size] * source_size_ * kMaxMatchesToCheck)
//
// In empirical testing on real HTML text, using a block size of 4,
// the number of true matches per call to FindBestMatch() did not exceed 78;
// with a block size of 32, the number of matches did not exceed 3.
//
// The expected number of true matches scales super-linearly
// with the inverse of kBlockSize, but here a linear scale is used
// for block sizes smaller than 32.
static const int kMaxMatchesToCheck = (kBlockSize >= 32) ? 32 :
(32 * (32 / kBlockSize));
// Do not skip more than this number of non-matching hash collisions
// to find the next matching entry in the hash chain.
static const int kMaxProbes = 16;
// Internal routine which calculates a hash table size based on kBlockSize and
// the dictionary_size. Will return a power of two if successful, or 0 if an
// internal error occurs. Some calculations (such as GetHashTableIndex())
// depend on the table size being a power of two.
static size_t CalcTableSize(const size_t dictionary_size);
size_t GetNumberOfBlocks() const {
return source_size_ / kBlockSize;
}
// Use the lowest-order bits of the hash value
// as the index into the hash table.
uint32_t GetHashTableIndex(uint32_t hash_value) const {
return hash_value & hash_table_mask_;
}
// The index within source_data_ of the next block
// for which AddBlock() should be called.
int NextIndexToAdd() const {
return (last_block_added_ + 1) * kBlockSize;
}
static inline bool TooManyMatches(int* match_counter);
const char* source_data() { return source_data_; }
size_t source_size() { return source_size_; }
// Adds an entry to the hash table for one block of source data of length
// kBlockSize, starting at source_data_[block_number * kBlockSize],
// where block_number is always (last_block_added_ + 1). That is,
// AddBlock() must be called once for each block in source_data_
// in increasing order.
void AddBlock(uint32_t hash_value);
// Calls AddBlock() for each complete kBlockSize-byte block between
// source_data_ and (source_data_ + source_size_). It is equivalent
// to calling AddAllBlocksThroughIndex(source_data + source_size).
// This function is called when Init(true) is invoked.
void AddAllBlocks();
// Returns true if the contents of the kBlockSize-byte block
// beginning at block1 are identical to the contents of
// the block beginning at block2; false otherwise.
static bool BlockContentsMatch(const char* block1, const char* block2);
// Compares each machine word of the two (possibly unaligned) blocks, rather
// than each byte, thus reducing the number of test-and-branch instructions
// executed. Returns a boolean (do the blocks match?) rather than
// the signed byte difference returned by memcmp.
//
// BlockContentsMatch will use either this function or memcmp to do its work,
// depending on which is faster for a particular architecture.
//
// For gcc on x86-based architectures, this function has been shown to run
// about twice as fast as the library function memcmp(), and between five and
// nine times faster than the assembly instructions (repz and cmpsb) that gcc
// uses by default for builtin memcmp. On other architectures, or using
// other compilers, this function has not shown to be faster than memcmp.
static bool BlockCompareWords(const char* block1, const char* block2);
// Finds the first block number within the hashed data
// that represents a match for the given hash value.
// Returns -1 if no match was found.
//
// Init() must have been called and returned true before using
// FirstMatchingBlock or NextMatchingBlock. No check is performed
// for this condition; the code will crash if this condition is violated.
//
// The hash table is initially populated with -1 (not found) values,
// so if this function is called before the hash table has been populated
// using AddAllBlocks() or AddBlock(), it will simply return -1
// for any value of hash_value.
int FirstMatchingBlock(uint32_t hash_value, const char* block_ptr) const;
// Given a block number returned by FirstMatchingBlock()
// or by a previous call to NextMatchingBlock(), returns
// the next block number that matches the same hash value.
// Returns -1 if no match was found.
int NextMatchingBlock(int block_number, const char* block_ptr) const;
// Inline version of FirstMatchingBlock. This saves the cost of a function
// call when this routine is called from within the module. The external
// (non-inlined) version is called only by unit tests.
inline int FirstMatchingBlockInline(uint32_t hash_value,
const char* block_ptr) const;
// Walk through the hash entry chain, skipping over any false matches
// (for which the lowest bits of the fingerprints match,
// but the actual block data does not.) Returns the block number of
// the first true match found, or -1 if no true match was found.
// If block_number is a matching block, the function will return block_number
// without skipping to the next block.
int SkipNonMatchingBlocks(int block_number, const char* block_ptr) const;
// Returns the number of bytes to the left of source_match_start
// that match the corresponding bytes to the left of target_match_start.
// Will not examine more than max_bytes bytes, which is to say that
// the return value will be in the range [0, max_bytes] inclusive.
static int MatchingBytesToLeft(const char* source_match_start,
const char* target_match_start,
int max_bytes);
// Returns the number of bytes starting at source_match_end
// that match the corresponding bytes starting at target_match_end.
// Will not examine more than max_bytes bytes, which is to say that
// the return value will be in the range [0, max_bytes] inclusive.
static int MatchingBytesToRight(const char* source_match_end,
const char* target_match_end,
int max_bytes);
// The protected functions BlockContentsMatch, FirstMatchingBlock,
// NextMatchingBlock, MatchingBytesToLeft, and MatchingBytesToRight
// should be made accessible to unit tests.
friend class BlockHashTest;
private:
const char* const source_data_;
const size_t source_size_;
// The size of this array is determined using CalcTableSize(). It has at
// least one element for each kBlockSize-byte block in the source data.
// GetHashTableIndex() returns an index into this table for a given hash
// value. The value of each element of hash_table_ is the lowest block
// number in the source data whose hash value would return the same value from
// GetHashTableIndex(), or -1 if there is no matching block. This value can
// then be used as an index into next_block_table_ to retrieve the entire set
// of matching block numbers.
std::vector<int> hash_table_;
// An array containing one element for each source block. Each element is
// either -1 (== not found) or the index of the next block whose hash value
// would produce a matching result from GetHashTableIndex().
std::vector<int> next_block_table_;
// This vector has the same size as next_block_table_. For every block number
// B that is referenced in hash_table_, last_block_table_[B] will contain
// the maximum block number that has the same GetHashTableIndex() value
// as block B. This number may be B itself. For a block number B' that
// is not referenced in hash_table_, the value of last_block_table_[B'] is -1.
// This table is used only while populating the hash table, not while looking
// up hash values in the table. Keeping track of the last block number in the
// chain allows us to construct the block chains as FIFO rather than LIFO
// lists, so that the match with the lowest index is returned first. This
// should result in a more compact encoding because the VCDIFF format favors
// smaller index values and repeated index values.
std::vector<int> last_block_table_;
// Performing a bitwise AND with hash_table_mask_ will produce a value ranging
// from 0 to the number of elements in hash_table_.
uint32_t hash_table_mask_;
// The offset of the first byte of source data (the data at source_data_[0]).
// For the purpose of computing offsets, the source data and target data
// are considered to be concatenated -- not literally in a single memory
// buffer, but conceptually as described in the RFC.
// The first byte of the previously encoded target data
// has an offset that is equal to dictionary_size, i.e., just after
// the last byte of source data.
// For a hash of source (dictionary) data, starting_offset_ will be zero;
// for a hash of previously encoded target data, starting_offset_ will be
// equal to the dictionary size.
const int starting_offset_;
// The last index added by AddBlock(). This determines the block number
// for successive calls to AddBlock(), and is also
// used to determine the starting block for AddAllBlocksThroughIndex().
int last_block_added_;
// Making these private avoids implicit copy constructor & assignment operator
BlockHash(const BlockHash&); // NOLINT
void operator=(const BlockHash&);
};
} // namespace open_vcdiff
#endif // OPEN_VCDIFF_BLOCKHASH_H_