forked from zxcalc/pyzx
-
Notifications
You must be signed in to change notification settings - Fork 0
/
parity_network.py
52 lines (46 loc) · 2.06 KB
/
parity_network.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
# PyZX - Python library for quantum circuit rewriting
# and optimization using the ZX-calculus
# Copyright (C) 2018 - Aleks Kissinger and John van de Wetering
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
# http://www.apache.org/licenses/LICENSE-2.0
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
from typing import List, Tuple, cast
from typing_extensions import Literal
from .circuit.gates import CNOT
def parity_network(n: int, S: List[List[Literal[0,1]]]) -> List[CNOT]:
"""Implements the parity network algorithm of https://arxiv.org/pdf/1712.01859.pdf.
Specifically, it follows the pseudo-code of page 14."""
c = [] # List of cnots
Q: List[Tuple[List[List[Literal[0,1]]],List[int],int]] = [] # stack
Q.append((S,list(range(n)),-1))
while Q:
S, I, i = Q.pop()
if not S or not I: continue
if i != -1:
while True:
for j in range(n):
if j==i: continue
if all(y[j] for y in S):
c.append(CNOT(j,i))
for (Sp,Ip,ip) in (Q+[(S,I,i)]):
for y in Sp:
y[j] = cast(Literal[0,1], (y[i]+y[j])%2)
break
else:
break
j = max(I, key=lambda j: max([len([y for y in S if y[j]==0]),len([y for y in S if y[j]==1])]))
S0 = [y.copy() for y in S if y[j]==0]
S1 = [y.copy() for y in S if y[j]==1]
Iprime = [jp for jp in I if jp!=j]
if i == -1:
Q.append((S1,Iprime,j))
else:
Q.append((S1,[jp for jp in I if jp!=i],i))
Q.append((S0,Iprime, i))
return c