quimb.tensor.circuit.peps¶
PEPS simple-update circuit simulator.
Classes¶
Quantum circuit simulation keeping the state as a generic tensor |
Module Contents¶
- class quimb.tensor.circuit.peps.CircuitPEPSSimpleUpdate(N=None, *, edges=None, gates=None, psi0=None, max_bond=None, cutoff=1e-10, renorm=False, gauge_smudge=1e-12, equilibrate_every=None, equilibrate_opts=None, gate_opts=None, dtype=None, to_backend=None, convert_eager=True, **circuit_opts)[source]¶
Bases:
quimb.tensor.circuit.exact.CircuitQuantum circuit simulation keeping the state as a generic tensor network (a “PEPS” defined by an arbitrary graph of
edges) and applying gates with the simple update rule. The state always keeps a single tensor per site, with bonds only along the supplied edges; two-qubit gates are only supported on those edges. Bond singular values are tracked as Vidal-style gauges, which makes gate application and the computation of local expectations cheap and approximate.This is useful for circuits on lattices that build up more than 1D worth of entanglement, where an exact or MPS simulation is intractable but a truncated, gauged tensor network state is a good approximation.
- Parameters:
N (int, optional) – The number of qubits in the circuit. If not given it is inferred from the geometry. Supply it to pad the geometry up to
Nsites, including any that have no edges.edges (sequence[tuple[int, int]], optional) – The edges defining the geometry of the PEPS. A bond is placed between each pair of sites, and two-qubit gates are only supported on these edges. Every site appearing in
edgesis included. If not given the geometry is taken fromgatesorpsi0instead.gates (sequence, optional) – If
edgesis not given, infer the geometry from the two-qubit gates in this sequence. The gates are only inspected here, not applied, so you still pass them toapply_gates()afterwards.psi0 (TensorNetworkGenVector, optional) – Supply the initial state directly instead of starting from the
|00...0>product state. Ifedgesis not given the geometry is read from the bonds of this state, and the bond gauges are seeded from it. Only a single seeding sweep is performed; unlike imaginary time simple update the gauge matters immediately, so for an arbitrarypsi0you may want to callequilibrate()once before applying gates.max_bond (int, optional) – The maximum bond dimension to truncate to when applying gates.
cutoff (float, optional) – The singular value cutoff to use when truncating after applying gates.
renorm (bool, optional) – Whether to renormalize the singular values of a bond after each gate. The default
Falsetracks the norm of the state rather than forcing it to one, which is the sensible choice for real time and general circuit dynamics. SetTrueto instead keep the state normalized after every gate, e.g. for the near-identity gates of imaginary time evolution.gauge_smudge (float, optional) – Small value added to the gauges before they are multiplied in and inverted, for numerical stability with very small singular values.
equilibrate_every (int, optional) – If given, automatically call
equilibrate()after every this many gates have been applied.equilibrate_opts (dict, optional) – Default options forwarded to
equilibrate().gate_opts (dict, optional) – Default options to pass to
gate_simple_such asmax_bondandcutoff.dtype (str, optional) – If given, ensure the state tensors are cast to this data type.
to_backend (callable, optional) – If given, apply this function to the state tensors to convert them to a particular array backend.
convert_eager (bool, optional) – Whether to apply the
dtypeandto_backendconversions eagerly as each gate is applied. The defaultTruematches the other running simulators (e.g.CircuitMPS), since the simple update rule contracts each gate into the state immediately rather than building a lazy network to contract later.
- gauges¶
The current Vidal-style bond gauges (singular values), keyed by bond index, updated in place as gates are applied.
Notes
The gates applied must address qubits using the same labels that appear in
edges. Two-qubit gates are only supported along an existing edge.Examples
>>> import quimb.tensor as qtn >>> edges = [(0, 1), (1, 2), (0, 3), (1, 4), (2, 5), (3, 4), (4, 5)] >>> circ = qtn.CircuitPEPSSimpleUpdate(edges=edges, max_bond=8) >>> circ.apply_gates(gates) >>> peps = circ.psi
See also
CircuitMPS,CircuitDense- _edges¶
- _sites¶
- _site_set¶
- _edge_set¶
- gauges¶
- _equilibrate_every = None¶
- _equilibrate_opts¶
- copy()[source]¶
Copy the circuit, including its state, gauges and geometry. The base
Circuitcopy does not know about the extra simple update attributes, so they are carried over here (the gauges are copied so the two circuits can be evolved independently).
- property edges¶
The unique edges defining the PEPS geometry.
- property sites¶
The sites (qubit labels) of the PEPS.
- _apply_gate(gate, tags=None, **gate_opts)[source]¶
Apply a
Gateto thisCircuit. This is the main method that all calls to apply a gate should go through.
- apply_gates(gates, progbar=False, **gate_opts)[source]¶
Apply a sequence of gates to this tensor network quantum circuit.
- Parameters:
gates (Sequence[Gate] or Sequence[Tuple]) – The sequence of gates to apply.
gate_opts – Supplied to
apply_gate().
- equilibrate(**gauge_opts)[source]¶
Re-gauge the whole state with the simple update rule, improving the consistency of the tracked bond gauges. This does not change the state represented, only the gauge, and can be called periodically between rounds of gates to keep the simple update approximation well behaved.
The default options given at construction via
equilibrate_optsare applied first, with any keyword arguments here taking precedence.- Parameters:
gauge_opts – Supplied to
gauge_all_simple_(), for examplemax_iterationsandtol.
- local_expectation(G, where, *, max_distance=0, normalized=True, **contract_opts)[source]¶
Compute the local expectation value of operator
Gat the site(s)where, using the simple update bond gauges to approximate the environment beyondmax_distance.- Parameters:
G (array_like) – The local operator.
where (hashable or sequence[hashable]) – The site or sites to compute the expectation at. A single site label (which may itself be a tuple, e.g. a 2D coordinate) is detected by membership in the set of sites.
max_distance (int, optional) – How many graph hops of neighboring tensors to include in the local cluster used to approximate the reduced density matrix. The default
0uses only the target site(s) and their gauges, matchingcompute_local_expectation_cluster().normalized (bool, optional) – Whether to normalize by the local norm.
contract_opts – Supplied to
compute_local_expectation_cluster().
- Return type:
- get_state(absorb_gauges=True)[source]¶
Return the current PEPS state, optionally absorbing the bond gauges.
- Parameters:
absorb_gauges (bool or "return", optional) – How to handle the tracked Vidal-style bond gauges. If
True(the default) the gauges are absorbed, so the returned tensor network is the actual wavefunction (up to the simple update approximation). IfFalsethe gauges are added to the network as uncontracted diagonal tensors. If"return"the raw gauged network and a copy of the gauges are returned separately. The internal state is left untouched in every case.- Returns:
psi (TensorNetwork) – The current state.
gauges (dict) – The current gauges, only if
absorb_gauges == "return".
- property psi¶
The PEPS tensor network state, with the simple update bond gauges absorbed back in so that it represents the actual wavefunction (a proper contraction of
psigives the state, up to the simple update approximation). The internal gauged form is left untouched. Shorthand forget_state(absorb_gauges=True).
- calc_qubit_ordering(qubits=None)[source]¶
Get a order to measure
qubitsin, by greedily choosing whichever has the smallest reverse lightcone followed by whichever expands this lightcone least.
- to_dense(*args, **kwargs)[source]¶
Contract the gauged PEPS into a dense wavefunction, a column-vector
qarrayof length2**Nordered likesites, matching the output ofCircuit.to_dense(). This is the actual (approximate) state, so the cost grows exponentially with the number of qubits.Arguments are forwarded to
to_dense().
- amplitude(*args, **kwargs)[source]¶
Get the amplitude coefficient of bitstring
b.\[c_b = \langle b | \psi \rangle\]- Parameters:
b (str or sequence of int) – The bitstring to compute the transition amplitude for.
optimize (str, optional) – Contraction path optimizer to use for the amplitude, can be a non-reusable path optimizer as only called once (though path won’t be cached for later use in that case).
simplify_sequence (str, optional) – Which local tensor network simplifications to perform and in which order, see
full_simplify().simplify_atol (float, optional) – The tolerance with which to compare to zero when applying
full_simplify().simplify_equalize_norms (bool, optional) – Actively renormalize tensor norms during simplification.
backend (str, optional) – Backend to perform the contraction with, e.g.
'numpy','cupy'or'jax'. Passed tocotengra.dtype (str, optional) – Data type to cast the TN to before contraction.
rehearse (bool or "tn", optional) – If
True, generate and cache the simplified tensor network and contraction tree but don’t actually perform the contraction. Returns a dict with keys"tn"and'tree'with the tensor network that will be contracted and the corresponding contraction tree if so.
- sample(*args, **kwargs)[source]¶
Sample the circuit given by
gates,Ctimes, using lightcone cancelling and caching marginal distribution results. This is a generator. This proceeds as a chain of marginal computations.Assuming we have
group_size=1, and some ordering of the qubits, \(\{q_0, q_1, q_2, q_3, \ldots\}\) we first compute:\[p(q_0) = \mathrm{diag} \mathrm{Tr}_{1, 2, 3,\ldots} | \psi_{0} \rangle \langle \psi_{0} |\]I.e. simply the probability distribution on a single qubit, conditioned on nothing. The subscript on \(\psi\) refers to the fact that we only need gates from the causal cone of qubit 0. From this we can sample an outcome, either 0 or 1, if we call this \(r_0\) we can then move on to the next marginal:
\[p(q_1 | r_0) = \mathrm{diag} \mathrm{Tr}_{2, 3,\ldots} \langle r_0 | \psi_{0, 1} \rangle \langle \psi_{0, 1} | r_0 \rangle\]I.e. the probability distribution of the next qubit, given our prior result. We can sample from this to get \(r_1\). Then we compute:
\[p(q_2 | r_0 r_1) = \mathrm{diag} \mathrm{Tr}_{3,\ldots} \langle r_0 r_1 | \psi_{0, 1, 2} \rangle \langle \psi_{0, 1, 2} | r_0 r_1 \rangle\]Eventually we will reach the ‘final marginal’, which we can compute as
\[|\langle r_0 r_1 r_2 r_3 \ldots | \psi \rangle|^2\]since there is nothing left to trace out.
- Parameters:
C (int) – The number of times to sample.
qubits (None or sequence of int, optional) – Which qubits to measure, defaults (
None) to all qubits.order (None or sequence of int, optional) – Which order to measure the qubits in, defaults (
None) to an order based on greedily expanding the smallest reverse lightcone. If specified it should be a permutation ofqubits.group_size (int, optional) – How many qubits to group together into marginals, the larger this is the fewer marginals need to be computed, which can be faster at the cost of higher memory. The marginal themselves will each be of size
2**group_size.max_marginal_storage (int, optional) – The total cumulative number of marginal probabilites to cache, once this is exceeded caching will be turned off.
seed (None or int, optional) – A random seed, passed to
numpy.random.seedif given.optimize (str, optional) – Contraction path optimizer to use for the marginals, shouldn’t be a non-reusable path optimizer as called on many different TNs. Passed to
cotengra.array_contract_tree().backend (str, optional) – Backend to perform the marginal contraction with, e.g.
'numpy','cupy'or'jax'. Passed tocotengra.dtype (str, optional) – Data type to cast the TN to before contraction.
simplify_sequence (str, optional) – Which local tensor network simplifications to perform and in which order, see
full_simplify().simplify_atol (float, optional) – The tolerance with which to compare to zero when applying
full_simplify().simplify_equalize_norms (bool, optional) – Actively renormalize tensor norms during simplification.
- Yields:
bitstrings (sequence of str)
- sample_chaotic(*args, **kwargs)[source]¶
Sample from this circuit, assuming it to be chaotic. Which is to say, only compute and sample correctly from the final marginal, assuming that the distribution on the other qubits is uniform. Given
marginal_qubits=5for instance, for each sample a random bit-string \(r_0 r_1 r_2 \ldots r_{N - 6}\) for the remaining \(N - 5\) qubits will be chosen, then the final marginal will be computed as\[p(q_{N-5}q_{N-4}q_{N-3}q_{N-2}q_{N-1} | r_0 r_1 r_2 \ldots r_{N-6}) = |\langle r_0 r_1 r_2 \ldots r_{N - 6} | \psi \rangle|^2\]and then sampled from. Note the expression on the right hand side has 5 open indices here and so is a tensor, however if
marginal_qubitsis not too big then the cost of contracting this is very similar to a single amplitude.Note
This method assumes the circuit is chaotic, if its not, then the samples produced will not be an accurate representation of the probability distribution.
- Parameters:
C (int) – The number of times to sample.
marginal_qubits (int or sequence of int) – The number of qubits to treat as marginal, or the actual qubits. If an int is given then the qubits treated as marginal will be
circuit.calc_qubit_ordering()[:marginal_qubits].fix (None or dict[int, str], optional) – Measurement results on other qubits to fix. These will be randomly sampled if
fixis not given or a qubit is missing.seed (None or int, optional) – A random seed, passed to
numpy.random.seedif given.optimize (str, optional) – Contraction path optimizer to use for the marginal, can be a non-reusable path optimizer as only called once (though path won’t be cached for later use in that case).
backend (str, optional) – Backend to perform the marginal contraction with, e.g.
'numpy','cupy'or'jax'. Passed tocotengra.dtype (str, optional) – Data type to cast the TN to before contraction.
simplify_sequence (str, optional) – Which local tensor network simplifications to perform and in which order, see
full_simplify().simplify_atol (float, optional) – The tolerance with which to compare to zero when applying
full_simplify().simplify_equalize_norms (bool, optional) – Actively renormalize tensor norms during simplification.
- Yields:
str
- sample_chaotic_rehearse(*args, **kwargs)[source]¶
Rehearse chaotic sampling (perform just the TN simplifications and contraction tree finding).
- Parameters:
marginal_qubits (int or sequence of int) – The number of qubits to treat as marginal, or the actual qubits. If an int is given then the qubits treated as marginal will be
circuit.calc_qubit_ordering()[:marginal_qubits].result (None or dict[int, str], optional) – Explicitly check the computational cost of this result, assumed to be all zeros if not given.
optimize (str, optional) – Contraction path optimizer to use for the marginal, can be a non-reusable path optimizer as only called once (though path won’t be cached for later use in that case).
simplify_sequence (str, optional) – Which local tensor network simplifications to perform and in which order, see
full_simplify().simplify_atol (float, optional) – The tolerance with which to compare to zero when applying
full_simplify().simplify_equalize_norms (bool, optional) – Actively renormalize tensor norms during simplification.
dtype (str, optional) – Data type to cast the TN to before contraction.
- Returns:
The contraction path information for the main computation, the key is the qubits that formed the final marginal. The value is itself a dict with keys
'tn'- a representative tensor network - and'tree'- the contraction tree.- Return type:
- partial_trace(*args, **kwargs)[source]¶
Perform the partial trace on the circuit wavefunction, retaining only qubits in
keep, and making use of reverse lightcone cancellation:\[\rho_{\bar{q}} = Tr_{\bar{p}} |\psi_{\bar{q}} \rangle \langle \psi_{\bar{q}}|\]Where \(\bar{q}\) is the set of qubits to keep, \(\psi_{\bar{q}}\) is the circuit wavefunction only with gates in the causal cone of this set, and \(\bar{p}\) is the remaining qubits.
- Parameters:
keep (int or sequence of int) – The qubit(s) to keep as we trace out the rest.
optimize (str, optional) – Contraction path optimizer to use for the reduced density matrix, can be a non-reusable path optimizer as only called once (though path won’t be cached for later use in that case).
simplify_sequence (str, optional) – Which local tensor network simplifications to perform and in which order, see
full_simplify().simplify_atol (float, optional) – The tolerance with which to compare to zero when applying
full_simplify().simplify_equalize_norms (bool, optional) – Actively renormalize tensor norms during simplification.
backend (str, optional) – Backend to perform the marginal contraction with, e.g.
'numpy','cupy'or'jax'. Passed tocotengra.dtype (str, optional) – Data type to cast the TN to before contraction.
rehearse (bool or "tn", optional) – If
True, generate and cache the simplified tensor network and contraction tree but don’t actually perform the contraction. Returns a dict with keys"tn"and'tree'with the tensor network that will be contracted and the corresponding contraction tree if so.
- Return type:
- abstract property uni¶
- get_psi_reverse_lightcone(where, keep_psi0=False)[source]¶
Get just the bit of the wavefunction in the reverse lightcone of sites in
where- i.e. causally linked.- Parameters:
where (int, or sequence of int) – The sites to propagate the the lightcone back from, supplied to
get_reverse_lightcone_tags().keep_psi0 (bool, optional) – Keep the tensors corresponding to the initial wavefunction regardless of whether they are outside of the lightcone.
- Returns:
psi_lc
- Return type: