//@ run-pass //@ compile-flags: -Z validate-mir //@ revisions: edition2021 edition2024 //@ [edition2021] edition: 2021 //@ [edition2024] compile-flags: -Z unstable-options //@ [edition2024] edition: 2024 #![feature(let_chains)] use std::cell::RefCell; use std::convert::TryInto; #[derive(Default)] struct DropOrderCollector(RefCell>); struct LoudDrop<'a>(&'a DropOrderCollector, u32); impl Drop for LoudDrop<'_> { fn drop(&mut self) { println!("{}", self.1); self.0.0.borrow_mut().push(self.1); } } impl DropOrderCollector { fn option_loud_drop(&self, n: u32) -> Option { Some(LoudDrop(self, n)) } fn loud_drop(&self, n: u32) -> LoudDrop { LoudDrop(self, n) } fn print(&self, n: u32) { println!("{}", n); self.0.borrow_mut().push(n) } fn if_(&self) { if self.option_loud_drop(1).is_some() { self.print(2); } if self.option_loud_drop(3).is_none() { unreachable!(); } else if self.option_loud_drop(4).is_some() { self.print(5); } if { if self.option_loud_drop(6).is_some() && self.option_loud_drop(7).is_some() { self.loud_drop(8); true } else { false } } { self.print(9); } } fn if_let(&self) { #[cfg(edition2021)] if let None = self.option_loud_drop(2) { unreachable!(); } else { self.print(1); } #[cfg(edition2024)] if let None = self.option_loud_drop(1) { unreachable!(); } else { self.print(2); } if let Some(_) = self.option_loud_drop(4) { self.print(3); } if let Some(_d) = self.option_loud_drop(6) { self.print(5); } } fn match_(&self) { match self.option_loud_drop(2) { _any => self.print(1), } match self.option_loud_drop(4) { _ => self.print(3), } match self.option_loud_drop(6) { Some(_) => self.print(5), _ => unreachable!(), } match { let _ = self.loud_drop(7); let _d = self.loud_drop(9); self.print(8); () } { () => self.print(10), } #[cfg(edition2021)] match { match self.option_loud_drop(14) { _ => { self.print(11); self.option_loud_drop(13) } } } { _ => self.print(12), } #[cfg(edition2024)] match { match self.option_loud_drop(12) { _ => { self.print(11); self.option_loud_drop(14) } } } { _ => self.print(13), } match { loop { break match self.option_loud_drop(16) { _ => { self.print(15); self.option_loud_drop(18) } }; } } { _ => self.print(17), } } fn and_chain(&self) { // issue-103107 if self.option_loud_drop(1).is_some() // 1 && self.option_loud_drop(2).is_some() // 2 && self.option_loud_drop(3).is_some() // 3 && self.option_loud_drop(4).is_some() // 4 && self.option_loud_drop(5).is_some() // 5 { self.print(6); // 6 } let _ = self.option_loud_drop(7).is_some() // 1 && self.option_loud_drop(8).is_some() // 2 && self.option_loud_drop(9).is_some(); // 3 self.print(10); // 4 // Test associativity if self.option_loud_drop(11).is_some() // 1 && (self.option_loud_drop(12).is_some() // 2 && self.option_loud_drop(13).is_some() // 3 && self.option_loud_drop(14).is_some()) // 4 && self.option_loud_drop(15).is_some() // 5 { self.print(16); // 6 } } fn or_chain(&self) { // issue-103107 if self.option_loud_drop(1).is_none() // 1 || self.option_loud_drop(2).is_none() // 2 || self.option_loud_drop(3).is_none() // 3 || self.option_loud_drop(4).is_none() // 4 || self.option_loud_drop(5).is_some() // 5 { self.print(6); // 6 } let _ = self.option_loud_drop(7).is_none() // 1 || self.option_loud_drop(8).is_none() // 2 || self.option_loud_drop(9).is_none(); // 3 self.print(10); // 4 // Test associativity if self.option_loud_drop(11).is_none() // 1 || (self.option_loud_drop(12).is_none() // 2 || self.option_loud_drop(13).is_none() // 3 || self.option_loud_drop(14).is_none()) // 4 || self.option_loud_drop(15).is_some() // 5 { self.print(16); // 6 } } fn mixed_and_or_chain(&self) { // issue-103107 if self.option_loud_drop(1).is_none() // 1 || self.option_loud_drop(2).is_none() // 2 || self.option_loud_drop(3).is_some() // 3 && self.option_loud_drop(4).is_some() // 4 && self.option_loud_drop(5).is_none() // 5 || self.option_loud_drop(6).is_none() // 6 || self.option_loud_drop(7).is_some() // 7 { self.print(8); // 8 } } fn let_chain(&self) { // take the "then" branch if self.option_loud_drop(1).is_some() // 1 && self.option_loud_drop(2).is_some() // 2 && let Some(_d) = self.option_loud_drop(4) { // 4 self.print(3); // 3 } #[cfg(edition2021)] // take the "else" branch if self.option_loud_drop(5).is_some() // 1 && self.option_loud_drop(6).is_some() // 2 && let None = self.option_loud_drop(8) { // 4 unreachable!(); } else { self.print(7); // 3 } #[cfg(edition2024)] // take the "else" branch if self.option_loud_drop(5).is_some() // 1 && self.option_loud_drop(6).is_some() // 2 && let None = self.option_loud_drop(7) { // 4 unreachable!(); } else { self.print(8); // 3 } // let exprs interspersed if self.option_loud_drop(9).is_some() // 1 && let Some(_d) = self.option_loud_drop(13) // 5 && self.option_loud_drop(10).is_some() // 2 && let Some(_e) = self.option_loud_drop(12) { // 4 self.print(11); // 3 } // let exprs first if let Some(_d) = self.option_loud_drop(18) // 5 && let Some(_e) = self.option_loud_drop(17) // 4 && self.option_loud_drop(14).is_some() // 1 && self.option_loud_drop(15).is_some() { // 2 self.print(16); // 3 } // let exprs last if self.option_loud_drop(19).is_some() // 1 && self.option_loud_drop(20).is_some() // 2 && let Some(_d) = self.option_loud_drop(23) // 5 && let Some(_e) = self.option_loud_drop(22) { // 4 self.print(21); // 3 } } fn while_(&self) { let mut v = self.option_loud_drop(4); while let Some(_d) = v && self.option_loud_drop(1).is_some() && self.option_loud_drop(2).is_some() { self.print(3); v = None; } } fn assert_sorted(self) { assert!( self.0 .into_inner() .into_iter() .enumerate() .all(|(idx, item)| idx + 1 == item.try_into().unwrap()) ); } } fn main() { println!("-- if --"); let collector = DropOrderCollector::default(); collector.if_(); collector.assert_sorted(); println!("-- and chain --"); let collector = DropOrderCollector::default(); collector.and_chain(); collector.assert_sorted(); println!("-- or chain --"); let collector = DropOrderCollector::default(); collector.or_chain(); collector.assert_sorted(); println!("-- mixed and/or chain --"); let collector = DropOrderCollector::default(); collector.mixed_and_or_chain(); collector.assert_sorted(); println!("-- if let --"); let collector = DropOrderCollector::default(); collector.if_let(); collector.assert_sorted(); println!("-- match --"); let collector = DropOrderCollector::default(); collector.match_(); collector.assert_sorted(); println!("-- let chain --"); let collector = DropOrderCollector::default(); collector.let_chain(); collector.assert_sorted(); println!("-- while --"); let collector = DropOrderCollector::default(); collector.while_(); collector.assert_sorted(); }