-
Notifications
You must be signed in to change notification settings - Fork 7
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
fix: binary compute_signature returning a PolyFuncType with binders #710
Merged
+127
−4
Merged
Changes from all commits
Commits
Show all changes
6 commits
Select commit
Hold shift + click to select a range
098d623
Test binary compute_signature returning a PolyFuncType with binders
acl-cqc d973909
Bonus: check simple TypeSchemes can be instantiated with type vars too
acl-cqc 2162552
ci fmt
acl-cqc fb76697
Ah-ha, use correct fmt
acl-cqc d0d9da4
Use `vec![...;3]` syntax
acl-cqc 5f4e71a
comments
acl-cqc File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -180,9 +180,6 @@ impl OpDef { | |
args: &[TypeArg], | ||
exts: &ExtensionRegistry, | ||
) -> Result<FunctionType, SignatureError> { | ||
// Hugr's are monomorphic, so check the args have no free variables | ||
args.iter().try_for_each(|ta| ta.validate(exts, &[]))?; | ||
|
||
let temp: PolyFuncType; // to keep alive | ||
let (pf, args) = match &self.signature_func { | ||
SignatureFunc::TypeScheme(ts) => (ts, args), | ||
|
@@ -369,11 +366,16 @@ impl Extension { | |
|
||
#[cfg(test)] | ||
mod test { | ||
use std::num::NonZeroU64; | ||
|
||
use smol_str::SmolStr; | ||
|
||
use crate::builder::{DFGBuilder, Dataflow, DataflowHugr}; | ||
use crate::extension::prelude::USIZE_T; | ||
use crate::extension::{ExtensionRegistry, PRELUDE}; | ||
use crate::extension::{ | ||
CustomSignatureFunc, ExtensionRegistry, SignatureError, EMPTY_REG, PRELUDE, | ||
PRELUDE_REGISTRY, | ||
}; | ||
use crate::ops::custom::ExternalOp; | ||
use crate::ops::LeafOp; | ||
use crate::std_extensions::collections::{EXTENSION, LIST_TYPENAME}; | ||
|
@@ -413,4 +415,119 @@ mod test { | |
|
||
Ok(()) | ||
} | ||
|
||
#[test] | ||
fn binary_polyfunc() -> Result<(), Box<dyn std::error::Error>> { | ||
// Test a custom binary `compute_signature` that returns a PolyFuncType | ||
// where the latter declares more type params itself. In particular, | ||
// we should be able to substitute (external) type variables into the latter, | ||
// but not pass them into the former (custom binary function). | ||
struct SigFun(); | ||
impl CustomSignatureFunc for SigFun { | ||
fn compute_signature( | ||
&self, | ||
_name: &SmolStr, | ||
arg_values: &[TypeArg], | ||
_misc: &std::collections::HashMap<String, serde_yaml::Value>, | ||
_exts: &ExtensionRegistry, | ||
) -> Result<PolyFuncType, SignatureError> { | ||
const TP: TypeParam = TypeParam::Type(TypeBound::Any); | ||
let [TypeArg::BoundedNat { n }] = arg_values else { | ||
return Err(SignatureError::InvalidTypeArgs); | ||
}; | ||
let n = *n as usize; | ||
let tvs: Vec<Type> = (0..n) | ||
.map(|_| Type::new_var_use(0, TypeBound::Any)) | ||
.collect(); | ||
Ok(PolyFuncType::new( | ||
vec![TP], | ||
FunctionType::new(tvs.clone(), vec![Type::new_tuple(tvs)]), | ||
)) | ||
} | ||
} | ||
let mut e = Extension::new(EXT_ID); | ||
let def = e.add_op_custom_sig_simple( | ||
"MyOp".into(), | ||
"".to_string(), | ||
vec![TypeParam::max_nat()], | ||
SigFun(), | ||
)?; | ||
|
||
// Base case, no type variables: | ||
let args = [TypeArg::BoundedNat { n: 3 }, USIZE_T.into()]; | ||
assert_eq!( | ||
def.compute_signature(&args, &PRELUDE_REGISTRY), | ||
Ok(FunctionType::new( | ||
vec![USIZE_T; 3], | ||
vec![Type::new_tuple(vec![USIZE_T; 3])] | ||
)) | ||
); | ||
assert_eq!(def.validate_args(&args, &PRELUDE_REGISTRY, &[]), Ok(())); | ||
|
||
// Second arg may be a variable (substitutable) | ||
let tyvar = Type::new_var_use(0, TypeBound::Eq); | ||
let tyvars: Vec<Type> = vec![tyvar.clone(); 3]; | ||
let args = [TypeArg::BoundedNat { n: 3 }, tyvar.clone().into()]; | ||
assert_eq!( | ||
def.compute_signature(&args, &PRELUDE_REGISTRY), | ||
Ok(FunctionType::new( | ||
tyvars.clone(), | ||
vec![Type::new_tuple(tyvars)] | ||
)) | ||
); | ||
def.validate_args(&args, &PRELUDE_REGISTRY, &[TypeParam::Type(TypeBound::Eq)]) | ||
.unwrap(); | ||
|
||
// quick sanity check that we are validating the args - note changed bound: | ||
assert_eq!( | ||
def.validate_args(&args, &PRELUDE_REGISTRY, &[TypeParam::Type(TypeBound::Any)]), | ||
Err(SignatureError::TypeVarDoesNotMatchDeclaration { | ||
actual: TypeBound::Any.into(), | ||
cached: TypeBound::Eq.into() | ||
}) | ||
); | ||
|
||
// First arg must be concrete, not a variable | ||
let kind = TypeParam::bounded_nat(NonZeroU64::new(5).unwrap()); | ||
let args = [TypeArg::new_var_use(0, kind.clone()), USIZE_T.into()]; | ||
// We can't prevent this from getting into our compute_signature implementation: | ||
assert_eq!( | ||
def.compute_signature(&args, &PRELUDE_REGISTRY), | ||
Err(SignatureError::InvalidTypeArgs) | ||
); | ||
// But validation rules it out, even when the variable is declared: | ||
assert_eq!( | ||
def.validate_args(&args, &PRELUDE_REGISTRY, &[kind]), | ||
Err(SignatureError::FreeTypeVar { | ||
idx: 0, | ||
num_decls: 0 | ||
}) | ||
); | ||
|
||
Ok(()) | ||
} | ||
|
||
#[test] | ||
fn type_scheme_instantiate_var() -> Result<(), Box<dyn std::error::Error>> { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. not sure what this one is testing exactly, maybe add a comment? |
||
// Check that we can instantiate a PolyFuncType-scheme with an (external) | ||
// type variable | ||
let mut e = Extension::new(EXT_ID); | ||
let def = e.add_op_type_scheme_simple( | ||
"SimpleOp".into(), | ||
"".into(), | ||
PolyFuncType::new( | ||
vec![TypeParam::Type(TypeBound::Any)], | ||
FunctionType::new_endo(vec![Type::new_var_use(0, TypeBound::Any)]), | ||
), | ||
)?; | ||
let tv = Type::new_var_use(1, TypeBound::Eq); | ||
let args = [TypeArg::Type { ty: tv.clone() }]; | ||
let decls = [TypeParam::Extensions, TypeBound::Eq.into()]; | ||
def.validate_args(&args, &EMPTY_REG, &decls).unwrap(); | ||
assert_eq!( | ||
def.compute_signature(&args, &EMPTY_REG), | ||
Ok(FunctionType::new_endo(vec![tv])) | ||
); | ||
Ok(()) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I mean, we could do this for
args.take(self.static_params.len())
sort of thingThere was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm guessing the test fails without removing this line?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
(The case that tries to pass a typevar in as the second, substitutable, parameter, i.e. which should succeed)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
and the
type_scheme_instantiate_var