summaryrefslogtreecommitdiff
path: root/src/main/scala/chisel3/util/TransitName.scala
blob: cc8f24560e14edc2d7f72f1c952b1f01486cf269 (plain)
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
53
54
55
56
57
58
59
60
61
62
63
64
65
// SPDX-License-Identifier: Apache-2.0

package chisel3.util

import chisel3.internal.HasId

/** The purpose of `TransitName` is to improve the naming of some object created in a different scope by "transiting"
  * the name from the outer scope to the inner scope.
  *
  * Consider the example below. This shows three ways of instantiating `MyModule` and returning the IO. Normally, the
  * instance will be named `MyModule`. However, it would be better if the instance was named using the name of the `val`
  * that user provides for the returned IO. `TransitName` can then be used to "transit" the name ''from'' the IO ''to''
  * the module:
  *
  * {{{
  * /* Assign the IO of a new MyModule instance to value "foo". The instance will be named "MyModule". */
  * val foo = Module(new MyModule).io
  *
  * /* Assign the IO of a new MyModule instance to value "bar". The instance will be named "bar". */
  * val bar = {
  *   val x = Module(new MyModule)
  *   TransitName(x.io, x) // TransitName returns the first argument
  * }
  *
  * /* Assign the IO of a new MyModule instance to value "baz". The instance will be named "baz_generated". */
  * val baz = {
  *   val x = Module(new MyModule)
  *   TransitName.withSuffix("_generated")(x.io, x) // TransitName returns the first argument
  * }
  * }}}
  *
  * `TransitName` helps library writers following the [[https://en.wikipedia.org/wiki/Factory_method_pattern Factory
  * Method Pattern]] where modules may be instantiated inside an enclosing scope. For an example of this, see how the
  * [[Queue$ Queue]] factory uses `TransitName` in
  * [[https://github.com/freechipsproject/chisel3/blob/master/src/main/scala/chisel3/util/Decoupled.scala
  * Decoupled.scala]] factory.
  */
object TransitName {

  /** Transit a name from one type to another
    * @param from the thing with a "good" name
    * @param to the thing that will receive the "good" name
    * @return the `from` parameter
    */
  def apply[T <: HasId](from: T, to: HasId): T = {
    // To transit a name, we need to hook on both the suggestName and autoSeed mechanisms
    from.addSuggestPostnameHook((given_name: String) => { to.suggestName(given_name) })
    from.addAutoPostnameHook((given_name: String) => { to.autoSeed(given_name) })
    from
  }

  /** Transit a name from one type to another ''and add a suffix''
    * @param suffix the suffix to append
    * @param from the thing with a "good" name
    * @param to the thing that will receive the "good" name
    * @return the `from` parameter
    */
  def withSuffix[T <: HasId](suffix: String)(from: T, to: HasId): T = {
    // To transit a name, we need to hook on both the suggestName and autoSeed mechanisms
    from.addSuggestPostnameHook((given_name: String) => { to.suggestName(given_name + suffix) })
    from.addAutoPostnameHook((given_name: String) => { to.autoSeed(given_name + suffix) })
    from
  }

}