Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update cirq.decompose protocol to perform a DFS instead of a BFS on the decomposed OP-TREE #6116

Merged
merged 3 commits into from
Jun 1, 2023

Conversation

tanujkhattar
Copy link
Collaborator

@tanujkhattar tanujkhattar commented Jun 1, 2023

As mentioned in the title, the PR updates the internal implementation of cirq.decompose() protocol to perform DFS on the implicit compute graph instead of a BFS. This is useful in order to add support for qubit allocations and deallocations using a qubit manager inside the decompose protocol. This PR does the following:

  1. Delegates the cirq.decompose logic to a recursive DFS based implementation as part of _decompose_dfs
  2. Significantly simplifies the logic for preserve_structure=True by deleting the special purpose _decompose_preserving_structure method.

Note that all previously existing tests still pass, which implies that the decompose behavior is unaffected and it's only an internal implementation detail. I will add more tests in a follow-up PR with cleanups that show the benefits of this recursive implementation from the perspective of qubit allocation / deallocation.

cc @NoureldinYosri

part of #6040

@CirqBot CirqBot added the size: M 50< lines changed <250 label Jun 1, 2023
@tanujkhattar tanujkhattar changed the title update cirq.decompose protocol to perform a DFS instead of a BFS on the decomposed OP-TREE Update cirq.decompose protocol to perform a DFS instead of a BFS on the decomposed OP-TREE Jun 1, 2023
@tanujkhattar tanujkhattar marked this pull request as ready for review June 1, 2023 10:31
@tanujkhattar tanujkhattar requested review from a team, vtomole and cduck as code owners June 1, 2023 10:31
output = []
queue: List[Any] = [val]
while queue:
item = queue.pop(0)
Copy link
Collaborator

@NoureldinYosri NoureldinYosri Jun 1, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

wouldn't replacing queue.pop(0) with queue.pop() and queue[:0] = x with queue.extend(reversed(x)) do the trick ?

dfs order is just stack order so replacing the lines (and renaming queue to stack) should do the job.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, this would be sufficient for replacing the BFS order to a DFS order. But what we really is not just replacing the output ordering but also make sure that the items in the recursive OP-TREE are iterated upon in true DFS based ordering. So, in a follow up PR (since it requires a bunch of changes to the interfaces), I want the following test to pass:

def test_decompose_recursive_dfs():
    mock_qm = mock.Mock()

    class RecursiveDecompose(cirq.Gate):
        def __init__(self, recurse: bool = True):
            self.recurse = recurse

        def _num_qubits_(self) -> int:
            return 1

        def _decompose_(self, qubits):
            mock_qm.qalloc(self.recurse)
            yield RecursiveDecompose(recurse=False).on(*qubits) if self.recurse else cirq.Z(*qubits)
            mock_qm.qfree(self.recurse)

    gate = RecursiveDecompose()
    _ = cirq.decompose(gate.on(cirq.NamedQubit("q")))
    expected_calls = [
        mock.call.qalloc(True),
        mock.call.qalloc(False),
        mock.call.qfree(False),
        mock.call.qfree(True),
    ]
    mock_qm.assert_has_calls(expected_calls, any_order=False)

If you simply use a stack instead of a queue but continue the with the iterative approach; in the first step of iteration you'll end up yielding all the elements of RecursiveDecompose(True)._decompose_() instead of getting only the first value from the generator and then recursively decomposing it before moving on to the next statement.

If we get a list of all elements from RecursiveDecompose(True)._decompose_() in a single shot and insert them in a stack; the output ordering of decomposed operations would follow the DFS ordering but we would be processing all children of a node at once; instead of processing them one by one.

We could potentially get around this constraint by adding more if/else blocks and checking whether If the operation on top of the stack is an OP-TREE; only yield the first value from the generator and then push the modified OP-TREE back in the stack along with the first child. Then pop the first child and continue to decompose. But I think the recursive implementation here is a lot more intuitive; so unless there are significant performance concerns with using the recursive approach; I'd say let's make the change to a recursive approach now and we can always optimize back to the iterative approach later.

What do you think? I can also prototype the iterative approach directly if you have a strong preference.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

flattening the OP_TREE and pushing it to the stack in reverse order is what we need

stack.extend(reversed(cirq.flatten_to_ops(cirq.decompose_once(val))))

I'm okay either way. it's just that it seems that modifying the iterative approach is a lot less work

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As I highlighted in my previous comment, flattening the OP-TREE will not work in this case. My argument will become more clear in my next follow up PR. I'll tag you once it's out and we can come back to the iterative vs recursive discussion at that point.

Copy link
Collaborator

@95-martin-orion 95-martin-orion left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Couple of pretty minor nits. The fact that the tests are passing gives me a godd amount of confidence in this PR - would it be worth adding a test that this now operates in DFS order instead of BFS?

cirq-core/cirq/protocols/decompose_protocol.py Outdated Show resolved Hide resolved
cirq-core/cirq/protocols/decompose_protocol.py Outdated Show resolved Hide resolved
@tanujkhattar tanujkhattar enabled auto-merge (squash) June 1, 2023 20:04
@tanujkhattar
Copy link
Collaborator Author

Thanks for the review @95-martin-orion!

would it be worth adding a test that this now operates in DFS order instead of BFS?

Watch out for my next PR with a detailed test with a couple more changes to the interfaces. Since decompose is fairly complex; I wanted to keep the interface changes separate from the internal implementation changes (like this one). This PR only gets us to a half-way point. See my comment above for more details on the upcoming test - #6116 (comment)

@tanujkhattar tanujkhattar merged commit b1e09a9 into quantumlib:master Jun 1, 2023
navaro1 pushed a commit to navaro1/Cirq that referenced this pull request Jun 3, 2023
…he decomposed OP-TREE (quantumlib#6116)

* update cirq.decompose protocol to perform a DFS instead of a BFS on the decomposed OP-TREE

* Fix mypy error

* Use a dataclass instead of kwargs
harry-phasecraft pushed a commit to PhaseCraft/Cirq that referenced this pull request Oct 31, 2024
…he decomposed OP-TREE (quantumlib#6116)

* update cirq.decompose protocol to perform a DFS instead of a BFS on the decomposed OP-TREE

* Fix mypy error

* Use a dataclass instead of kwargs
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
size: M 50< lines changed <250
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants