From 63beb784c8915ce84f21e5950801bfaf88639591 Mon Sep 17 00:00:00 2001 From: Agustin Borgna Date: Mon, 17 Jun 2024 13:46:04 +0100 Subject: [PATCH] Bring in cfg extraction from the pytket->guppy PR --- Cargo.lock | 317 +++++-------------------------- tket2/src/circuit.rs | 18 +- tket2/src/circuit/extract_dfg.rs | 111 +++++++++++ 3 files changed, 161 insertions(+), 285 deletions(-) create mode 100644 tket2/src/circuit/extract_dfg.rs diff --git a/Cargo.lock b/Cargo.lock index 58f04526..9aa003d2 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -179,9 +179,9 @@ checksum = "37b2a672a2cb129a2e41c10b1224bb368f9f37a2b16b612598138befd7b37eb5" [[package]] name = "cc" -version = "1.0.99" +version = "1.0.98" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "96c51067fd44124faa7f870b4b1c969379ad32b2ba805aa959430ceaa384f695" +checksum = "41c270e7540d725e65ac7f1b212ac8ce349719624d7bcff99f8e2e488e8cf03f" [[package]] name = "cesu8" @@ -510,17 +510,6 @@ dependencies = [ "syn 2.0.66", ] -[[package]] -name = "displaydoc" -version = "0.2.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "487585f4d0c6655fe74905e2504d8ad6908e4db67f744eb140876906c2f3175d" -dependencies = [ - "proc-macro2", - "quote", - "syn 2.0.66", -] - [[package]] name = "downcast-rs" version = "1.2.1" @@ -835,134 +824,14 @@ dependencies = [ "cc", ] -[[package]] -name = "icu_collections" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "db2fa452206ebee18c4b5c2274dbf1de17008e874b4dc4f0aea9d01ca79e4526" -dependencies = [ - "displaydoc", - "yoke", - "zerofrom", - "zerovec", -] - -[[package]] -name = "icu_locid" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "13acbb8371917fc971be86fc8057c41a64b521c184808a698c02acc242dbf637" -dependencies = [ - "displaydoc", - "litemap", - "tinystr", - "writeable", - "zerovec", -] - -[[package]] -name = "icu_locid_transform" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "01d11ac35de8e40fdeda00d9e1e9d92525f3f9d887cdd7aa81d727596788b54e" -dependencies = [ - "displaydoc", - "icu_locid", - "icu_locid_transform_data", - "icu_provider", - "tinystr", - "zerovec", -] - -[[package]] -name = "icu_locid_transform_data" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fdc8ff3388f852bede6b579ad4e978ab004f139284d7b28715f773507b946f6e" - -[[package]] -name = "icu_normalizer" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "19ce3e0da2ec68599d193c93d088142efd7f9c5d6fc9b803774855747dc6a84f" -dependencies = [ - "displaydoc", - "icu_collections", - "icu_normalizer_data", - "icu_properties", - "icu_provider", - "smallvec", - "utf16_iter", - "utf8_iter", - "write16", - "zerovec", -] - -[[package]] -name = "icu_normalizer_data" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f8cafbf7aa791e9b22bec55a167906f9e1215fd475cd22adfcf660e03e989516" - -[[package]] -name = "icu_properties" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1f8ac670d7422d7f76b32e17a5db556510825b29ec9154f235977c9caba61036" -dependencies = [ - "displaydoc", - "icu_collections", - "icu_locid_transform", - "icu_properties_data", - "icu_provider", - "tinystr", - "zerovec", -] - -[[package]] -name = "icu_properties_data" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "67a8effbc3dd3e4ba1afa8ad918d5684b8868b3b26500753effea8d2eed19569" - -[[package]] -name = "icu_provider" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6ed421c8a8ef78d3e2dbc98a973be2f3770cb42b606e3ab18d6237c4dfde68d9" -dependencies = [ - "displaydoc", - "icu_locid", - "icu_provider_macros", - "stable_deref_trait", - "tinystr", - "writeable", - "yoke", - "zerofrom", - "zerovec", -] - -[[package]] -name = "icu_provider_macros" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1ec89e9337638ecdc08744df490b221a7399bf8d164eb52a665454e60e075ad6" -dependencies = [ - "proc-macro2", - "quote", - "syn 2.0.66", -] - [[package]] name = "idna" -version = "1.0.0" +version = "0.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4716a3a0933a1d01c2f72450e89596eb51dd34ef3c211ccd875acdf1f8fe47ed" +checksum = "634d9b1461af396cad843f47fdba5597a4f9e6ddd4bfb6ff5d85028c25cb12f6" dependencies = [ - "icu_normalizer", - "icu_properties", - "smallvec", - "utf8_iter", + "unicode-bidi", + "unicode-normalization", ] [[package]] @@ -1077,12 +946,6 @@ version = "0.2.155" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "97b3888a4aecf77e811145cadf6eef5901f4782c53886191b2f693f24761847c" -[[package]] -name = "litemap" -version = "0.7.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "643cb0b8d4fcc284004d5fd0d67ccf61dfffadb7f75e1e71bc420f4688a3a704" - [[package]] name = "lock_api" version = "0.4.12" @@ -1534,9 +1397,9 @@ dependencies = [ [[package]] name = "regex" -version = "1.10.5" +version = "1.10.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b91213439dad192326a0d7c6ee3955910425f441d7038e0d6933b0aec5c4517f" +checksum = "c117dbdfde9c8308975b6a18d71f3f385c89461f7b3fb054288ecf2a2058ba4c" dependencies = [ "aho-corasick", "memchr", @@ -1546,9 +1409,9 @@ dependencies = [ [[package]] name = "regex-automata" -version = "0.4.7" +version = "0.4.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "38caf58cc5ef2fed281f89292ef23f6365465ed9a41b7a7754eb4e26496c92df" +checksum = "86b83b8b9847f9bf95ef68afb0b8e6cdb80f498442f5179a29fad448fcc1eaea" dependencies = [ "aho-corasick", "memchr", @@ -1557,9 +1420,9 @@ dependencies = [ [[package]] name = "regex-syntax" -version = "0.8.4" +version = "0.8.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7a66a03ae7c801facd77a29370b4faec201768915ac14a721ba36f20bc9c209b" +checksum = "adad44e29e4c806119491a7f06f03de4d1af22c3a680dd47f1e6e179439d1f56" [[package]] name = "relative-path" @@ -1746,12 +1609,6 @@ dependencies = [ "serde", ] -[[package]] -name = "stable_deref_trait" -version = "1.2.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a8f112729512f8e442d81f95a8a7ddf2b7c6b8a1a6f509a95864142b30cab2d3" - [[package]] name = "strsim" version = "0.11.1" @@ -1802,17 +1659,6 @@ dependencies = [ "unicode-ident", ] -[[package]] -name = "synstructure" -version = "0.13.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c8af7666ab7b6390ab78131fb5b0fce11d6b7a6951602017c35fa82800708971" -dependencies = [ - "proc-macro2", - "quote", - "syn 2.0.66", -] - [[package]] name = "tap" version = "1.0.1" @@ -1906,16 +1752,6 @@ dependencies = [ "time-core", ] -[[package]] -name = "tinystr" -version = "0.7.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9117f5d4db391c1cf6927e7bea3db74b9a1c1add8f7eda9ffd5364f40f57b82f" -dependencies = [ - "displaydoc", - "zerovec", -] - [[package]] name = "tinytemplate" version = "1.2.1" @@ -1926,6 +1762,21 @@ dependencies = [ "serde_json", ] +[[package]] +name = "tinyvec" +version = "1.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87cc5ceb3875bb20c2890005a4e226a4651264a5c75edb2421b52861a0a0cb50" +dependencies = [ + "tinyvec_macros", +] + +[[package]] +name = "tinyvec_macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" + [[package]] name = "tket-json-rs" version = "0.4.1" @@ -2111,12 +1962,27 @@ dependencies = [ "syn 2.0.66", ] +[[package]] +name = "unicode-bidi" +version = "0.3.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08f95100a766bf4f8f28f90d77e0a5461bbdb219042e7679bebe79004fed8d75" + [[package]] name = "unicode-ident" version = "1.0.12" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b" +[[package]] +name = "unicode-normalization" +version = "0.1.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a56d1686db2308d901306f92a263857ef59ea39678a5458e7cb17f01415101f5" +dependencies = [ + "tinyvec", +] + [[package]] name = "unindent" version = "0.2.3" @@ -2131,9 +1997,9 @@ checksum = "673aac59facbab8a9007c7f6108d11f63b603f7cabff99fabf650fea5c32b861" [[package]] name = "url" -version = "2.5.1" +version = "2.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f7c25da092f0a868cdf09e8674cd3b7ef3a7d92a24253e663a2fb85e2496de56" +checksum = "31e6302e3bb753d46e83516cae55ae196fc0c309407cf11ab35cc51a4c2a4633" dependencies = [ "form_urlencoded", "idna", @@ -2146,29 +2012,17 @@ version = "2.1.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "daf8dba3b7eb870caf1ddeed7bc9d2a049f3cfdfae7cb521b087cc33ae4c49da" -[[package]] -name = "utf16_iter" -version = "1.0.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c8232dd3cdaed5356e0f716d285e4b40b932ac434100fe9b7e0e8e935b9e6246" - [[package]] name = "utf8-width" version = "0.1.7" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "86bd8d4e895da8537e5315b8254664e6b769c4ff3db18321b297a1e7004392e3" -[[package]] -name = "utf8_iter" -version = "1.0.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b6c140620e7ffbb22c2dee59cafe6084a59b5ffc27a8859a5f0d494b5d52b6be" - [[package]] name = "utf8parse" -version = "0.2.2" +version = "0.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "06abde3611657adf66d383f00b093d7faecc7fa57071cce2578660c9f1010821" +checksum = "711b9620af191e0cdc7468a8d14e709c3dcdb115b36f838e601583af800a370a" [[package]] name = "uuid" @@ -2456,18 +2310,6 @@ version = "0.52.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bec47e5bfd1bff0eeaf6d8b485cc1074891a197ab4225d504cb7a1ab88b02bf0" -[[package]] -name = "write16" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d1890f4022759daae28ed4fe62859b1236caebfc61ede2f63ed4e695f3f6d936" - -[[package]] -name = "writeable" -version = "0.5.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1e9df38ee2d2c3c5948ea468a8406ff0db0b29ae1ffde1bcf20ef305bcc95c51" - [[package]] name = "wyz" version = "0.5.1" @@ -2476,70 +2318,3 @@ checksum = "05f360fc0b24296329c78fda852a1e9ae82de9cf7b27dae4b7f62f118f77b9ed" dependencies = [ "tap", ] - -[[package]] -name = "yoke" -version = "0.7.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6c5b1314b079b0930c31e3af543d8ee1757b1951ae1e1565ec704403a7240ca5" -dependencies = [ - "serde", - "stable_deref_trait", - "yoke-derive", - "zerofrom", -] - -[[package]] -name = "yoke-derive" -version = "0.7.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "28cc31741b18cb6f1d5ff12f5b7523e3d6eb0852bbbad19d73905511d9849b95" -dependencies = [ - "proc-macro2", - "quote", - "syn 2.0.66", - "synstructure", -] - -[[package]] -name = "zerofrom" -version = "0.1.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "91ec111ce797d0e0784a1116d0ddcdbea84322cd79e5d5ad173daeba4f93ab55" -dependencies = [ - "zerofrom-derive", -] - -[[package]] -name = "zerofrom-derive" -version = "0.1.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0ea7b4a3637ea8669cedf0f1fd5c286a17f3de97b8dd5a70a6c167a1730e63a5" -dependencies = [ - "proc-macro2", - "quote", - "syn 2.0.66", - "synstructure", -] - -[[package]] -name = "zerovec" -version = "0.10.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bb2cc8827d6c0994478a15c53f374f46fbd41bea663d809b14744bc42e6b109c" -dependencies = [ - "yoke", - "zerofrom", - "zerovec-derive", -] - -[[package]] -name = "zerovec-derive" -version = "0.10.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "97cf56601ee5052b4417d90c8755c6683473c926039908196cf35d99f893ebe7" -dependencies = [ - "proc-macro2", - "quote", - "syn 2.0.66", -] diff --git a/tket2/src/circuit.rs b/tket2/src/circuit.rs index ed27d58c..afa9d5a2 100644 --- a/tket2/src/circuit.rs +++ b/tket2/src/circuit.rs @@ -2,6 +2,7 @@ pub mod command; pub mod cost; +mod extract_dfg; mod hash; pub mod units; @@ -15,11 +16,10 @@ use itertools::Either::{Left, Right}; use hugr::hugr::hugrmut::HugrMut; use hugr::hugr::NodeType; use hugr::ops::dataflow::IOTrait; -use hugr::ops::{Input, NamedOp, OpParent, OpTag, OpTrait, Output, DFG}; +use hugr::ops::{Input, NamedOp, OpParent, OpTag, OpTrait, Output}; use hugr::types::{FunctionType, PolyFuncType}; use hugr::{Hugr, PortIndex}; use hugr::{HugrView, OutgoingPort}; -use hugr_core::hugr::internal::HugrMutInternals; use itertools::Itertools; use thiserror::Error; @@ -308,7 +308,7 @@ impl Circuit { /// Regions that are not descendants of the parent node are not included in the new HUGR. /// This may invalidate calls to functions defined elsewhere. Make sure to inline any /// external functions before calling this method. - pub fn extract_dfg(self) -> Result, CircuitMutError> + pub fn extract_dfg(&self) -> Result, CircuitMutError> where T: ExtractHugr, { @@ -319,17 +319,7 @@ impl Circuit { .expect("Circuit parent was not a dataflow container."); view.extract_hugr().into() }; - - // Replace the parent node with a DFG node, if necessary. - let nodetype = circ.hugr.get_nodetype(circ.parent()); - if !matches!(nodetype.op(), OpType::DFG(_)) { - let dfg = DFG { - signature: circ.circuit_signature(), - }; - let input_extensions = nodetype.input_extensions().cloned(); - let nodetype = NodeType::new(OpType::DFG(dfg), input_extensions); - circ.hugr.replace_op(circ.parent(), nodetype)?; - } + extract_dfg::rewrite_into_dfg(&mut circ)?; Ok(circ) } } diff --git a/tket2/src/circuit/extract_dfg.rs b/tket2/src/circuit/extract_dfg.rs new file mode 100644 index 00000000..d7deb6de --- /dev/null +++ b/tket2/src/circuit/extract_dfg.rs @@ -0,0 +1,111 @@ +//! Internal implementation of `Circuit::extract_dfg`. + +use hugr::hugr::hugrmut::HugrMut; +use hugr::hugr::NodeType; +use hugr::ops::{OpTrait, OpType, Output, DFG}; +use hugr::types::{FunctionType, SumType, TypeEnum}; +use hugr::HugrView; +use hugr_core::hugr::internal::HugrMutInternals; +use itertools::Itertools; + +use crate::{Circuit, CircuitMutError}; + +/// Internal method used by [`extract_dfg`] to replace the parent node with a DFG node. +pub(super) fn rewrite_into_dfg(circ: &mut Circuit) -> Result<(), CircuitMutError> { + // Replace the parent node with a DFG node, if necessary. + let old_optype = circ.hugr.get_optype(circ.parent()); + if matches!(old_optype, OpType::DFG(_)) { + return Ok(()); + } + + // If the region was a cfg with a single successor, unpack the output sum type. + let signature = circ.circuit_signature(); + let signature = match old_optype { + OpType::DataflowBlock(_) => remove_cfg_empty_output_tuple(circ, signature)?, + _ => signature, + }; + + let dfg = DFG { signature }; + let nodetype = circ.hugr.get_nodetype(circ.parent()); + let input_extensions = nodetype.input_extensions().cloned(); + let nodetype = NodeType::new(OpType::DFG(dfg), input_extensions); + circ.hugr.replace_op(circ.parent(), nodetype)?; + + Ok(()) +} + +/// Remove an empty sum from a cfg's DataflowBlock output node, if possible. +/// +/// Bails out if it cannot match the exact pattern, without modifying the +/// circuit. +/// +/// TODO: This function is specialized towards the specific functions generated +/// by guppy. We should generalize this to work with non-empty sum types +/// when possible. +fn remove_cfg_empty_output_tuple( + circ: &mut Circuit, + signature: FunctionType, +) -> Result { + let sig = signature; + let parent = circ.parent(); + + let output_node = circ.output_node(); + let output_nodetype = circ.hugr.get_nodetype(output_node).clone(); + let output_op = output_nodetype.op(); + + let output_sig = output_op + .dataflow_signature() + .expect("Exit node with no dataflow signature."); + + // Only remove the port if it's an empty sum type. + if !matches!( + output_sig.input[0].as_type_enum(), + TypeEnum::Sum(SumType::Unit { size: 1 }) + ) { + return Ok(sig); + } + + // There must be a zero-sized `Tag` operation. + let Some((tag_node, _)) = circ.hugr.single_linked_output(output_node, 0) else { + return Ok(sig); + }; + + let tag_op = circ.hugr.get_optype(tag_node); + if !matches!(tag_op, OpType::Tag(_)) { + return Ok(sig); + } + + // Hacky replacement for the nodes. + + // Drop the old nodes + let hugr = circ.hugr_mut(); + let input_neighs = hugr.all_linked_outputs(output_node).skip(1).collect_vec(); + + hugr.remove_node(output_node); + hugr.remove_node(tag_node); + + // Add a new output node. + let new_types = output_sig.input[1..].to_vec(); + let new_op = Output { + types: new_types.clone().into(), + }; + let new_node = hugr.add_node_with_parent( + parent, + NodeType::new( + new_op, + output_nodetype + .input_extensions() + .cloned() + .unwrap_or_default(), + ), + ); + + // Reconnect the outputs. + for (i, (neigh, port)) in input_neighs.into_iter().enumerate() { + hugr.connect(neigh, port, new_node, i); + } + + // Return the updated circuit signature. + let sig = FunctionType::new(sig.input, new_types); + Ok(sig) +}