diff --git a/nalgebra-sparse/src/proptest.rs b/nalgebra-sparse/src/proptest.rs index fae49028..80eb88a6 100644 --- a/nalgebra-sparse/src/proptest.rs +++ b/nalgebra-sparse/src/proptest.rs @@ -4,11 +4,42 @@ use crate::coo::CooMatrix; use proptest::prelude::*; -use proptest::collection::{vec, hash_map}; +use proptest::collection::{vec, hash_map, btree_set}; use nalgebra::Scalar; use std::cmp::min; -use std::iter::repeat; +use std::iter::{repeat}; use proptest::sample::{Index}; +use crate::csr::CsrMatrix; +use crate::pattern::SparsityPattern; +use std::sync::Arc; +use crate::csc::CscMatrix; + +fn dense_row_major_coord_strategy(nrows: usize, ncols: usize, nnz: usize) + -> impl Strategy> +{ + let mut booleans = vec![true; nnz]; + booleans.append(&mut vec![false; (nrows * ncols) - nnz]); + // Make sure that exactly `nnz` of the booleans are true + Just(booleans) + // Need to shuffle to make sure they are randomly distributed + .prop_shuffle() + .prop_map(move |booleans| { + booleans + .into_iter() + .enumerate() + .filter_map(|(index, is_entry)| { + if is_entry { + // Convert linear index to row/col pair + let i = index / ncols; + let j = index % ncols; + Some((i, j)) + } else { + None + } + }) + .collect::>() + }) +} /// A strategy for generating `nnz` triplets. /// @@ -177,4 +208,112 @@ where } coo }) +} + +fn sparsity_pattern_from_row_major_coords(nmajor: usize, nminor: usize, coords: I) -> SparsityPattern +where + I: Iterator + ExactSizeIterator, +{ + let mut minors = Vec::with_capacity(coords.len()); + let mut offsets = Vec::with_capacity(nmajor + 1); + let mut current_major = 0; + offsets.push(0); + for (idx, (i, j)) in coords.enumerate() { + assert!(i >= current_major); + assert!(i < nmajor && j < nminor, "Generated coords are out of bounds"); + while current_major < i{ + offsets.push(idx); + current_major += 1; + } + minors.push(j); + } + + while current_major < nmajor { + offsets.push(minors.len()); + current_major += 1; + } + + assert_eq!(offsets.first().unwrap(), &0); + assert_eq!(offsets.len(), nmajor + 1); + + SparsityPattern::try_from_offsets_and_indices(nmajor, + nminor, + offsets, + minors) + .expect("Internal error: Generated sparsity pattern is invalid") +} + +/// TODO +pub fn sparsity_pattern( + major_lanes: impl Strategy + 'static, + minor_lanes: impl Strategy + 'static, + max_nonzeros: usize) +-> impl Strategy +{ + (major_lanes, minor_lanes) + .prop_flat_map(move |(nmajor, nminor)| { + let max_nonzeros = min(nmajor * nminor, max_nonzeros); + (Just(nmajor), Just(nminor), 0 ..= max_nonzeros) + }).prop_flat_map(move |(nmajor, nminor, nnz)| { + if 10 * nnz < nmajor * nminor { + // If nnz is small compared to a dense matrix, then use a sparse sampling strategy + btree_set((0..nmajor, 0..nminor), nnz) + .prop_map(move |coords| { + sparsity_pattern_from_row_major_coords(nmajor, nminor, coords.into_iter()) + }) + .boxed() + } else { + // If the required number of nonzeros is sufficiently dense, + // we instead use a dense sampling + dense_row_major_coord_strategy(nmajor, nminor, nnz) + .prop_map(move |triplets| { + let coords = triplets.into_iter(); + sparsity_pattern_from_row_major_coords(nmajor, nminor, coords) + }).boxed() + } + }) +} + +/// TODO +pub fn csr(value_strategy: T, + rows: impl Strategy + 'static, + cols: impl Strategy + 'static, + max_nonzeros: usize) + -> impl Strategy> +where + T: Strategy + Clone + 'static, + T::Value: Scalar, +{ + sparsity_pattern(rows, cols, max_nonzeros) + .prop_flat_map(move |pattern| { + let nnz = pattern.nnz(); + let values = vec![value_strategy.clone(); nnz]; + (Just(pattern), values) + }) + .prop_map(|(pattern, values)| { + CsrMatrix::try_from_pattern_and_values(Arc::new(pattern), values) + .expect("Internal error: Generated CsrMatrix is invalid") + }) +} + +/// TODO +pub fn csc(value_strategy: T, + rows: impl Strategy + 'static, + cols: impl Strategy + 'static, + max_nonzeros: usize) + -> impl Strategy> + where + T: Strategy + Clone + 'static, + T::Value: Scalar, +{ + sparsity_pattern(cols, rows, max_nonzeros) + .prop_flat_map(move |pattern| { + let nnz = pattern.nnz(); + let values = vec![value_strategy.clone(); nnz]; + (Just(pattern), values) + }) + .prop_map(|(pattern, values)| { + CscMatrix::try_from_pattern_and_values(Arc::new(pattern), values) + .expect("Internal error: Generated CscMatrix is invalid") + }) } \ No newline at end of file diff --git a/nalgebra-sparse/tests/unit_tests/proptest.rs b/nalgebra-sparse/tests/unit_tests/proptest.rs index 72a3d40c..8e5731e0 100644 --- a/nalgebra-sparse/tests/unit_tests/proptest.rs +++ b/nalgebra-sparse/tests/unit_tests/proptest.rs @@ -131,4 +131,6 @@ mod slow { // is contained in the set of visited matrices assert!(all_combinations.is_subset(&visited_combinations)); } -} \ No newline at end of file +} + +// TODO: Tests for csr, csc and sparsity_pattern strategies \ No newline at end of file