{-# LANGUAGE Rank2Types #-}
{-# LANGUAGE DeriveDataTypeable #-}
{-# LANGUAGE MultiParamTypeClasses #-}
-- | This module provides functions for defining general-purpose
-- transformations on low-level circuits. The uses of this include:
--
-- * gate transformations, where a whole circuit is transformed by
-- replacing each kind of gate with another gate or circuit;
--
-- * error correcting codes, where a whole circuit is transformed
-- replacing each qubit by some fixed number of qubits, and each gate
-- by a circuit; and
--
-- * simulations, where a whole circuit is mapped to a semantic
-- function by specifying a semantic function for each gate.
--
-- The interface is designed to allow the programmer to specify new
-- transformers easily. To define a specific transformation, the
-- programmer has to specify only four pieces of information:
--
-- * A type /a/=⟦Qubit⟧, to serve as a semantic domain for qubits.
--
-- * A type /b/=⟦Bit⟧, to serve as a semantic domain for bits.
--
-- * A monad /m/. This is to allow translations to have side effects
-- if desired; one can use the identity monad otherwise.
--
-- * For every gate /G/, a corresponding semantic function ⟦/G/⟧. The
-- type of this function depends on what kind of gate /G/ is. For example:
--
-- @
-- If /G/ :: Qubit -> Circ Qubit, then ⟦/G/⟧ :: /a/ -> /m/ /a/.
-- If /G/ :: (Qubit, Bit) -> Circ (Bit, Bit), then ⟦/G/⟧ :: (/a/, /b/) -> /m/ (/b/, /b/).
-- @
--
-- The programmer provides this information by defining a function of
-- type 'Transformer' /m/ /a/ /b/. See <#Transformers> below. Once a
-- particular transformer has been defined, it can then be applied to
-- entire circuits. For example, for a circuit with 1 inputs and 2
-- outputs:
--
-- @
-- If /C/ :: Qubit -> (Bit, Qubit), then ⟦/C/⟧ :: /a/ -> /m/ (/b/, /a/).
-- @
-- ----------------------------------------------------------------------
-- Grammar note for developers: a "transformer" does a
-- "transformation" by "transforming" gates. We use "transform" as a
-- verb, "transformation" to describe the process of transforming, and
-- "transformer" for the code that describes or does the transformation.
--
-- I had initially used the words "iteration", "translation",
-- "transform", "transformation", "interpretation", and "semantics"
-- interchangeably, which was a huge linguistic mess.
module Quipper.Internal.Transformer where
-- import other Quipper stuff
import Quipper.Internal.Circuit
import Quipper.Utils.Auxiliary
-- import other stuff
import Control.Monad
import Control.Monad.State
import Data.Map (Map)
import qualified Data.Map as Map
import qualified Data.IntMap as IntMap
import Data.Typeable
-- ======================================================================
-- * An example transformer
--
-- $EXAMPLE
--
-- The following is a short but complete example of how to write and
-- use a simple transformer. As usual, we start by importing Quipper:
--
-- > import Quipper
--
-- We will write a transformer called @sample_transformer@, which maps
-- every swap gate to a sequence of three controlled-not gates, and
-- leaves all other gates unchanged. For convenience, Quipper
-- pre-defines an 'Quipper.identity_transformer', which can be used as
-- a catch-all clause to take care of all the gates that don't need to
-- be rewritten.
--
-- > mytransformer :: Transformer Circ Qubit Bit
-- > mytransformer (T_QGate "swap" 2 0 _ ncf f) = f $
-- > \[q0, q1] [] ctrls -> do
-- > without_controls_if ncf $ do
-- > with_controls ctrls $ do
-- > qnot_at q0 `controlled` q1
-- > qnot_at q1 `controlled` q0
-- > qnot_at q0 `controlled` q1
-- > return ([q0, q1], [], ctrls)
-- > mytransformer g = identity_transformer g
--
-- Note how Quipper syntax has been used to define the replacement
-- circuit, consisting of three controlled-not gates. Also, since the
-- original swap gate may have been controlled, we have added the
-- additional controls with a 'Quipper.with_controls' operator.
--
-- To try this out, we define some random circuit using swap gates:
--
-- > mycirc a b c d = do
-- > swap_at a b
-- > hadamard_at b
-- > swap_at b c `controlled` [a, d]
-- > hadamard_at c
-- > swap_at c d
--
-- To apply the transformer to this circuit, we use the generic
-- operator 'Quipper.transform_generic':
--
-- > mycirc2 = transform_generic mytransformer mycirc
--
-- Finally, we use a @main@ function to display the original circuit
-- and then the transformed one:
--
-- > main = do
-- > print_simple Preview mycirc
-- > print_simple Preview mycirc2
-- ======================================================================
-- * Bindings
-- $bindings
--
-- We introduce the notion of a /binding/ as a low-level way to
-- describe functions of varying arities. A binding assigns a value to
-- a wire in a circuit (much like a \"valuation\" in logic or semantics
-- assigns values to variables).
--
-- To iterate through a circuit, one will typically specify initial
-- bindings for the input wires. This encodes the input of the function
-- ⟦/C/⟧ mentioned in the introduction. The bindings are updated as
-- one passes through each gate. When the iteration is finished, the
-- final bindings assign a value to each output wire of the
-- circuit. This encodes the output of the function ⟦/C/⟧. Therefore,
-- the interpretation of a circuit is representable as a function from
-- bindings (of input wires) to bindings (of output wires), i.e., it
-- has the type ⟦/C/⟧ :: 'Bindings' /a/ /b/ -> 'Bindings' /a/ /b/.
-- | An /endpoint/ is either a /qubit/ or a /bit/. In a transformer,
-- we have ⟦B_Endpoint Qubit Bit⟧ = ⟦Qubit⟧ + ⟦Bit⟧. The type 'B_Endpoint'
-- /a/ /b/ is the same as 'Either' /a/ /b/, but we use more suggestive
-- field names.
data B_Endpoint a b =
Endpoint_Qubit a
| Endpoint_Bit b
deriving (Eq, Ord, Typeable, Show)
-- | A binding is a map from a set of wires to the disjoint union of
-- /a/ and /b/.
type Bindings a b = Map Wire (B_Endpoint a b)
-- | Return the list of bound wires from a binding.
wires_of_bindings :: Bindings a b -> [Wire]
wires_of_bindings = Map.keys
-- | The empty binding.
bindings_empty :: Bindings a b
bindings_empty = Map.empty
-- | Bind a wire to a value, and add it to the given bindings.
bind :: Wire -> B_Endpoint a b -> Bindings a b -> Bindings a b
bind r x bindings = Map.insert r x bindings
-- | Bind a qubit wire to a value, and add it to the given bindings.
bind_qubit_wire :: Wire -> a -> Bindings a b -> Bindings a b
bind_qubit_wire r x bindings = bind r (Endpoint_Qubit x) bindings
-- | Bind a bit wire to a value, and add it to the given bindings.
bind_bit_wire :: Wire -> b -> Bindings a b -> Bindings a b
bind_bit_wire r x bindings = bind r (Endpoint_Bit x) bindings
-- | Retrieve the value of a wire from the given bindings.
unbind :: Bindings a b -> Wire -> B_Endpoint a b
unbind bindings w = case Map.lookup w bindings of
Nothing -> error ("unbind: wire (" ++ show w ++ ") not in bindings: " ++ show (wires_of_bindings bindings))
Just a -> a
-- | Retrieve the value of a qubit wire from the given bindings.
-- Throws an error if the wire was bound to a classical bit.
unbind_qubit_wire :: Bindings a b -> Wire -> a
unbind_qubit_wire bindings w =
case unbind bindings w of
Endpoint_Qubit x -> x
Endpoint_Bit x -> error "Transformer error: expected a qubit, got a bit"
-- | Retrieve the value of a bit wire from the given bindings.
-- Throws an error if the wire was bound to a qubit.
unbind_bit_wire :: Bindings a b -> Wire -> b
unbind_bit_wire bindings w =
case unbind bindings w of
Endpoint_Bit x -> x
Endpoint_Qubit x -> error "Transformer error: expected a bit, got a qubit"
-- | Delete a wire from the given bindings.
bind_delete :: Wire -> Bindings a b -> Bindings a b
bind_delete r bindings = Map.delete r bindings
-- | Like 'bind', except bind a list of wires to a list of values. The
-- lists must be of the same length.
bind_list :: [Wire] -> [B_Endpoint a b] -> Bindings a b -> Bindings a b
bind_list ws xs bindings =
foldr (\ (w, x) -> bind w x) bindings (zip ws xs)
-- | Like 'bind_qubit_wire', except bind a list of qubit wires to a list of
-- values. The lists must be of the same length.
bind_qubit_wire_list :: [Wire] -> [a] -> Bindings a b -> Bindings a b
bind_qubit_wire_list ws xs bindings =
foldr (\ (w, x) -> bind_qubit_wire w x) bindings (zip ws xs)
-- | Like 'bind_bit_wire', except bind a list of bit wires to a list of
-- values. The lists must be of the same length.
bind_bit_wire_list :: [Wire] -> [b] -> Bindings a b -> Bindings a b
bind_bit_wire_list ws xs bindings =
foldr (\ (w, x) -> bind_bit_wire w x) bindings (zip ws xs)
-- | Like 'unbind', except retrieve a list of values.
unbind_list :: Bindings a b -> [Wire] -> [B_Endpoint a b]
unbind_list bindings ws =
map (unbind bindings) ws
-- | Like 'unbind_qubit_wire', except retrieve a list of values.
unbind_qubit_wire_list :: Bindings a b -> [Wire] -> [a]
unbind_qubit_wire_list bindings ws =
map (unbind_qubit_wire bindings) ws
-- | Like 'unbind_bit_wire', except retrieve a list of values.
unbind_bit_wire_list :: Bindings a b -> [Wire] -> [b]
unbind_bit_wire_list bindings ws =
map (unbind_bit_wire bindings) ws
-- | A list of signed values of type ⟦B_Endpoint⟧. This type is an
-- abbreviation defined for convenience.
type Ctrls a b = [Signed (B_Endpoint a b)]
-- | Given a list of signed wires (controls), and a list of signed
-- values, make a bindings from the wires to the values. Ignore the signs.
bind_controls :: Controls -> Ctrls a b -> Bindings a b -> Bindings a b
bind_controls controls xs bindings =
bind_list (map from_signed controls) (map from_signed xs) bindings
-- | Like 'unbind', but retrieve binding for all wires in a list of
-- controls.
unbind_controls :: Bindings a b -> Controls -> Ctrls a b
unbind_controls bindings c =
[Signed (unbind bindings w) b | Signed w b <- c ]
-- $transformers_anchor #Transformers#
-- ----------------------------------------------------------------------
-- * Transformers
-- $transformers
--
-- The types 'T_Gate' and 'Transformer' are at the heart of the
-- circuit transformer functionality. Their purpose is to give a
-- concise syntax in which to express semantic functions for gates. As
-- mentioned in the introduction, the programmer needs to specify two
-- type /a/ and /b/, a monad /m/, and a semantic function for each
-- gate. With the T_Gate' and 'Transformer' types, the definition
-- takes the following form:
--
-- > my_transformer :: Transformer m a b
-- > my_transformer (T_Gate1 <parameters> f) = f $ <semantic function for gate 1>
-- > my_transformer (T_Gate2 <parameters> f) = f $ <semantic function for gate 2>
-- > my_transformer (T_Gate3 <parameters> f) = f $ <semantic function for gate 3>
-- > ...
--
-- The type 'T_Gate' is very higher-order, involving a function /f/
-- that consumes the semantic function for each gate. The reason for
-- this higher-orderness is that the semantic functions for different
-- gates may have different types.
--
-- This higher-orderness makes the 'T_Gate' mechanism hard to read,
-- but easy to use. Effectively we only have to write lengthy and
-- messy code once and for all, rather than once for each transformer.
-- In particular, all the required low-level bindings and unbindings
-- can be handled by general-purpose code, and do not need to clutter
-- each transformer.
-- | The type 'T_Gate' is used to define case distinctions over gates
-- in the definition of transformers. For each kind of gate /X/, it
-- contains a constructor of the form @(T_X f)@. Here, /X/ identifies
-- the gate, and /f/ is a higher-order function to pass the
-- translation of /X/ to.
-- Implementation note: in the future, perhaps we can also add two
-- variants of this type: one that is specialized to the "simple"
-- case, where the semantics functions are assumed not to modify the
-- controls; another that is specialized to m = Id. This would make
-- the definition of most circuit transformers look less cluttered.
data T_Gate m a b x =
T_QGate String Int Int InverseFlag NoControlFlag (([a] -> [a] -> Ctrls a b -> m ([a], [a], Ctrls a b)) -> x)
| T_QRot String Int Int InverseFlag Timestep NoControlFlag (([a] -> [a] -> Ctrls a b -> m ([a], [a], Ctrls a b)) -> x)
| T_GPhase Double NoControlFlag (([B_Endpoint a b] -> Ctrls a b -> m (Ctrls a b)) -> x)
| T_CNot NoControlFlag ((b -> Ctrls a b -> m (b, Ctrls a b)) -> x)
| T_CGate String NoControlFlag (([b] -> m (b, [b])) -> x)
| T_CGateInv String NoControlFlag ((b -> [b] -> m [b]) -> x)
| T_CSwap NoControlFlag ((b -> b -> Ctrls a b -> m (b, b, Ctrls a b)) -> x)
| T_QPrep NoControlFlag ((b -> m a) -> x)
| T_QUnprep NoControlFlag ((a -> m b) -> x)
| T_QInit Bool NoControlFlag (m a -> x)
| T_CInit Bool NoControlFlag (m b -> x)
| T_QTerm Bool NoControlFlag ((a -> m ()) -> x)
| T_CTerm Bool NoControlFlag ((b -> m ()) -> x)
| T_QMeas ((a -> m b) -> x)
| T_QDiscard ((a -> m ()) -> x)
| T_CDiscard ((b -> m ()) -> x)
| T_DTerm Bool ((b -> m ()) -> x)
| T_Subroutine BoxId InverseFlag NoControlFlag ControllableFlag [Wire] Arity [Wire] Arity RepeatFlag ((Namespace -> [B_Endpoint a b] -> Ctrls a b -> m ([B_Endpoint a b], Ctrls a b)) -> x)
| T_Comment String InverseFlag (([(B_Endpoint a b, String)] -> m ()) -> x)
-- Make 'T_Gate' an instance of 'Show', to enable transformers to
-- produce better error messages about unimplemented gates etc.
instance Show (T_Gate m a b x) where
show (T_QGate name n m inv ncf f) = "QGate[" ++ name ++ "," ++ show n ++ "," ++ show m ++ "]" ++ optional inv "*"
show (T_QRot name n m inv t ncf f) = "QRot[" ++ name ++ "," ++ show t ++ "," ++ show n ++ "," ++ show m ++ "]" ++ optional inv "*"
show (T_GPhase t ncf f) = "GPhase[" ++ show t ++ "]"
show (T_CNot ncf f) = "CNot"
show (T_CGate n ncf f) = "CGate[" ++ n ++ "]"
show (T_CGateInv n ncf f) = "CGate[" ++ n ++ "]*"
show (T_CSwap ncf f) = "CSwap"
show (T_QPrep ncf f) = "QPrep"
show (T_QUnprep ncf f) = "QUnprep"
show (T_QInit b ncf f) = "QInit" ++ if b then "1" else "0"
show (T_CInit b ncf f) = "CInit" ++ if b then "1" else "0"
show (T_QTerm b ncf f) = "QTerm" ++ if b then "1" else "0"
show (T_CTerm b ncf f) = "CTerm" ++ if b then "1" else "0"
show (T_QMeas f) = "QMeas"
show (T_QDiscard f) = "QDiscard"
show (T_CDiscard f) = "CDiscard"
show (T_DTerm b f) = "DTerm" ++ if b then "1" else "0"
show (T_Subroutine n inv ncf scf ws a1 vs a2 rep f) = "Subroutine(x" ++ (show rep) ++ ")[" ++ show n ++ "]" ++ optional inv "*"
show (T_Comment n inv f) = "Comment[" ++ n ++ "]" ++ optional inv "*"
-- | A circuit transformer is specified by defining a function of type
-- 'Transformer' /m/ /a/ /b/. This involves specifying a monad /m/,
-- semantic domains /a/=⟦Qubit⟧ and /b/=⟦Bit⟧, and a semantic function
-- for each gate, like this:
--
-- > my_transformer :: Transformer m a b
-- > my_transformer (T_Gate1 <parameters> f) = f $ <semantic function for gate 1>
-- > my_transformer (T_Gate2 <parameters> f) = f $ <semantic function for gate 2>
-- > my_transformer (T_Gate3 <parameters> f) = f $ <semantic function for gate 3>
-- > ...
-- Implementation note: the use of \"forall\" in this type is to allow
-- some freedom in the return type of the continuation 'f' in the
-- definition of 'T_Gate'. This makes it easier, for example, to
-- compose transformers with other transformers. The use of \"forall\"
-- implies that any module that uses the 'Transformer' type may have
-- to declare the @Rank2Types@ language extension. This was not
-- required in GHC 7.4, but seems to be required in GHC 7.6.
type Transformer m a b = forall x . T_Gate m a b x -> x
-- | A \"binding transformer\" is a function from bindings to
-- bindings. The semantics of any gate or circuit is ultimately a
-- binding transformer, for some types /a/, /b/ and some monad /m/. We
-- introduce an abbreviation for this type primarily as a convenience
-- for the definition of 'bind_gate', but also because this type can
-- be completely hidden from user code.
type BT m a b = Bindings a b -> m (Bindings a b)
-- | Turn a 'Gate' into a 'T_Gate'. This is the function that actually
-- handles the explicit bindings/unbindings required for the inputs
-- and outputs of each gate. Effectively it gives a way, for each
-- gate, of turning a semantic function into a binding transformer.
-- Additionally, this function is passed a Namespace, so that the
-- semantic function for T_Subroutine can use it.
bind_gate :: Monad m => Namespace -> Gate -> T_Gate m a b (BT m a b)
bind_gate namespace gate = case gate of
QGate name inv ws vs c ncf -> T_QGate name n m inv ncf (list_binary ws vs c)
where
n = length ws
m = length vs
QRot name inv t ws vs c ncf -> T_QRot name n m inv t ncf (list_binary ws vs c)
where
n = length ws
m = length vs
GPhase t w c ncf -> T_GPhase t ncf (phase_ary w c)
CNot w c ncf -> T_CNot ncf (cunary w c)
CGate n w vs ncf -> T_CGate n ncf (cgate_ary w vs)
CGateInv n w vs ncf -> T_CGateInv n ncf (cgateinv_ary w vs)
CSwap w v c ncf -> T_CSwap ncf (binary_c w v c)
QPrep w ncf -> T_QPrep ncf (qprep_ary w)
QUnprep w ncf -> T_QUnprep ncf (qunprep_ary w)
QInit b w ncf -> T_QInit b ncf (qinit_ary w)
CInit b w ncf -> T_CInit b ncf (cinit_ary w)
QTerm b w ncf -> T_QTerm b ncf (qterm_ary w)
CTerm b w ncf -> T_CTerm b ncf (cterm_ary w)
QMeas w -> T_QMeas (qunprep_ary w)
QDiscard w -> T_QDiscard (qterm_ary w)
CDiscard w -> T_CDiscard (cterm_ary w)
DTerm b w -> T_DTerm b (cterm_ary w)
Subroutine n inv ws a1 vs a2 c ncf scf rep
-> T_Subroutine n inv ncf scf ws a1 vs a2 rep
(\f -> subroutine_ary ws vs c (f namespace))
Comment s inv ws -> T_Comment s inv (comment_ary ws)
where
unary :: Monad m => Wire -> Controls -> (a -> Ctrls a b -> m (a, Ctrls a b)) -> BT m a b
unary w c f bindings = do
let w' = unbind_qubit_wire bindings w
let c' = unbind_controls bindings c
(w'', c'') <- f w' c'
let bindings1 = bind_qubit_wire w w'' bindings
let bindings2 = bind_controls c c'' bindings1
return bindings2
binary :: Monad m => Wire -> Wire -> Controls -> (a -> a -> Ctrls a b -> m (a, a, Ctrls a b)) -> BT m a b
binary w v c f bindings = do
let w' = unbind_qubit_wire bindings w
let v' = unbind_qubit_wire bindings v
let c' = unbind_controls bindings c
(w'', v'', c'') <- f w' v' c'
let bindings1 = bind_qubit_wire w w'' bindings
let bindings2 = bind_qubit_wire v v'' bindings1
let bindings3 = bind_controls c c'' bindings2
return bindings3
binary_c :: Monad m => Wire -> Wire -> Controls -> (b -> b -> Ctrls a b -> m (b, b, Ctrls a b)) -> BT m a b
binary_c w v c f bindings = do
let w' = unbind_bit_wire bindings w
let v' = unbind_bit_wire bindings v
let c' = unbind_controls bindings c
(w'', v'', c'') <- f w' v' c'
let bindings1 = bind_bit_wire w w'' bindings
let bindings2 = bind_bit_wire v v'' bindings1
let bindings3 = bind_controls c c'' bindings2
return bindings3
list_unary :: Monad m => [Wire] -> Controls -> ([a] -> Ctrls a b -> m ([a], Ctrls a b)) -> BT m a b
list_unary ws c f bindings = do
let ws' = unbind_qubit_wire_list bindings ws
let c' = unbind_controls bindings c
(ws'', c'') <- f ws' c'
let bindings1 = bind_qubit_wire_list ws ws'' bindings
let bindings2 = bind_controls c c'' bindings1
return bindings2
list_binary :: Monad m => [Wire] -> [Wire] -> Controls -> ([a] -> [a] -> Ctrls a b -> m ([a], [a], Ctrls a b)) -> BT m a b
list_binary ws vs c f bindings = do
let ws' = unbind_qubit_wire_list bindings ws
let vs' = unbind_qubit_wire_list bindings vs
let c' = unbind_controls bindings c
(ws'', vs'', c'') <- f ws' vs' c'
let bindings1 = bind_qubit_wire_list ws ws'' bindings
let bindings2 = bind_qubit_wire_list vs vs'' bindings1
let bindings3 = bind_controls c c'' bindings2
return bindings3
qprep_ary :: Monad m => Wire -> (b -> m a) -> BT m a b
qprep_ary w f bindings = do
let w' = unbind_bit_wire bindings w
w'' <- f w'
let bindings1 = bind_qubit_wire w w'' bindings
return bindings1
qunprep_ary :: Monad m => Wire -> (a -> m b) -> BT m a b
qunprep_ary w f bindings = do
let w' = unbind_qubit_wire bindings w
w'' <- f w'
let bindings1 = bind_bit_wire w w'' bindings
return bindings1
cunary :: Monad m => Wire -> Controls -> (b -> Ctrls a b -> m (b, Ctrls a b)) -> BT m a b
cunary w c f bindings = do
let w' = unbind_bit_wire bindings w
let c' = unbind_controls bindings c
(w'', c'') <- f w' c'
let bindings1 = bind_bit_wire w w'' bindings
let bindings2 = bind_controls c c'' bindings1
return bindings2
qinit_ary :: Monad m => Wire -> m a -> BT m a b
qinit_ary w f bindings = do
w'' <- f
let bindings1 = bind_qubit_wire w w'' bindings
return bindings1
cinit_ary :: Monad m => Wire -> m b -> BT m a b
cinit_ary w f bindings = do
w'' <- f
let bindings1 = bind_bit_wire w w'' bindings
return bindings1
qterm_ary :: Monad m => Wire -> (a -> m ()) -> BT m a b
qterm_ary w f bindings = do
let w' = unbind_qubit_wire bindings w
() <- f w'
let bindings1 = bind_delete w bindings
return bindings1
cterm_ary :: Monad m => Wire -> (b -> m ()) -> BT m a b
cterm_ary w f bindings = do
let w' = unbind_bit_wire bindings w
() <- f w'
let bindings1 = bind_delete w bindings
return bindings1
cgate_ary :: Monad m => Wire -> [Wire] -> ([b] -> m (b, [b])) -> BT m a b
cgate_ary w vs f bindings = do
let vs' = unbind_bit_wire_list bindings vs
(w'', vs'') <- f vs'
let bindings1 = bind_bit_wire w w'' bindings
let bindings2 = bind_bit_wire_list vs vs'' bindings1
return bindings2
cgateinv_ary :: Monad m => Wire -> [Wire] -> (b -> [b] -> m [b]) -> BT m a b
cgateinv_ary w vs f bindings = do
let vs' = unbind_bit_wire_list bindings vs
let w' = unbind_bit_wire bindings w
vs'' <- f w' vs'
let bindings1 = bind_bit_wire_list vs vs'' bindings
return bindings1
subroutine_ary :: Monad m => [Wire] -> [Wire] -> Controls
-> ([B_Endpoint a b] -> Ctrls a b -> m ([B_Endpoint a b], Ctrls a b))
-> BT m a b
subroutine_ary ws vs c f bindings = do
let c' = unbind_controls bindings c
let ws' = unbind_list bindings ws
(vs'',c'') <- f ws' c'
let bindings1 = bind_list vs vs'' bindings
let bindings2 = bind_controls c c'' bindings1
return bindings2
phase_ary :: Monad m => [Wire] -> Controls -> ([B_Endpoint a b] -> Ctrls a b -> m (Ctrls a b)) -> BT m a b
phase_ary w c f bindings = do
let w' = map (unbind bindings) w
let c' = unbind_controls bindings c
c'' <- f w' c'
let bindings1 = bind_controls c c'' bindings
return bindings1
comment_ary :: Monad m => [(Wire, String)] -> (([(B_Endpoint a b, String)] -> m ()) -> BT m a b)
comment_ary ws f bindings = do
let ws' = zip (unbind_list bindings $ map fst ws) (map snd ws)
f ws'
return bindings
-- ----------------------------------------------------------------------
-- * Applying transformers to circuits
-- | Apply a 'Transformer' ⟦-⟧ to a 'Circuit' /C/, and output the
-- semantic function ⟦/C/⟧ :: bindings -> bindings.
transform_circuit :: Monad m => Transformer m a b -> Circuit -> Bindings a b -> m (Bindings a b)
transform_circuit transformer c bindings =
foldM apply bindings gs
where
(_,gs,_,_) = c
apply bindings g = transformer (bind_gate namespace_empty g) bindings
-- | Like 'transform_circuit', but for boxed circuits.
--
-- The handling of subroutines will depend on the transformer.
-- For \"gate transformation\" types of applications, one typically
-- would like to leave the boxed structure intact.
-- For \"simulation\" types of applications, one would generally
-- recurse through the boxed structure.
--
-- The difference is specified in the definition of the transformer
-- within the semantic function of the Subroutine gate, whether to
-- create another boxed gate or open the box.
transform_bcircuit_rec :: Monad m => Transformer m a b -> BCircuit -> Bindings a b -> m (Bindings a b)
transform_bcircuit_rec transformer (c,namespace) bindings =
foldM apply bindings gs
where
(_,gs,_,_) = c
apply bindings g = transformer (bind_gate namespace g) bindings
-- | Same as 'transform_bcircuit_rec', but specialized to when /m/ is
-- the identity operation.
transform_bcircuit_id :: Transformer Id a b -> BCircuit -> Bindings a b -> Bindings a b
transform_bcircuit_id t c b = getId (transform_bcircuit_rec t c b)
-- | To transform Dynamic Boxed circuits, we require a Transformer to define the
-- behavior on static gates, but we also require functions for what to do when
-- a subroutine is defined, and for when a dynamic_lift operation occurs. This is
-- all wrapped in the DynamicTransformer data type.
data DynamicTransformer m a b = DT {
transformer :: Transformer m a b,
define_subroutine :: BoxId -> TypedSubroutine -> m (),
lifting_function :: b -> m Bool
}
-- | Like 'transform_bcircuit_rec', but for dynamic-boxed circuits.
--
-- \"Write\" operations can be thought of as gates, and so they are passed to
-- the given transformer. The handling of \"Read\" operations is taken care of
-- by the \"lifting_function\" of the DynamicTransformer. \"Subroutine\" operations
-- call the 'define_subroutine' function of the DynamicTransformer.
transform_dbcircuit :: Monad m => DynamicTransformer m a b -> DBCircuit x -> Bindings a b -> m (x,Bindings a b)
transform_dbcircuit dt (a0,rw) bindings = evalStateT (inner_transform dt (a0,rw) bindings) namespace_empty where
inner_transform :: Monad m => DynamicTransformer m a b -> DBCircuit x -> Bindings a b -> (StateT Namespace m) (x,Bindings a b)
inner_transform dt (a0,rw) bindings =
case rw of
(RW_Return (_,_,x)) -> return (x,bindings)
(RW_Write gate rw') -> do
namespace <- get
bindings' <- lift $ (transformer dt) (bind_gate namespace gate) bindings
inner_transform dt (a0,rw') bindings'
(RW_Read wire rw_cont) -> do
let bit = unbind_bit_wire bindings wire
bool <- lift $ (lifting_function dt) bit
let rw' = rw_cont bool
inner_transform dt (a0,rw') bindings
(RW_Subroutine name subroutine rw') -> do
lift $ (define_subroutine dt) name subroutine
namespace <- get
let namespace' = map_provide name subroutine namespace
put namespace'
inner_transform dt (a0,rw') bindings