2 stable releases

1.1.0 Aug 13, 2024
1.0.0 May 29, 2024

#2120 in Encoding

Download history 30/week @ 2024-08-26 4/week @ 2024-09-02 24/week @ 2024-09-16 16/week @ 2024-09-23 22/week @ 2024-09-30 12/week @ 2024-10-07 5/week @ 2024-10-14 32/week @ 2024-10-21 18/week @ 2024-10-28 19/week @ 2024-11-04 8/week @ 2024-11-11 15/week @ 2024-11-18 9/week @ 2024-11-25 8/week @ 2024-12-02 26/week @ 2024-12-09

59 downloads per month

0BSD license

34KB
627 lines

vu128: Efficient variable-length integers

vu128 is a variable-length integer encoding, with smaller values being encoded using fewer bytes. Integer sizes up to 128 bits are supported. The compression ratio of vu128 equals or exceeds the widely used VLQ and LEB128 encodings, and is faster on modern pipelined architectures.

Encoding details

Values in the range [0, 2^7) are encoded as a single byte with the same bits as the original value.

Values in the range [2^7, 2^28) are encoded as a unary length prefix, followed by (length*7) bits, in little-endian order. This is conceptually similar to LEB128, but the continuation bits are placed in upper half of the initial byte. This arrangement is also known as a "prefix varint".

MSB ------------------ LSB

      10101011110011011110  Input value (0xABCDE)
   0101010 1111001 1011110  Zero-padded to a multiple of 7 bits
01010101 11100110 ___11110  Grouped into octets, with 3 continuation bits
01010101 11100110 11011110  Continuation bits `110` added
    0x55     0xE6     0xDE  In hexadecimal

        [0xDE, 0xE6, 0x55]  Encoded output (order is little-endian)

Values in the range [2^28, 2^128) are encoded as a binary length prefix, followed by payload bytes, in little-endian order. To differentiate this format from the format of smaller values, the top 4 bits of the first byte are set. The length prefix value is the number of payload bytes minus one; equivalently it is the total length of the encoded value minus two.

MSB ------------------------------------ LSB

               10010001101000101011001111000  Input value (0x12345678)
         00010010 00110100 01010110 01111000  Zero-padded to a multiple of 8 bits
00010010 00110100 01010110 01111000 11110011  Prefix byte is `0xF0 | (4 - 1)`
    0x12     0x34     0x56     0x78     0xF3  In hexadecimal

              [0xF3, 0x78, 0x56, 0x34, 0x12]  Encoded output (order is little-endian)

Handling of over-long encodings

The vu128 format permits over-long encodings, which encode a value using a byte sequence that is unnecessarily long:

  • Zero-padding beyond that required to reach a multiple of 7 or 8 bits.
  • Using a length prefix byte for a value in the range [0, 2^7).
  • Using a binary length prefix byte for a value in the range [0, 2^28).

The encode_* functions in this module will not generate such over-long encodings, but the decode_* functions will accept them. This is intended to allow vu128 values to be placed in a buffer before the value to be written is known. Applications that require a single canonical encoding for any given value should perform appropriate checking in their own code.

Signed integers and floating-point values

Signed integers and IEEE-754 floating-point values may be encoded with vu128 by mapping them to unsigned integers. It is recommended that the mapping functions be chosen so as to minimize the number of zeroes in the higher-order bits, which enables better compression.

This library includes helper functions that use Protocol Buffer's "ZigZag" encoding for signed integers and reverse-endian layout for floating-point.

No runtime deps