From 4f71c4eaa40537637db56fe49b7284e53087a434 Mon Sep 17 00:00:00 2001 From: Michael Bryant Date: Wed, 22 Nov 2023 16:55:15 -0800 Subject: [PATCH 1/2] fix!: make NaN == NaN inside Expression This is the easiest way to resolve #316 in a way that doesn't break the API (the other way being to use `OrderedFloat`). --- Cargo.lock | 6 +++--- quil-rs/proptest-regressions/expression/mod.txt | 1 + quil-rs/src/expression/mod.rs | 14 +++++++++++++- 3 files changed, 17 insertions(+), 4 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index f4112525..3ea61d62 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1139,7 +1139,7 @@ checksum = "a1d01941d82fa2ab50be1e79e6714289dd7cde78eba4c074bc5a4374f650dfe0" [[package]] name = "quil-cli" -version = "0.6.3-rc.0" +version = "0.7.0-rc.0" dependencies = [ "anyhow", "clap", @@ -1148,7 +1148,7 @@ dependencies = [ [[package]] name = "quil-py" -version = "0.13.3-rc.1" +version = "0.14.0-rc.0" dependencies = [ "indexmap", "ndarray", @@ -1163,7 +1163,7 @@ dependencies = [ [[package]] name = "quil-rs" -version = "0.29.3-rc.0" +version = "0.30.0-rc.0" dependencies = [ "approx", "clap", diff --git a/quil-rs/proptest-regressions/expression/mod.txt b/quil-rs/proptest-regressions/expression/mod.txt index 846843a0..e830f9be 100644 --- a/quil-rs/proptest-regressions/expression/mod.txt +++ b/quil-rs/proptest-regressions/expression/mod.txt @@ -8,3 +8,4 @@ cc 4c32128d724ed0f840715fae4e194c99262dc153c64be39d2acf45b8903b20f7 # shrinks to cc 5cc95f2159ad7120bbaf296d3a9fb26fef30f57b61e76b3e0dc99f4759009fdb # shrinks to e = Number(Complex { re: 0.0, im: -2.772221265116396 }) cc de70a1853ccef983fac85a87761ba08bfb2d54b2d4e880d5d90e7b4a75ecafb5 # shrinks to e = Address(MemoryReference { name: "mut", index: 0 }) cc 9ad50859b68cb403ce1a67af0feef1f55d25587466878e364ba2810be5910b14 # shrinks to e = Address(MemoryReference { name: "iNf", index: 0 }) +cc 2b6281fa61604062f364d8e92d4b4e2753352d5ae1e76b7753bcb1201e8912b4 # shrinks to e = Infix(InfixExpression { left: Variable("ra"), operator: Slash, right: FunctionCall(FunctionCallExpression { function: Sine, expression: PiConstant }) }) diff --git a/quil-rs/src/expression/mod.rs b/quil-rs/src/expression/mod.rs index 1221afe2..97301b38 100644 --- a/quil-rs/src/expression/mod.rs +++ b/quil-rs/src/expression/mod.rs @@ -115,7 +115,10 @@ impl PartialEq for Expression { match (self, other) { (Self::Address(left), Self::Address(right)) => left == right, (Self::Infix(left), Self::Infix(right)) => left == right, - (Self::Number(left), Self::Number(right)) => left == right, + (Self::Number(left), Self::Number(right)) => { + (left.re == right.re || left.re.is_nan() && right.re.is_nan()) + && (left.im == right.im || left.im.is_nan() && right.im.is_nan()) + } (Self::Prefix(left), Self::Prefix(right)) => left == right, (Self::FunctionCall(left), Self::FunctionCall(right)) => left == right, (Self::Variable(left), Self::Variable(right)) => left == right, @@ -1010,6 +1013,7 @@ mod tests { prop_assert!(p.is_ok()); let p = p.unwrap(); let simple_p = p.clone().into_simplified(); + prop_assert_eq!( simple_p.clone(), simple_e.clone(), @@ -1039,12 +1043,20 @@ mod tests { } } + #[test] + fn test_nan_is_equal() { + let left = Expression::Number(f64::NAN.into()); + let right = left.clone(); + assert_eq!(left, right); + } + #[test] fn specific_simplification_tests() { for (input, expected) in [ ("pi", Expression::Number(PI.into())), ("pi/2", Expression::Number((PI / 2.0).into())), ("pi * pi", Expression::Number((PI.powi(2)).into())), + ("1.0/(1.0-1.0)", Expression::Number(f64::NAN.into())), ( "(a[0]*2*pi)/6.283185307179586", Expression::Address(MemoryReference { From 2eba287ec9cb54dce8a09b2be1526d352370eabc Mon Sep 17 00:00:00 2001 From: Antal Spector-Zabusky Date: Thu, 9 Jan 2025 20:05:53 -0500 Subject: [PATCH 2/2] docs: document NaN equality for expressions --- quil-rs/src/expression/mod.rs | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/quil-rs/src/expression/mod.rs b/quil-rs/src/expression/mod.rs index 97301b38..91383797 100644 --- a/quil-rs/src/expression/mod.rs +++ b/quil-rs/src/expression/mod.rs @@ -51,6 +51,10 @@ pub enum EvaluationError { NotANumber, } +/// The type of Quil expressions. +/// +/// Note that when comparing Quil expressions, any embedded NaNs are treated as *equal* to other +/// NaNs, not unequal, in contravention of the IEEE 754 spec. #[derive(Clone, Debug)] pub enum Expression { Address(MemoryReference), @@ -62,6 +66,10 @@ pub enum Expression { Variable(String), } +/// The type of function call Quil expressions, e.g. `sin(e)`. +/// +/// Note that when comparing Quil expressions, any embedded NaNs are treated as *equal* to other +/// NaNs, not unequal, in contravention of the IEEE 754 spec. #[derive(Clone, Debug, PartialEq, Eq, Hash)] pub struct FunctionCallExpression { pub function: ExpressionFunction, @@ -77,6 +85,10 @@ impl FunctionCallExpression { } } +/// The type of infix Quil expressions, e.g. `e1 + e2`. +/// +/// Note that when comparing Quil expressions, any embedded NaNs are treated as *equal* to other +/// NaNs, not unequal, in contravention of the IEEE 754 spec. #[derive(Clone, Debug, PartialEq, Eq, Hash)] pub struct InfixExpression { pub left: Box, @@ -94,6 +106,10 @@ impl InfixExpression { } } +/// The type of prefix Quil expressions, e.g. `-e`. +/// +/// Note that when comparing Quil expressions, any embedded NaNs are treated as *equal* to other +/// NaNs, not unequal, in contravention of the IEEE 754 spec. #[derive(Clone, Debug, PartialEq, Eq, Hash)] pub struct PrefixExpression { pub operator: PrefixOperator,