

# **BSV Training**

Lec\_Interop\_RTL



#### BSV interoperates with V, SV, VHDL, SystemC, and C/C++





V (Verilog), SV (SystemVerilog), VHDL, or SystemC (event-driven)

BSV

C, C++



Plugging RTL (or RTL-like SystemC) into BSV

#### When does BSV interface to RTL (V, SV, VHDL)?

#### Plugging RTL into BSV:

- Many BSV designs are used in projects where there is already some existing verified RTL IP, which we simply wish to re-use
- Users may wish to add a new "primitive" to BSV that is important for an application domain
  - In fact, all BSV "primitives" are imported this way—knowledge about primitives is not built into the *bsc* compiler

This is done using BSV's "import BVI" capability

#### Plugging BSV into RTL:

- A BSV design may fit into a larger RTL design
- A BSV design may be verified in an existing verification environment that is based on SystemVerilog (e.g., VMM), 'e', etc.

This is done using by scheduling and naming control on the top-level interface, to be compatible with the RTL environment into which it fits



## Basic structure of importing a Verilog component



- Define the BSV interface type that the module should provide to its environment
- Use the "import BVI" mechanism to define a wrapper module, which will:
  - Describe how the interface method arguments, results, enables, clocks, resets, etc. connect to the Verilog module ports
  - Describe the BSV scheduling constraints on the methods



#### Example: importing a MAC (multiply-accumulate) module

The Verilog module we wish to import:

```
module mymac(EN, a, b, clear_value, clear, out, clk, rst_b);
input a, b, EN, clear, clear value, clk, rst b;
output out;
reg [15:0] out;
wire [15:0] a, b, clear value;
always@(posedge clk or negedge rst b)
if (!rst b)
  out <= 0;
else
  out <= clear ? clear_value : (EN ? out+a*b: out);
endmodule
```

- When clear == 1, out <= clear value</li>
- When EN == 1, out <= out + a \* b
- The value of the "out" register is continuously available as an output



#### Define the desired BSV interface

```
interface Mac_IFC;
  method Action acc (Int#(16) aa, Int#(16) bb);
  method Action reset_acc (Int#(16) value);
  method Int#(16) read_y;
endinterface
```

- When clear == 1, out <= clear\_value</li>
  - The related Verilog signals "clear" and "clear\_value" will be part of the BSV "reset\_acc" method
- When EN == 1, out <= out + a \* b</li>
  - The related Verilog signals "EN", "a" and "b" will be part of the BSV "acc" method
- The value of the "out" register is continuously available as an output
  - This will be done with the "read\_y" method



#### Define the module wrapper

```
BSV module name
Verilog module name
        import "BVI" mymac =
              module mkMac (Mac_IFC);
               default_clock dclk (clk);
                                                              Clock and reset
               default_reset dreset (rst b);
               method acc(a, b)
                                             enable(EN);
                                                              Methods
               method reset acc(clear value) enable(clear);
               method out read y();
               schedule (read y) SB (reset acc, acc);
                                                               Scheduling
               schedule (acc) C (reset acc);
                                                               constraints
              endmodule
```

- The "default\_clock" line indicates that the BSV default clock will be connected to Verilog "clk", and that it will be known within mkMac as "dclk" (although in this example it is not used elsewhere; typically it's used in a "clocked\_by" clause).
- In the "method" lines we can see how method signals (arguments, results, ENABLEs) connect to various Verilog ports (which are shown in green). Since the optional "ready()" clauses are missing, these methods will be always ready.



#### Define the module wrapper

```
BSV module name
Verilog module name
        import "BVI" mymac =
              module mkMac (Mac_IFC);
               default_clock dclk (clk);
                                                              Clock and reset
               default reset dreset (rst b);
               method acc(a, b)
                                             enable(EN);
                                                              Methods
               method reset_acc(clear_value) enable(clear);
               method out read y();
               schedule (read y) SB (reset acc, acc);
                                                               Scheduling
               schedule (acc) C (reset acc);
                                                               constraints
              endmodule
```

The "schedule" lines inform the *bsc* compiler about scheduling constraints on the methods.

- The first line indicates that read\_y < reset\_acc and read\_y < acc ("SB" = "sequenced before" = "<"). This captures the Verilog semantics that "out" carries the current value in the register, and that any change due to reset\_acc or acc will only be visible on the next clock.
- The second line indicates that acc and reset\_acc conflict ("C"), so they can never fire concurrently (in the same clock). This design choice is stricter than the Verilog, which does allow both to happen in the same clock, giving priority to reset\_acc. To match those semantics we could have instead specified: read\_y SB acc and acc SB reset\_acc

## Notes on the import Verilog mechanism

The previous slides showed only a simple example. The Reference Guide Sec. 15 goes into a lot more detail:

- Connecting clocks and resets
- Connecting method ENABLE and RDY signals
- The method-to-RTL connections need not be 1-to-1, and can involve other logic
- The available scheduling annotations
- Caveat: the schedule is just an assertion by you taken at face value by the compiler when using this module in your BSV program; the compiler makes no attempt to 'verify' that the scheduling assertions are sensible

The actual Verilog code is not touched by the *bsc* compiler. The *bsc* compiler simply generates a Verilog instantiation of the module, and your RTL simulator finds and instantiates the Verilog module

These import mechanisms can also be used for importing VHDL and SystemVerilog, and RTL-style SystemC

 Most RTL simulators allow free intermixing of Verilog, VHDL, SystemVerilog and SystemC



### What about Bluesim, when you import RTL?

Bluesim does not currently support "co-simulation" of Bluesim with a Verilog simulator

- If you just import a Verilog module, your only simulation option is Verilog simulation, i.e.,
  - Use bsc to generate Verilog from the BSV code
  - Use a Verilog simulator to simulate all the Verilog code (BSV-generated, and imported)

How do the primitives in the BSV library work in Bluesim?

- Every primitive in the BSV library is implemented both in Verilog and in C
  - The C code is imported using the "import BDPI" mechanism (described later), into a module with exactly the same BSV interface
- For every primitive, we have a wrapper module that instantiates one or the other, like this:

• "genVerilog" is a built-in boolean which is true when compiling for Verilog and False when compiling for Bluesim (Reference Guide Sec. B.6). Thus, at compile time, *bsc* chooses the appropriate instantiation (i.e., this is a "static elaboration" choice).



# Plugging BSV into RTL



## Plugging BSV into RTL: key issues

When plugging BSV into RTL (more accurately: BSV-generated RTL into RTL), the key issues are:



#### We need:

- Exactly the right set of input and output signal ports, with exactly the right port names, as expected by the RTL environment
- Exactly the right signaling protocols on these ports, as expected by the RTL environment (may be different from standard BSV method protocol of RDY and EN signal)

Example: the "Verilog environment" may be an AMBA AXI bus port



### Exposing method conditions

- A method's condition can always be exposed as an explicit Bool method
- The examples below are from the BSV library
  - Note: in the lib FIFOF, enq, first and deq are still guarded by their implicit conditions; the notFull and notEmpty methods are just additional methods

```
interface FIFO#(type t);
// enq has "notFull" condition
method Action enq(t x);
// first/deq have "notEmpty" condition
method t first;
method Action deq;
method Action clear;
endinterface: FIFO
```

```
interface FIFOF#(type t);
  // enq has "notFull" condition
  method Action enq(t x);
  // first/deq have "notEmpty" condition
  method t first;
  method Action deq;
  method Action clear;
  method Bool notEmpty;
  method Bool notFull;
  endinterface: FIFOF
```

#### Exposing method conditions

- Implementing exposed conditions is trivial: just replicate the method condition
  - E.g., below, enq()'s condition can Enque is returned explicitly as a Boolean in notFull()
  - Don't worry: the generated code will share a single instance of the condition logic
    - Or, share it explicity by writing let x = canEnque and using x twice

```
module mkFIFOF (FIFOF#(type t));
...
method Action enq(t x) if (canEnque); ... endmethod
method t first if (canDeque); ... endmethod
method Action deq if (canDeque); ... endmethod
method Bool notFull; return canEnque; endmethod
method Bool notEmpty; return canDeque; endmethod
endmodule: mkFIFOF
```



### Removing RDY signals

- The "always\_ready" attribute can be applied to a method to indicate
  - that it is always ready (the compiler will check this!)
  - that the RDY signal must be removed from the generated Verilog

```
(* always_ready = "add, result" *)
module mkAdder (Adder);
Reg#(int) acc <- mkRegA(0);

method Action add(a, b);
acc <= a + b;
endmethod

method int result;
return acc;
endmethod
endmodule</pre>
```

```
interface Adder;
method Action add(int a, int b);
method int result;
endinterface
```

The compiler will check that add and result are always ready to be invoked, i.e. their implicit and explicit conditions are always True

The compiler will not generate any RDY signal for add and result



## Removing ENABLE signals

- The "always\_enabled" attribute can be attached to a method (Action or ActionValue) to indicate
  - that it is assumed True, i.e., the data input signals (args) are driven by the environment on *every* cycle
  - Of course, this implies "always ready", i.e., the method's condition must be always True (otherwise it would be an error to drive the EN signal)
  - that the EN signal must be removed from the generated Verilog

```
(* always_enabled = "add" *)
module mkAdder (Adder);
Reg#(int) acc <- mkRegA(0);

method Action add(a, b);
acc <= a + b;
endmethod

method int result;
return acc;
endmethod
endmodule</pre>
```

```
interface Adder;
method Action add(int a, int b);
method int result;
endinterface
```

The compiler will check that add can be always invoked, i.e. its method condition is always True

The compiler will not generate any EN signal for add

#### Unguarded methods

Suppose a module must read datum on EVERY CYCLE

```
(* always_enabled = "accept" *)
module mkFoo (IfcFoo);
FIFO#(int) fifo <- mkFIFO;
...
method Action accept (int);
fifo.enq(bus);
endmethod
endmodule</pre>
```

The bsc compiler will signal an error, since the enq() method may not be ready (the fifo may be full).

It cannot verify that accept() is always ready, so it cannot allow the always\_enabled spec



#### Unguarded methods

- Answer: use "unguarded" methods, i.e., methods with no conditions, relying on you to guard them explicitly
  - The BSV library provides FIFOs with unguarded methods
  - Warning: these are dangerous! Use with care, only in these "impedance matching" situations

```
(* always_enabled = "accept" *)
module mkFoo (IfcFoo);
FIFO#(int) fifo <- mkUGFIFOF;
...
method Action accept (int);
fifo.enq(bus);
endmethod
endmodule

This will work.
```

Warning: you have to use some other means to avoid buffer overflow!



### Summary: plugging BSV into RTL

 A BSV Action method with one argument that is always ready and always enabled becomes, exactly, a Verilog input port. E.g.,

```
(* always_ready, always_enabled *) method Action accept (int);
```

 A BSV value method that is always ready becomes, exactly, a Verilog output port. E.g.,

```
(* always_ready *)
method int yield_data;
```

 Sec. 13.2.1 shows attributes by which you can control the exact names of these generated Verilog ports

With these capabilities, and unguarded methods, you can produce a Verilog interface with any desired ports, with any desired names, and with any desired behavior (protocol)

• The BSV AXI library has examples of interfaces to the AMBA AXI bus (masters, slaves, etc.)



#### Hands-on

BSV-by-Example book: Examples in Chapter 15





# End

