Skip to content
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

try adding FunctionTypeVarArgs #1062

Closed
wants to merge 45 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
7b242ac
Refactor: break out fn valid_row for validating a TypeRow
acl-cqc Jan 10, 2024
ba05010
Type::{validate,substitute}_in_row, {Substitution,SubstValues}::apply…
acl-cqc Jan 10, 2024
59c5854
[test] types.rs::construct - also validate
acl-cqc Jan 12, 2024
72b4fb2
[tests] poly_func.rs: test row variables
acl-cqc Jan 12, 2024
2cf1e7f
[test] validate Hugr passing around Function of unknown arity
acl-cqc Jan 12, 2024
8899806
[tests]Validation prevents row variables being used in node signatures
acl-cqc Jan 12, 2024
e1f1825
Simplify default apply_typevar_in_row
acl-cqc Jan 12, 2024
1074093
Separate TypeEnum::RowVariable
acl-cqc Jan 12, 2024
4291848
Merge remote-tracking branch 'origin/main' into feat/row_typevar
acl-cqc Apr 15, 2024
98c75a2
Doc updates - no DeBruijn, ref OpDef vs FuncDefn
acl-cqc Apr 15, 2024
b918bee
Remove unused test setup changes
acl-cqc Apr 15, 2024
a096f1a
Merge remote-tracking branch 'origin/main' into feat/row_typevar
acl-cqc May 9, 2024
4b0e23f
Combine Type::substitute(_in_row), doc, add TODOs
acl-cqc May 9, 2024
a0aba20
FunctionType/PolyFuncType validate_varargs (fixed-len not reqd)
acl-cqc May 9, 2024
bd0689f
Type::validate(=>_1type), fix TypeParam to call validate_in_row
acl-cqc May 10, 2024
72f4b98
{subst,valid}_row => TypeRow::{substitute,validate_var_len}
acl-cqc May 10, 2024
d9653aa
Normalize representation of row variable as TypeArg; extend check_typ…
acl-cqc May 10, 2024
15bf177
Test bad schema, test can't fit row variable into type hole
acl-cqc May 10, 2024
bc11c68
Extend validation tests using eval+parallel
acl-cqc May 10, 2024
f39373b
And extend no_outer_row_variables to include unconnected out'port's o…
acl-cqc May 10, 2024
bd39303
comments re. canonicalization, tests
acl-cqc May 10, 2024
ed64d12
clippy
acl-cqc May 11, 2024
d02c9c0
single_type_seq -> seq1ty
acl-cqc May 11, 2024
50b9c85
Merge remote-tracking branch 'origin/main' into feat/row_typevar
acl-cqc May 11, 2024
742ab55
Combine helpers giving extension_with_eval_parallel
acl-cqc May 12, 2024
ce72762
Add RowVar to serialization/tys.py
acl-cqc May 12, 2024
c556a8a
Regenerate schema
acl-cqc May 12, 2024
7214cbc
Really regenerate schema (after poetry install)
acl-cqc May 12, 2024
ce28154
Use Vec<TypeArg> .into() -> TypeArg
acl-cqc May 14, 2024
0e4b232
Add TypeParam::new_list
acl-cqc May 14, 2024
5b39b9c
rowparam() -> rowp.clone()
acl-cqc May 14, 2024
9ae2255
new_row_var => new_row_var_use, some cleanups using .into()
acl-cqc May 14, 2024
412ec08
squash! Add TypeParam::new_list
acl-cqc May 14, 2024
34ed4c9
Fix apply_row_var...TODO cover this
acl-cqc May 14, 2024
85d1969
check_type_arg_rv: improve comment
acl-cqc May 14, 2024
3cf333d
Remove rowvar_in_list using TypeParam::contains
acl-cqc May 14, 2024
2929135
Add Type::row_var_bound
acl-cqc May 14, 2024
0afdaf9
Remove check_type_arg_rv
acl-cqc May 14, 2024
5a8bf88
Use Type .into() -> TypeArg more
acl-cqc May 14, 2024
2e53529
Extend comment that TypeArg::Variable's are not row-vars
acl-cqc May 14, 2024
ed09ae6
type_row.rs: comment ...Variable]*s*
acl-cqc May 14, 2024
8c93c9d
Add check_type_arg tests as rstest cases
acl-cqc May 14, 2024
8811e1a
More tests! (giving up on rstest)
acl-cqc May 14, 2024
d728714
Add a couple of PolyFuncType-serialization tests
acl-cqc May 14, 2024
5ec995f
try adding FunctionTypeVarArgs
doug-q May 16, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 18 additions & 1 deletion hugr-py/src/hugr/serialization/tys.py
Original file line number Diff line number Diff line change
Expand Up @@ -214,6 +214,15 @@ class Variable(ConfiguredBaseModel):
b: "TypeBound"


class RowVar(ConfiguredBaseModel):
"""A variable standing for a row of some (unknown) number of types.
May occur only within a row; not a node input/output."""

t: Literal["R"] = "R"
i: int
b: "TypeBound"


class USize(ConfiguredBaseModel):
"""Unsigned integer size type."""

Expand Down Expand Up @@ -321,7 +330,15 @@ class Type(RootModel):
"""A HUGR type."""

root: Annotated[
Qubit | Variable | USize | FunctionType | Array | SumType | Opaque | Alias,
Qubit
| Variable
| RowVar
| USize
| FunctionType
| Array
| SumType
| Opaque
| Alias,
WrapValidator(_json_custom_error_validator),
Field(discriminator="t"),
]
Expand Down
4 changes: 2 additions & 2 deletions hugr/src/builder/build_traits.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ use crate::{
types::EdgeKind,
};

use crate::extension::{ExtensionRegistry, ExtensionSet, PRELUDE_REGISTRY};
use crate::extension::{ExtensionRegistry, ExtensionSet, SignatureError, PRELUDE_REGISTRY};
use crate::types::{FunctionType, PolyFuncType, Type, TypeArg, TypeRow};

use itertools::Itertools;
Expand Down Expand Up @@ -87,7 +87,7 @@ pub trait Container {
name: impl Into<String>,
signature: PolyFuncType,
) -> Result<FunctionBuilder<&mut Hugr>, BuildError> {
let body = signature.body().clone();
let body = signature.body_norowvars()?.clone();
let f_node = self.add_child_node(NodeType::new_pure(ops::FuncDefn {
name: name.into(),
signature,
Expand Down
2 changes: 1 addition & 1 deletion hugr/src/builder/dataflow.rs
Original file line number Diff line number Diff line change
Expand Up @@ -147,7 +147,7 @@ impl FunctionBuilder<Hugr> {
///
/// Error in adding DFG child nodes.
pub fn new(name: impl Into<String>, signature: PolyFuncType) -> Result<Self, BuildError> {
let body = signature.body().clone();
let body = signature.body_norowvars()?.clone();
let op = ops::FuncDefn {
signature,
name: name.into(),
Expand Down
2 changes: 1 addition & 1 deletion hugr/src/builder/module.rs
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,7 @@ impl<T: AsMut<Hugr> + AsRef<Hugr>> ModuleBuilder<T> {
op_desc: "crate::ops::OpType::FuncDecl",
})?
.clone();
let body = signature.body().clone();
let body = signature.body_norowvars()?.clone();
self.hugr_mut()
.replace_op(
f_node,
Expand Down
3 changes: 3 additions & 0 deletions hugr/src/extension.rs
Original file line number Diff line number Diff line change
Expand Up @@ -161,6 +161,9 @@ pub enum SignatureError {
/// A type variable that was used has not been declared
#[error("Type variable {idx} was not declared ({num_decls} in scope)")]
FreeTypeVar { idx: usize, num_decls: usize },
/// A row variable was found outside of a variable-length row
#[error("Row variable {idx} was found outside of a type row")]
RowTypeVarOutsideRow { idx: usize },
/// The result of the type application stored in a [Call]
/// is not what we get by applying the type-args to the polymorphic function
///
Expand Down
19 changes: 17 additions & 2 deletions hugr/src/extension/op_def.rs
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ use super::{

use crate::ops::{OpName, OpNameRef};
use crate::types::type_param::{check_type_args, TypeArg, TypeParam};
use crate::types::{FunctionType, PolyFuncType};
use crate::types::{FunctionType, FunctionTypeVarArgs, PolyFuncType, Substitution};
use crate::Hugr;

/// Trait necessary for binary computations of OpDef signature
Expand Down Expand Up @@ -405,7 +405,10 @@ impl OpDef {
// TODO https://github.com/CQCL/hugr/issues/624 validate declared TypeParams
// for both type scheme and custom binary
if let SignatureFunc::TypeScheme(ts) = &self.signature_func {
ts.poly_func.validate(exts)?;
// The type scheme may contain row variables so be of variable length;
// these will have to be substituted to fixed-length concrete types when
// the OpDef is instantiated into an actual OpType.
ts.poly_func.validate_varargs(exts)?;
}
Ok(())
}
Expand Down Expand Up @@ -482,6 +485,7 @@ mod test {
use crate::extension::{SignatureError, EMPTY_REG, PRELUDE_REGISTRY};
use crate::ops::{CustomOp, OpName};
use crate::std_extensions::collections::{EXTENSION, LIST_TYPENAME};
use crate::types::type_param::TypeArgError;
use crate::types::Type;
use crate::types::{type_param::TypeParam, FunctionType, PolyFuncType, TypeArg, TypeBound};
use crate::{const_extension_ids, Extension};
Expand Down Expand Up @@ -639,6 +643,17 @@ mod test {
def.compute_signature(&args, &EMPTY_REG),
Ok(FunctionType::new_endo(vec![tv]))
);
// But not with an external row variable
let arg: TypeArg = Type::new_row_var_use(0, TypeBound::Eq).into();
assert_eq!(
def.compute_signature(&[arg.clone()], &EMPTY_REG),
Err(SignatureError::TypeArgMismatch(
TypeArgError::TypeMismatch {
param: TypeBound::Any.into(),
arg
}
))
);
Ok(())
}

Expand Down
21 changes: 20 additions & 1 deletion hugr/src/hugr/serialize/test.rs
Original file line number Diff line number Diff line change
Expand Up @@ -392,13 +392,32 @@ fn polyfunctype1() -> PolyFuncType {
PolyFuncType::new([TypeParam::max_nat(), TypeParam::Extensions], function_type)
}

fn polyfunctype2() -> PolyFuncType {
let tv0 = Type::new_row_var_use(0, TypeBound::Any);
let tv1 = Type::new_row_var_use(1, TypeBound::Eq);
let params = [TypeBound::Any, TypeBound::Eq].map(TypeParam::new_list);
let inputs = vec![
Type::new_function(FunctionType::new(tv0.clone(), tv1.clone())),
tv0,
];
let res = PolyFuncType::new(params, FunctionType::new(inputs, tv1));
// Just check we've got the arguments the right way round
// (not that it really matters for the serialization schema we have)
res.validate_varargs(&EMPTY_REG).unwrap();
res
}

#[rstest]
#[case(FunctionType::new_endo(type_row![]).into())]
#[case(polyfunctype1())]
#[case(PolyFuncType::new([TypeParam::Opaque { ty: int_custom_type(TypeArg::BoundedNat { n: 1 }) }], FunctionType::new_endo(type_row![Type::new_var_use(0, TypeBound::Copyable)])))]
#[case(PolyFuncType::new([TypeBound::Eq.into()], FunctionType::new_endo(type_row![Type::new_var_use(0, TypeBound::Eq)])))]
#[case(PolyFuncType::new([TypeParam::List { param: Box::new(TypeBound::Any.into()) }], FunctionType::new_endo(type_row![])))]
#[case(PolyFuncType::new([TypeParam::new_list(TypeBound::Any)], FunctionType::new_endo(type_row![])))]
#[case(PolyFuncType::new([TypeParam::Tuple { params: [TypeBound::Any.into(), TypeParam::bounded_nat(2.try_into().unwrap())].into() }], FunctionType::new_endo(type_row![])))]
#[case(PolyFuncType::new(
[TypeParam::new_list(TypeBound::Any)],
FunctionType::new_endo(Type::new_tuple(Type::new_row_var_use(0, TypeBound::Any)))))]
#[case(polyfunctype2())]
fn roundtrip_polyfunctype(#[case] poly_func_type: PolyFuncType) {
check_testing_roundtrip(poly_func_type)
}
Expand Down
9 changes: 6 additions & 3 deletions hugr/src/hugr/validate.rs
Original file line number Diff line number Diff line change
Expand Up @@ -301,11 +301,14 @@ impl<'a, 'b> ValidationContext<'a, 'b> {
var_decls: &[TypeParam],
) -> Result<(), SignatureError> {
match &port_kind {
EdgeKind::Value(ty) => ty.validate(self.extension_registry, var_decls),
EdgeKind::Value(ty) => ty.validate_1type(self.extension_registry, var_decls),
// Static edges must *not* refer to type variables declared by enclosing FuncDefns
// as these are only types at runtime.
EdgeKind::Const(ty) => ty.validate(self.extension_registry, &[]),
EdgeKind::Function(pf) => pf.validate(self.extension_registry),
EdgeKind::Const(ty) => ty.validate_1type(self.extension_registry, &[]),
// Allow function "value" to have unknown arity. A Call node will have to provide
// TypeArgs that produce a known arity, but a LoadFunction might pass the function
// value ("function pointer") around without knowing how to call it.
EdgeKind::Function(pf) => pf.validate_varargs(self.extension_registry),
_ => Ok(()),
}
}
Expand Down
146 changes: 144 additions & 2 deletions hugr/src/hugr/validate/test.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
use cool_asserts::assert_matches;
use rstest::rstest;

use super::*;
use crate::builder::test::closed_dfg_root_hugr;
Expand All @@ -10,7 +11,7 @@ use crate::extension::prelude::{BOOL_T, PRELUDE, PRELUDE_ID, USIZE_T};
use crate::extension::{Extension, ExtensionSet, TypeDefBound, EMPTY_REG, PRELUDE_REGISTRY};
use crate::hugr::hugrmut::sealed::HugrMutInternals;
use crate::hugr::HugrMut;
use crate::ops::dataflow::IOTrait;
use crate::ops::dataflow::{IOTrait, LoadFunction};
use crate::ops::handle::NodeHandle;
use crate::ops::leaf::MakeTuple;
use crate::ops::{self, Noop, OpType, Value};
Expand Down Expand Up @@ -546,6 +547,147 @@ fn no_polymorphic_consts() -> Result<(), Box<dyn std::error::Error>> {
Ok(())
}

fn extension_with_eval_parallel() -> Extension {
let rowp = TypeParam::new_list(TypeBound::Any);
let mut e = Extension::new(EXT_ID);

let inputs = Type::new_row_var_use(0, TypeBound::Any);
let outputs = Type::new_row_var_use(1, TypeBound::Any);
let evaled_fn = Type::new_function(FunctionType::new(inputs.clone(), outputs.clone()));
let pf = PolyFuncType::new(
[rowp.clone(), rowp.clone()],
FunctionType::new(vec![evaled_fn, inputs], outputs),
);
e.add_op("eval".into(), "".into(), pf).unwrap();

let rv = |idx| Type::new_row_var_use(idx, TypeBound::Any);
let pf = PolyFuncType::new(
[rowp.clone(), rowp.clone(), rowp.clone(), rowp.clone()],
FunctionType::new(
vec![
Type::new_function(FunctionType::new(rv(0), rv(2))),
Type::new_function(FunctionType::new(rv(1), rv(3))),
],
Type::new_function(FunctionType::new(vec![rv(0), rv(1)], vec![rv(2), rv(3)])),
),
);
e.add_op("parallel".into(), "".into(), pf).unwrap();

e
}

#[test]
fn instantiate_row_variables() -> Result<(), Box<dyn std::error::Error>> {
fn uint_seq(i: usize) -> TypeArg {
vec![TypeArg::Type { ty: USIZE_T }; i].into()
}
let e = extension_with_eval_parallel();
let mut dfb = DFGBuilder::new(FunctionType::new(
vec![
Type::new_function(FunctionType::new(USIZE_T, vec![USIZE_T, USIZE_T])),
USIZE_T,
], // inputs: function + its argument
vec![USIZE_T; 4], // outputs (*2^2, three calls)
))?;
let [func, int] = dfb.input_wires_arr();
let eval = e.instantiate_extension_op("eval", [uint_seq(1), uint_seq(2)], &PRELUDE_REGISTRY)?;
let [a, b] = dfb.add_dataflow_op(eval, [func, int])?.outputs_arr();
let par = e.instantiate_extension_op(
"parallel",
[uint_seq(1), uint_seq(1), uint_seq(2), uint_seq(2)],
&PRELUDE_REGISTRY,
)?;
let [par_func] = dfb.add_dataflow_op(par, [func, func])?.outputs_arr();
let eval2 =
e.instantiate_extension_op("eval", [uint_seq(2), uint_seq(4)], &PRELUDE_REGISTRY)?;
let eval2 = dfb.add_dataflow_op(eval2, [par_func, a, b])?;
dfb.finish_hugr_with_outputs(
eval2.outputs(),
&ExtensionRegistry::try_new([PRELUDE.to_owned(), e]).unwrap(),
)?;
Ok(())
}

fn seq1ty(t: Type) -> TypeArg {
TypeArg::Sequence {
elems: vec![t.into()],
}
}

#[test]
fn inner_row_variables() -> Result<(), Box<dyn std::error::Error>> {
let e = extension_with_eval_parallel();
let tv = Type::new_row_var_use(0, TypeBound::Any);
let inner_ft = Type::new_function(FunctionType::new_endo(tv.clone()));
let ft_usz = Type::new_function(FunctionType::new_endo(vec![tv.clone(), USIZE_T]));
let mut fb = FunctionBuilder::new(
"id",
PolyFuncType::new(
[TypeParam::new_list(TypeBound::Any)],
FunctionType::new(inner_ft.clone(), ft_usz),
),
)?;
// All the wires here are carrying higher-order Function values
let [func_arg] = fb.input_wires_arr();
let [id_usz] = {
let bldr = fb.define_function("id_usz", FunctionType::new_endo(USIZE_T).into())?;
let vals = bldr.input_wires();
let [inner_def] = bldr.finish_with_outputs(vals)?.outputs_arr();
let loadf = LoadFunction::try_new(
FunctionType::new_endo(USIZE_T).into(),
[],
&PRELUDE_REGISTRY,
)
.unwrap();
fb.add_dataflow_op(loadf, [inner_def])?.outputs_arr()
};
let par = e.instantiate_extension_op(
"parallel",
[tv.clone(), USIZE_T, tv.clone(), USIZE_T].map(seq1ty),
&PRELUDE_REGISTRY,
)?;
let par_func = fb.add_dataflow_op(par, [func_arg, id_usz])?;
fb.finish_hugr_with_outputs(
par_func.outputs(),
&ExtensionRegistry::try_new([PRELUDE.to_owned(), e]).unwrap(),
)?;
Ok(())
}

#[rstest]
#[case(false)]
#[case(true)]
fn no_outer_row_variables(#[case] connect: bool) {
doug-q marked this conversation as resolved.
Show resolved Hide resolved
let e = extension_with_eval_parallel();
let tv = Type::new_row_var_use(0, TypeBound::Copyable);
let mut fb = FunctionBuilder::new(
"bad_eval",
PolyFuncType::new(
[TypeParam::new_list(TypeBound::Copyable)],
FunctionType::new(
Type::new_function(FunctionType::new(USIZE_T, tv.clone())),
if connect { vec![tv.clone()] } else { vec![] },
),
),
).unwrap();
let [func_arg] = fb.input_wires_arr();
let i = fb.add_load_value(crate::extension::prelude::ConstUsize::new(5));
let ev =
e.instantiate_extension_op("eval", [seq1ty(USIZE_T), seq1ty(tv)], &PRELUDE_REGISTRY).unwrap();
let ev = fb.add_dataflow_op(ev, [func_arg, i]).unwrap();
let reg = ExtensionRegistry::try_new([PRELUDE.to_owned(), e]).unwrap();
if connect {
fb.set_outputs(ev.outputs()).unwrap();
}
assert_eq!(
fb.finish_hugr(&reg).unwrap_err(),
ValidationError::SignatureError {
node: ev.node(),
cause: SignatureError::RowTypeVarOutsideRow { idx: 0 }
}
);
}

#[test]
fn test_polymorphic_call() -> Result<(), Box<dyn std::error::Error>> {
let mut e = Extension::new(EXT_ID);
Expand All @@ -562,7 +704,7 @@ fn test_polymorphic_call() -> Result<(), Box<dyn std::error::Error>> {
)
.with_extension_delta(ExtensionSet::type_var(1)),
);
// The higher-order "eval" operation - takes a function and its argument.
// Single-input/output version of the higher-order "eval" operation, with extension param.
// Note the extension-delta of the eval node includes that of the input function.
e.add_op(
"eval".into(),
Expand Down
3 changes: 2 additions & 1 deletion hugr/src/ops/module.rs
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,8 @@ impl StaticTag for FuncDefn {

impl DataflowParent for FuncDefn {
fn inner_signature(&self) -> FunctionType {
self.signature.body().clone()
// ok by validation
self.signature.body_norowvars().unwrap().clone()
}
}

Expand Down
Loading
Loading