bearing 0.1.0-alpha.5

A Rust port of Apache Lucene
Documentation
// SPDX-License-Identifier: Apache-2.0

//! ZigZag encoding/decoding for variable-length integers.
//!
//! Maps signed integers to unsigned so that small-magnitude values
//! (positive or negative) produce small encoded values, making them compact
//! when followed by variable-length encoding (VInt/VLong).
//!
//! Mapping: 0 -> 0, -1 -> 1, 1 -> 2, -2 -> 3, 2 -> 4, ...

/// Encodes a 32-bit integer.
pub fn encode_i32(i: i32) -> i32 {
    (i >> 31) ^ (i << 1)
}

/// Decodes a 32-bit integer.
pub fn decode_i32(i: i32) -> i32 {
    ((i as u32) >> 1) as i32 ^ -(i & 1)
}

/// Encodes a 64-bit integer.
pub fn encode_i64(i: i64) -> i64 {
    (i >> 63) ^ (i << 1)
}

/// Decodes a 64-bit integer.
pub fn decode_i64(i: i64) -> i64 {
    ((i as u64) >> 1) as i64 ^ -(i & 1)
}

#[cfg(test)]
mod tests {
    use super::*;

    // Ported from org.apache.lucene.util.TestBitUtil

    #[test]
    fn test_encode_i32() {
        assert_eq!(encode_i32(0), 0);
        assert_eq!(encode_i32(-1), 1);
        assert_eq!(encode_i32(1), 2);
        assert_eq!(encode_i32(-2), 3);
        assert_eq!(encode_i32(2), 4);
        assert_eq!(encode_i32(i32::MIN), -1); // 0xFFFFFFFF
        assert_eq!(encode_i32(i32::MAX), -2); // 0xFFFFFFFE
    }

    #[test]
    fn test_roundtrip_i32() {
        for &v in &[0, 1, -1, 127, -128, i32::MIN, i32::MAX, 42, -42] {
            assert_eq!(decode_i32(encode_i32(v)), v);
        }
    }

    #[test]
    fn test_encode_i64() {
        assert_eq!(encode_i64(0), 0);
        assert_eq!(encode_i64(-1), 1);
        assert_eq!(encode_i64(1), 2);
        assert_eq!(encode_i64(-2), 3);
        assert_eq!(encode_i64(i64::MIN), -1);
        assert_eq!(encode_i64(i64::MAX), -2);
    }

    #[test]
    fn test_roundtrip_i64() {
        for &v in &[0i64, 1, -1, 127, -128, i64::MIN, i64::MAX, 42, -42] {
            assert_eq!(decode_i64(encode_i64(v)), v);
        }
    }
}