|  | //! Test the pattern complexity limit. | 
|  |  | 
|  | #![allow(unused_crate_dependencies)] | 
|  |  | 
|  | use common::*; | 
|  | use rustc_pattern_analysis::MatchArm; | 
|  | use rustc_pattern_analysis::pat::DeconstructedPat; | 
|  | use rustc_pattern_analysis::usefulness::PlaceValidity; | 
|  |  | 
|  | #[macro_use] | 
|  | mod common; | 
|  |  | 
|  | /// Analyze a match made of these patterns. Ignore the report; we only care whether we exceeded the | 
|  | /// limit or not. | 
|  | fn check(patterns: &[DeconstructedPat<Cx>], complexity_limit: usize) -> Result<(), ()> { | 
|  | let ty = *patterns[0].ty(); | 
|  | let arms: Vec<_> = | 
|  | patterns.iter().map(|pat| MatchArm { pat, has_guard: false, arm_data: () }).collect(); | 
|  | compute_match_usefulness(arms.as_slice(), ty, PlaceValidity::ValidOnly, complexity_limit, false) | 
|  | .map(|_report| ()) | 
|  | } | 
|  |  | 
|  | /// Asserts that analyzing this match takes exactly `complexity` steps. | 
|  | #[track_caller] | 
|  | fn assert_complexity(patterns: Vec<DeconstructedPat<Cx>>, complexity: usize) { | 
|  | assert!(check(&patterns, complexity).is_ok()); | 
|  | assert!(check(&patterns, complexity - 1).is_err()); | 
|  | } | 
|  |  | 
|  | /// Construct a match like: | 
|  | /// ```ignore(illustrative) | 
|  | /// match ... { | 
|  | ///     BigStruct { field01: true, .. } => {} | 
|  | ///     BigStruct { field02: true, .. } => {} | 
|  | ///     BigStruct { field03: true, .. } => {} | 
|  | ///     BigStruct { field04: true, .. } => {} | 
|  | ///     ... | 
|  | ///     _ => {} | 
|  | /// } | 
|  | /// ``` | 
|  | fn diagonal_match(arity: usize) -> Vec<DeconstructedPat<Cx>> { | 
|  | let struct_ty = Ty::BigStruct { arity, ty: &Ty::Bool }; | 
|  | let mut patterns = vec![]; | 
|  | for i in 0..arity { | 
|  | patterns.push(pat!(struct_ty; Struct { .i: true })); | 
|  | } | 
|  | patterns.push(pat!(struct_ty; _)); | 
|  | patterns | 
|  | } | 
|  |  | 
|  | /// Construct a match like: | 
|  | /// ```ignore(illustrative) | 
|  | /// match ... { | 
|  | ///     BigStruct { field01: true, .. } => {} | 
|  | ///     BigStruct { field02: true, .. } => {} | 
|  | ///     BigStruct { field03: true, .. } => {} | 
|  | ///     BigStruct { field04: true, .. } => {} | 
|  | ///     ... | 
|  | ///     BigStruct { field01: false, .. } => {} | 
|  | ///     BigStruct { field02: false, .. } => {} | 
|  | ///     BigStruct { field03: false, .. } => {} | 
|  | ///     BigStruct { field04: false, .. } => {} | 
|  | ///     ... | 
|  | ///     _ => {} | 
|  | /// } | 
|  | /// ``` | 
|  | fn diagonal_exponential_match(arity: usize) -> Vec<DeconstructedPat<Cx>> { | 
|  | let struct_ty = Ty::BigStruct { arity, ty: &Ty::Bool }; | 
|  | let mut patterns = vec![]; | 
|  | for i in 0..arity { | 
|  | patterns.push(pat!(struct_ty; Struct { .i: true })); | 
|  | } | 
|  | for i in 0..arity { | 
|  | patterns.push(pat!(struct_ty; Struct { .i: false })); | 
|  | } | 
|  | patterns.push(pat!(struct_ty; _)); | 
|  | patterns | 
|  | } | 
|  |  | 
|  | #[test] | 
|  | fn test_diagonal_struct_match() { | 
|  | // These cases are nicely linear: we check `arity` patterns with exactly one `true`, matching | 
|  | // in 2 branches each, and a final pattern with all `false`, matching only the `_` branch. | 
|  | assert_complexity(diagonal_match(20), 41); | 
|  | assert_complexity(diagonal_match(30), 61); | 
|  | // This case goes exponential. | 
|  | assert!(check(&diagonal_exponential_match(10), 10000).is_err()); | 
|  | } | 
|  |  | 
|  | /// Construct a match like: | 
|  | /// ```ignore(illustrative) | 
|  | /// match ... { | 
|  | ///     BigEnum::Variant1(_) => {} | 
|  | ///     BigEnum::Variant2(_) => {} | 
|  | ///     BigEnum::Variant3(_) => {} | 
|  | ///     ... | 
|  | ///     _ => {} | 
|  | /// } | 
|  | /// ``` | 
|  | fn big_enum(arity: usize) -> Vec<DeconstructedPat<Cx>> { | 
|  | let enum_ty = Ty::BigEnum { arity, ty: &Ty::Bool }; | 
|  | let mut patterns = vec![]; | 
|  | for i in 0..arity { | 
|  | patterns.push(pat!(enum_ty; Variant.i)); | 
|  | } | 
|  | patterns.push(pat!(enum_ty; _)); | 
|  | patterns | 
|  | } | 
|  |  | 
|  | #[test] | 
|  | fn test_big_enum() { | 
|  | // We try 2 branches per variant. | 
|  | assert_complexity(big_enum(20), 40); | 
|  | } |