argon2rs 0.2.5

The pure Rust password hashing library that runs on Argon2.
Documentation
use octword::u64x2;
use std::ops::{BitXorAssign, Index, IndexMut};
use std::mem;
use std::slice::{Iter, IterMut};

pub const ARGON2_BLOCK_BYTES: usize = 1024;

macro_rules! per_kib {
    (u8) => { ARGON2_BLOCK_BYTES };
    (u64) => { ARGON2_BLOCK_BYTES / 8 };
    (u64x2) => { ARGON2_BLOCK_BYTES / 16 };
}

pub struct Block([u64x2; per_kib!(u64x2)]);

impl Clone for Block {
    #[inline(always)]
    fn clone(&self) -> Self {
        let inner = self.0;
        Block(inner)
    }
}

impl Block {
    pub fn iter_mut(&mut self) -> IterMut<u64x2> { self.0.iter_mut() }

    pub fn iter(&self) -> Iter<u64x2> { self.0.iter() }
}

impl<'a> BitXorAssign<&'a Block> for Block {
    #[inline(always)]
    fn bitxor_assign(&mut self, rhs: &Block) {
        for (d, r) in self.0.iter_mut().zip(rhs.0.iter()) {
            *d = *d ^ *r;
        }
    }
}

impl<'a, 'b> BitXorAssign<(&'a Block, &'b Block)> for Block {
    #[inline(always)]
    fn bitxor_assign(&mut self, (a, b): (&Block, &Block)) {
        for (d, (l, r)) in self.0.iter_mut().zip(a.0.iter().zip(b.0.iter())) {
            *d = *d ^ *l ^ *r;
        }
    }
}

impl Index<usize> for Block {
    type Output = u64x2;
    #[inline(always)]
    fn index(&self, idx: usize) -> &Self::Output { &self.0[idx] }
}

impl IndexMut<usize> for Block {
    #[inline(always)]
    fn index_mut(&mut self, idx: usize) -> &mut u64x2 { &mut self.0[idx] }
}

pub fn zero() -> Block { Block([u64x2(0, 0); per_kib!(u64x2)]) }

pub fn as_u8_mut(b: &mut Block) -> &mut [u8] {
    let rv: &mut [u8; per_kib!(u8)] = unsafe { mem::transmute(&mut b.0) };
    rv
}

pub fn as_u8(b: &Block) -> &[u8] {
    let rv: &[u8; per_kib!(u8)] = unsafe { mem::transmute(&b.0) };
    rv
}

pub fn as_u64(b: &Block) -> &[u64] {
    let rv: &[u64; per_kib!(u64)] = unsafe { mem::transmute(&b.0) };
    rv
}

pub struct Matrix(Vec<Vec<Block>>);

impl Index<(u32, u32)> for Matrix {
    type Output = Block;
    fn index(&self, idx: (u32, u32)) -> &Block {
        match idx {
            (row, col) => &(self.0[row as usize])[col as usize],
        }
    }
}

impl IndexMut<(u32, u32)> for Matrix {
    fn index_mut(&mut self, idx: (u32, u32)) -> &mut Block {
        match idx {
            (row, col) => &mut (self.0[row as usize])[col as usize],
        }
    }
}

impl Matrix {
    pub fn new(lanes: u32, lanelen: u32) -> Self {
        let newlane = || (0..lanelen).map(|_| zero()).collect();
        Matrix((0..lanes).map(|_| newlane()).collect())
    }

    pub fn get3(&mut self, wr: (u32, u32), rd0: (u32, u32), rd1: (u32, u32))
                -> (&mut Block, &Block, &Block) {
        assert!(wr != rd0 && wr != rd1);
        let p: *mut Matrix = self;
        let rv = unsafe { (&mut (*p)[wr], &(*p)[rd0], &(*p)[rd1]) };
        rv
    }

    pub fn lanes_as_mut(&mut self) -> Vec<&mut Self> {
        let p: *mut Self = self;
        (0..self.0.len()).map(|_| unsafe { &mut (*p) }).collect()
    }

    pub fn col(&self, col: u32) -> Vec<&Block> {
        self.0.iter().map(|l| &l[col as usize]).collect()
    }

    pub fn iter(&self) -> Iter<Vec<Block>> { self.0.iter() }
}

impl Drop for Matrix {
    fn drop(&mut self) {
        for lane in self.0.iter_mut() {
            for blk in lane.iter_mut() {
                *blk = zero();
            }
        }
    }
}