rars-crypto 0.3.1

RAR legacy and modern archive encryption primitives.
Documentation
use rars_crc32::{crc32_raw, table_entry as crc32_table_entry};
use zeroize::ZeroizeOnDrop;

#[derive(ZeroizeOnDrop)]
pub struct Rar15Cipher {
    key: [u16; 4],
}

impl Rar15Cipher {
    pub fn new(password: &[u8]) -> Self {
        let password = password.split(|&byte| byte == 0).next().unwrap_or(password);
        let password_crc = crc32_raw(password);
        let mut key = [
            (password_crc & 0xffff) as u16,
            (password_crc >> 16) as u16,
            0,
            0,
        ];

        for &byte in password {
            let crc = crc32_table_entry(byte);
            key[2] ^= u16::from(byte) ^ crc as u16;
            key[3] = key[3].wrapping_add(u16::from(byte).wrapping_add((crc >> 16) as u16));
        }

        Self { key }
    }

    pub fn crypt_in_place(&mut self, data: &mut [u8]) {
        for byte in data {
            *byte ^= self.next_mask();
        }
    }

    fn next_mask(&mut self) -> u8 {
        self.key[0] = self.key[0].wrapping_add(0x1234);
        let crc = crc32_table_entry(((self.key[0] & 0x01fe) >> 1) as u8);
        self.key[1] ^= crc as u16;
        self.key[2] = self.key[2].wrapping_sub((crc >> 16) as u16);
        self.key[0] ^= self.key[2];
        self.key[3] = self.key[3].rotate_right(1) ^ self.key[1];
        self.key[3] = self.key[3].rotate_right(1);
        self.key[0] ^= self.key[3];
        (self.key[0] >> 8) as u8
    }
}

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

    #[test]
    fn crypts_known_rar15_stream_vector() {
        let mut data = *b"hello world";
        Rar15Cipher::new(b"password").crypt_in_place(&mut data);
        assert_eq!(
            data,
            [0x2b, 0xb9, 0xf3, 0x9c, 0x41, 0xa6, 0xaa, 0xe7, 0x1a, 0x7a, 0x7b]
        );

        Rar15Cipher::new(b"password").crypt_in_place(&mut data);
        assert_eq!(&data, b"hello world");
    }

    #[test]
    fn password_is_legacy_c_string() {
        let mut truncated = *b"hello world";
        let mut plain = *b"hello world";

        Rar15Cipher::new(b"password\0ignored").crypt_in_place(&mut truncated);
        Rar15Cipher::new(b"password").crypt_in_place(&mut plain);

        assert_eq!(truncated, plain);
    }
}