From fd95225ee1632b271d68c380c2fe3cc6867ea0d7 Mon Sep 17 00:00:00 2001 From: Seyon Sivarajah Date: Tue, 19 Sep 2023 15:36:57 +0100 Subject: [PATCH 1/3] feat: add `new_array` operation to prelude --- src/extension/prelude.rs | 55 +++++++++++++++++++++++++++++++++++++++- 1 file changed, 54 insertions(+), 1 deletion(-) diff --git a/src/extension/prelude.rs b/src/extension/prelude.rs index 210b83fac..12cc9b689 100644 --- a/src/extension/prelude.rs +++ b/src/extension/prelude.rs @@ -7,7 +7,7 @@ use crate::{ extension::{ExtensionId, TypeDefBound}, types::{ type_param::{TypeArg, TypeParam}, - CustomCheckFailure, CustomType, Type, TypeBound, + CustomCheckFailure, CustomType, FunctionType, Type, TypeBound, }, values::{CustomConst, KnownTypeConst}, Extension, @@ -39,6 +39,23 @@ lazy_static! { ) .unwrap(); + prelude + .add_op_custom_sig_simple( + SmolStr::new_inline("new_array"), + "Create a new array from elements".to_string(), + vec![TypeParam::Type(TypeBound::Any), TypeParam::max_nat()], + |args: &[TypeArg]| { + let [TypeArg::Type { ty }, TypeArg::BoundedNat { n }] = args else { + panic!("should have been checked already.") + }; + Ok(FunctionType::new( + vec![ty.clone(); *n as usize], + vec![new_array(ty.clone(), *n)], + )) + }, + ) + .unwrap(); + prelude .add_type( SmolStr::new_inline("qubit"), @@ -117,3 +134,39 @@ impl CustomConst for ConstUsize { impl KnownTypeConst for ConstUsize { const TYPE: CustomType = USIZE_CUSTOM_T; } + +#[cfg(test)] +mod test { + use crate::{ + builder::{DFGBuilder, Dataflow, DataflowHugr}, + extension::EMPTY_REG, + ops::LeafOp, + }; + + use super::*; + + #[test] + /// Test building a HUGR involving a new_array operation. + fn test_new_array() { + let mut b = DFGBuilder::new(FunctionType::new( + vec![QB_T, QB_T], + vec![new_array(QB_T, 2)], + )) + .unwrap(); + + let [q1, q2] = b.input_wires_arr(); + + let op: LeafOp = PRELUDE + .instantiate_extension_op( + "new_array", + vec![TypeArg::Type { ty: QB_T }, TypeArg::BoundedNat { n: 2 }], + &EMPTY_REG, + ) + .unwrap() + .into(); + + let out = b.add_dataflow_op(op, [q1, q2]).unwrap(); + + b.finish_prelude_hugr_with_outputs(out.outputs()).unwrap(); + } +} From 4ba63dedd281d3eb299abd5478da2a28a8f8cdb3 Mon Sep 17 00:00:00 2001 From: Seyon Sivarajah Date: Wed, 20 Sep 2023 11:21:45 +0100 Subject: [PATCH 2/3] refactor!: op creation in to `new_array_op` and rename `new_array` -> `array_type` --- src/extension/prelude.rs | 41 ++++++++++++++++++++++------------------ src/types/serialize.rs | 4 ++-- 2 files changed, 25 insertions(+), 20 deletions(-) diff --git a/src/extension/prelude.rs b/src/extension/prelude.rs index 12cc9b689..48ceb37c5 100644 --- a/src/extension/prelude.rs +++ b/src/extension/prelude.rs @@ -5,6 +5,7 @@ use smol_str::SmolStr; use crate::{ extension::{ExtensionId, TypeDefBound}, + ops::LeafOp, types::{ type_param::{TypeArg, TypeParam}, CustomCheckFailure, CustomType, FunctionType, Type, TypeBound, @@ -13,7 +14,7 @@ use crate::{ Extension, }; -use super::ExtensionRegistry; +use super::{ExtensionRegistry, EMPTY_REG}; /// Name of prelude extension. pub const PRELUDE_ID: ExtensionId = ExtensionId::new_unchecked("prelude"); @@ -41,7 +42,7 @@ lazy_static! { prelude .add_op_custom_sig_simple( - SmolStr::new_inline("new_array"), + SmolStr::new_inline(NEW_ARRAY_OP_ID), "Create a new array from elements".to_string(), vec![TypeParam::Type(TypeBound::Any), TypeParam::max_nat()], |args: &[TypeArg]| { @@ -50,7 +51,7 @@ lazy_static! { }; Ok(FunctionType::new( vec![ty.clone(); *n as usize], - vec![new_array(ty.clone(), *n)], + vec![array_type(ty.clone(), *n)], )) }, ) @@ -88,7 +89,7 @@ pub const USIZE_T: Type = Type::new_extension(USIZE_CUSTOM_T); pub const BOOL_T: Type = Type::new_simple_predicate(2); /// Initialize a new array of type `typ` of length `size` -pub fn new_array(typ: Type, size: u64) -> Type { +pub fn array_type(typ: Type, size: u64) -> Type { let array_def = PRELUDE.get_type("array").unwrap(); let custom_t = array_def .instantiate_concrete(vec![ @@ -99,6 +100,21 @@ pub fn new_array(typ: Type, size: u64) -> Type { Type::new_extension(custom_t) } +/// Name of the operation in the prelude for creating new arrays. +pub const NEW_ARRAY_OP_ID: &str = "new_array"; + +/// Initialize a new array op of type `typ` of length `size` +pub fn new_array_op(typ: Type, size: u64) -> LeafOp { + PRELUDE + .instantiate_extension_op( + NEW_ARRAY_OP_ID, + vec![TypeArg::Type { ty: typ }, TypeArg::BoundedNat { n: size }], + &EMPTY_REG, + ) + .unwrap() + .into() +} + pub(crate) const ERROR_TYPE: Type = Type::new_extension(CustomType::new_simple( smol_str::SmolStr::new_inline("error"), PRELUDE_ID, @@ -137,11 +153,7 @@ impl KnownTypeConst for ConstUsize { #[cfg(test)] mod test { - use crate::{ - builder::{DFGBuilder, Dataflow, DataflowHugr}, - extension::EMPTY_REG, - ops::LeafOp, - }; + use crate::builder::{DFGBuilder, Dataflow, DataflowHugr}; use super::*; @@ -150,20 +162,13 @@ mod test { fn test_new_array() { let mut b = DFGBuilder::new(FunctionType::new( vec![QB_T, QB_T], - vec![new_array(QB_T, 2)], + vec![array_type(QB_T, 2)], )) .unwrap(); let [q1, q2] = b.input_wires_arr(); - let op: LeafOp = PRELUDE - .instantiate_extension_op( - "new_array", - vec![TypeArg::Type { ty: QB_T }, TypeArg::BoundedNat { n: 2 }], - &EMPTY_REG, - ) - .unwrap() - .into(); + let op = new_array_op(QB_T, 2); let out = b.add_dataflow_op(op, [q1, q2]).unwrap(); diff --git a/src/types/serialize.rs b/src/types/serialize.rs index a3cfe0867..f70cc57b0 100644 --- a/src/types/serialize.rs +++ b/src/types/serialize.rs @@ -4,7 +4,7 @@ use super::custom::CustomType; use super::FunctionType; -use crate::extension::prelude::{new_array, QB_T, USIZE_T}; +use crate::extension::prelude::{array_type, QB_T, USIZE_T}; use crate::ops::AliasDecl; use crate::types::primitive::PrimType; @@ -51,7 +51,7 @@ impl From for Type { SerSimpleType::G(sig) => Type::new_function(*sig), SerSimpleType::Tuple { inner } => Type::new_tuple(inner), SerSimpleType::Sum(sum) => sum.into(), - SerSimpleType::Array { inner, len } => new_array((*inner).into(), len), + SerSimpleType::Array { inner, len } => array_type((*inner).into(), len), SerSimpleType::Opaque(custom) => Type::new_extension(custom), SerSimpleType::Alias(a) => Type::new_alias(a), } From 49cb18a30c65508589cf6e52c543b28f3134a0f5 Mon Sep 17 00:00:00 2001 From: Seyon Sivarajah Date: Mon, 25 Sep 2023 10:41:19 +0100 Subject: [PATCH 3/3] clearer variable names --- src/extension/prelude.rs | 15 +++++++++------ 1 file changed, 9 insertions(+), 6 deletions(-) diff --git a/src/extension/prelude.rs b/src/extension/prelude.rs index 48ceb37c5..d01ebc149 100644 --- a/src/extension/prelude.rs +++ b/src/extension/prelude.rs @@ -88,12 +88,12 @@ pub const USIZE_T: Type = Type::new_extension(USIZE_CUSTOM_T); /// Boolean type - Sum of two units. pub const BOOL_T: Type = Type::new_simple_predicate(2); -/// Initialize a new array of type `typ` of length `size` -pub fn array_type(typ: Type, size: u64) -> Type { +/// Initialize a new array of element type `element_ty` of length `size` +pub fn array_type(element_ty: Type, size: u64) -> Type { let array_def = PRELUDE.get_type("array").unwrap(); let custom_t = array_def .instantiate_concrete(vec![ - TypeArg::Type { ty: typ }, + TypeArg::Type { ty: element_ty }, TypeArg::BoundedNat { n: size }, ]) .unwrap(); @@ -103,12 +103,15 @@ pub fn array_type(typ: Type, size: u64) -> Type { /// Name of the operation in the prelude for creating new arrays. pub const NEW_ARRAY_OP_ID: &str = "new_array"; -/// Initialize a new array op of type `typ` of length `size` -pub fn new_array_op(typ: Type, size: u64) -> LeafOp { +/// Initialize a new array op of element type `element_ty` of length `size` +pub fn new_array_op(element_ty: Type, size: u64) -> LeafOp { PRELUDE .instantiate_extension_op( NEW_ARRAY_OP_ID, - vec![TypeArg::Type { ty: typ }, TypeArg::BoundedNat { n: size }], + vec![ + TypeArg::Type { ty: element_ty }, + TypeArg::BoundedNat { n: size }, + ], &EMPTY_REG, ) .unwrap()