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

[FIRRTL][AdvancedLayerSink] Add the ability to sink ops by cloning #8308

Merged
merged 1 commit into from
Mar 12, 2025

Conversation

rwy7
Copy link
Contributor

@rwy7 rwy7 commented Mar 7, 2025

In advanced-layer-sink, add the ability to sink ops by cloning them to their uses. This PR is only cloning constant-like ops, but we have the option to clone more sorts of ops as needed.

Constants are heavily shared, so they typically aren't sinkable by movement. Cloning will put constants directly into the layerblock where they are used, which avoids having us capture the constant as a port when lowering layerblocks to modules. It is important to put constants directly in a module because it enables more canonicalization later in the pipeline.

We used to rely on IMCP to sink constants after lower-layers, but we lost that as we moved the layer lowering passes later in the pipeline, which is why we now have to clone constants ourselves.

@rwy7 rwy7 force-pushed the lower-layers-cloning branch from 7aee9c8 to a896b07 Compare March 11, 2025 13:42
@rwy7 rwy7 changed the title [LowerLayers][FIRRTL] Add the ability to sink ops by cloning [FIRRTL][AdvancedLayerSink] Add the ability to sink ops by cloning Mar 11, 2025
@rwy7 rwy7 marked this pull request as ready for review March 11, 2025 13:46
@rwy7 rwy7 added FIRRTL Involving the `firrtl` dialect enhancement New feature or request labels Mar 11, 2025
Copy link
Member

@seldridge seldridge left a comment

Choose a reason for hiding this comment

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

One question. Otherwise this is looking great!

// Copy the op down to each block where there is a user.
// Use a cache to ensure we don't create more than one copy per block.
DenseMap<Block *, Operation *> cache;
for (unsigned i = 0, e = op->getNumResults(); i < e; ++i) {
Copy link
Member

Choose a reason for hiding this comment

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

Is this correct to iterate through results and clone individually?

This seems like it would do something wrong with hypothetical constant operations with multiple outputs whose users are in different blocks. Does this work for the following?

%0, %1 = firrtl.aggconstant 1, 3 : () -> (i1, i2)
firrtl.layerblock @A {
  bar(%0) : (i1) -> ()
}
firrtl.layerblock @B {
  bar(%1) : (i2) -> ()
}

The right thing may be to either only work on operations which have one result or to invalidate the users of the results which are now unused.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I think this code should work fine, it would create a clone for each layerblock, and replace each use with the correct result of the correct clone (ie, old result-no == new result-no). I tried it out by letting advanced-layer-sink clone instance ops, here is an example:

// ./bin/circt-opt -pass-pipeline="builtin.module(firrtl.circuit(firrtl-advanced-layer-sink))" -allow-unregistered-dialect ../scratch.mlir

firrtl.circuit "Test" {
  firrtl.layer @A bind {}
  firrtl.layer @B bind {}

  firrtl.module @Foo(out %a : !firrtl.uint<1>, out %b : !firrtl.uint<1>) {
    %c1_ui1 = firrtl.constant 1 : !firrtl.uint<1>
    firrtl.matchingconnect %a, %c1_ui1 : !firrtl.uint<1>
    firrtl.matchingconnect %b, %c1_ui1 : !firrtl.uint<1>
  }

  firrtl.module public @Test() {
    %foo_a, %foo_b = firrtl.instance foo @Foo(out a : !firrtl.uint<1>, out b : !firrtl.uint<1>)
    firrtl.layerblock @A {
      "unknown"(%foo_a) : (!firrtl.uint<1>) -> ()
      "unknown"(%foo_b) : (!firrtl.uint<1>) -> ()
    }

    firrtl.layerblock @B {
      "unknown"(%foo_a) : (!firrtl.uint<1>) -> ()
      "unknown"(%foo_b) : (!firrtl.uint<1>) -> ()
    }
  }
}

is translated to:

module {
  firrtl.circuit "Test" {
    firrtl.layer @A bind {
    }
    firrtl.layer @B bind {
    }
    firrtl.module @Foo(out %a: !firrtl.uint<1>, out %b: !firrtl.uint<1>) {
      %c1_ui1 = firrtl.constant 1 : !firrtl.uint<1>
      firrtl.matchingconnect %a, %c1_ui1 : !firrtl.uint<1>
      firrtl.matchingconnect %b, %c1_ui1 : !firrtl.uint<1>
    }
    firrtl.module public @Test() {
      %foo_a, %foo_b = firrtl.instance foo @Foo(out a: !firrtl.uint<1>, out b: !firrtl.uint<1>)
      firrtl.layerblock @A {
        %foo_a_0, %foo_b_1 = firrtl.instance foo @Foo(out a: !firrtl.uint<1>, out b: !firrtl.uint<1>)
        "unknown"(%foo_a_0) : (!firrtl.uint<1>) -> ()
        "unknown"(%foo_b_1) : (!firrtl.uint<1>) -> ()
      }
      firrtl.layerblock @B {
        %foo_a_0, %foo_b_1 = firrtl.instance foo @Foo(out a: !firrtl.uint<1>, out b: !firrtl.uint<1>)
        "unknown"(%foo_a_0) : (!firrtl.uint<1>) -> ()
        "unknown"(%foo_b_1) : (!firrtl.uint<1>) -> ()
      }
    }
  }
}

(it leaves the original instance in place because instance ops are not pure and I am using isOpTriviallyDead rather than just checking for empty uses).

Copy link
Member

Choose a reason for hiding this comment

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

Yes, that seems like it works. And it will still clone if foo_a is only used in layerblock @A and foo_b is only used in layerblock @B?

This is also done with modifications to the code to weaken cloneable to allow it to clone instances, correct?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

And it will still clone if foo_a is only used in layerblock @A and foo_b is only used in layerblock @B?

Yep, here is an example:

firrtl.circuit "Test" {
  firrtl.layer @A bind {}
  firrtl.layer @B bind {}

  firrtl.module @Foo(out %a : !firrtl.uint<1>, out %b : !firrtl.uint<1>) {
    %c1_ui1 = firrtl.constant 1 : !firrtl.uint<1>
    firrtl.matchingconnect %a, %c1_ui1 : !firrtl.uint<1>
    firrtl.matchingconnect %b, %c1_ui1 : !firrtl.uint<1>
  }

  firrtl.module public @Test() {
    %foo_a, %foo_b = firrtl.instance foo @Foo(out a : !firrtl.uint<1>, out b : !firrtl.uint<1>)
    firrtl.layerblock @A {
      "unknown"(%foo_a) : (!firrtl.uint<1>) -> ()
    }

    firrtl.layerblock @B {
      "unknown"(%foo_a) : (!firrtl.uint<1>) -> ()
    }
  }
}

To:

module {
  firrtl.circuit "Test" {
    firrtl.layer @A bind {
    }
    firrtl.layer @B bind {
    }
    firrtl.module @Foo(out %a: !firrtl.uint<1>, out %b: !firrtl.uint<1>) {
      %c1_ui1 = firrtl.constant 1 : !firrtl.uint<1>
      firrtl.matchingconnect %a, %c1_ui1 : !firrtl.uint<1>
      firrtl.matchingconnect %b, %c1_ui1 : !firrtl.uint<1>
    }
    firrtl.module public @Test() {
      %foo_a, %foo_b = firrtl.instance foo @Foo(out a: !firrtl.uint<1>, out b: !firrtl.uint<1>)
      firrtl.layerblock @A {
        %foo_a_0, %foo_b_1 = firrtl.instance foo @Foo(out a: !firrtl.uint<1>, out b: !firrtl.uint<1>)
        "unknown"(%foo_a_0) : (!firrtl.uint<1>) -> ()
      }
      firrtl.layerblock @B {
        %foo_a_0, %foo_b_1 = firrtl.instance foo @Foo(out a: !firrtl.uint<1>, out b: !firrtl.uint<1>)
        "unknown"(%foo_a_0) : (!firrtl.uint<1>) -> ()
      }
    }
  }
}

This is also done with modifications to the code to weaken cloneable to allow it to clone instances, correct?

Right.

Copy link
Member

@youngar youngar left a comment

Choose a reason for hiding this comment

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

LGTM

@rwy7 rwy7 force-pushed the lower-layers-cloning branch from a896b07 to cc69f64 Compare March 11, 2025 17:33
Copy link
Member

@seldridge seldridge left a comment

Choose a reason for hiding this comment

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

LGTM

The instance cloning has me thinking that you could eventually think about a cost model for this. cloneable is really that and it's just that constant ops are free. There may eventually be other ops we can clone if the cost model allows. Just food for thought.

@rwy7 rwy7 merged commit 2c66ddc into llvm:main Mar 12, 2025
5 checks passed
@rwy7 rwy7 deleted the lower-layers-cloning branch March 12, 2025 13:41
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request FIRRTL Involving the `firrtl` dialect
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants