

# Chapter 7

## Transaction Level Modeling (TLM)

Zainalabedin Navabi

Slides updated by Nooshin Nosrati

© Zainalabedin Navabi – Transaction Level Modeling

# Transaction Level Modeling

Introduction

+ Processing Elements

+ Data

+ Abstract Communications

+ Complete System

# Transaction Level Modeling

## Introduction

### – Processing Elements

Initiators

Targets

Interconnects

### – Data

Generic Payload

## – Abstract Communications

+ Coding Styles

+ TLM-2.0 Transport Interfaces

## – Complete System

Blocking Example

Non-blocking Example

# Transaction Level Modeling

## Introduction

+ Processing Elements

+ Data

+ Abstract Communications

+ Complete System

# Electronic System Level (ESL)

- An abstraction level higher than RTL
  - Little implementation details
  - Abstract timing
- Appropriate for today's complex systems
- An entry level for simulation, synthesis, and test

# Transaction Level Modeling (TLM)

- Means of design at ESL
- Separates communication from computation within a system
- Communications are performed through function calls
- A transaction is the data transfer between two modules

# Why TLM?

---

- Higher simulation speed than RTL
- Design space exploration
- Early verification
- HW/SW communication

# OSCI TLM-2.0 Standard

- The Generic Payload  
Sockets

- Initiator and Target Sockets
  - Simple Sockets
  - Tagged Sockets
  - Multi Sockets

- TLM-2.0 Core Interfaces

- Transport interfaces
  - Blocking transport interface
  - Non-blocking transport interface
- Direct memory interface
- Debug transport interface

- Phases and Base Protocol

Such as RTL  
components:  
Bus, Register, ...

# OSCI TLM Development

Apr 2005

- TLM -1.0
- put, get and transport request-response interfaces

Dec 2006

- TLM-2.0-draft-1
- Generic payload

Nov 2007

- TLM 2.0-draft-2
- nb\_transport
- New payload & extensions

Jun 2008

- TLM-2.0
- Unified interfaces and sockets

July 2009

- TLM-2.0.1
- Minor additions and LRM

\* Open SystemC Initiative, 31 May 2009

# TLM-2.0 library Structure



# Introduction



# TLM-2.0 Library Hierarchy

# Transaction Level Modeling

## Introduction

### – Processing Elements

Initiators

Targets

Interconnects

+ Data

+ Abstract Communications

+ Complete System

# Processing Elements



# Transaction Level Modeling

## Introduction

### + Processing Elements

#### - Data

##### Generic Payload

#### + Data

#### + Abstract Communications

#### + Complete System

# The Generic Payload



- Standard type of the data that is transferred at system level
- Includes some of the typical memory-mapped bus protocols attributes such as:
  - Command, address, data, burst transfers, ...
- Includes an extension mechanism so that applications can add their own specialized attributes

# GP: *tlm\_gp.h* outline

```
7  namespace tlm {  
8  
9    class  
10   tlm_generic_payload;  
11  
12  class tlm_mm_interface { ... };  
13 //-----  
14 // Classes and helper functions for the extension mechanism  
15 //-----  
16 inline unsigned int max_num_extensions(bool increment=false) { ... }  
17 class tlm_extension_base { ... };  
18 template <typename T>  
19 class tlm_extension { ... };  
20 template <typename T>  
21 const  
22 unsigned int tlm_extension<T>::ID = tlm_extension_base::register_extension();  
23  
24 //-----  
25 // enumeration types  
26 //-----  
27 enum tlm_command { ... };  
28 enum tlm_response_status { ... };  
29 #define TLM_BYTE_DISABLED 0x0  
30 #define TLM_BYTE_ENABLED 0xff  
31  
32 //-----  
33 // The generic payload class:  
34 //-----  
35 class tlm_generic_payload { ... };  
36 } // namespace tlm
```

tlm\_gp.h

# GP: the Generic Payload Class

```
82 class tlm_generic_payload {  
371     private:  
372         sc_dt::uint64          m_address;  
373         tlm_command            m_command;  
374         unsigned char*         m_data;  
375         unsigned int           m_length;  
376         tlm_response_status    m_response_status;  
377         bool                  m_dmi;  
378         unsigned char*         m_byte_enable;  
379         unsigned int           m_byte_enable_length;  
380         unsigned int           m_streaming_width;  
381  
382     public:  
383  
384     /* ----- */  
385     /* Dynamic extension mechanism: */  
386     /* ----- */  
387     ...  
502     private:  
503         tlm_array<tlm_extension_base*> m_extensions;  
504         tlm_mm_interface*                 m_mm;  
505         unsigned int                     m_ref_count;  
506     };  
367     ...  
371     private:
```

class *tlm\_generic\_payload*

# GP Attributes

| Attribute           | Type                     | Modifiable?       |
|---------------------|--------------------------|-------------------|
| Command             | tlm_command              | No                |
| Address             | uint64                   | Interconnect only |
| Data pointer        | unsigned char*           | No (array – yes)  |
| Data length         | unsigned int             | No                |
| Byte enable pointer | unsigned char*           | No (array – yes)  |
| Byte enable length  | unsigned int             | No                |
| Streaming width     | unsigned int             | No                |
| DMI hint            | bool                     | Yes               |
| Response status     | tlm_response_status      | Target only       |
| Extensions          | (tlm_extension_base*)[ ] | Yes               |

\* Open SystemC Initiative, 31 May 2009

# GP Attributes, Command

- Shall be set by the **initiator**, and shall not be overwritten by any interconnect component or target

```
enum tlm_command {  
    TLM_READ_COMMAND,  
    TLM_WRITE_COMMAND,  
    TLM_IGNORE_COMMAND  
};
```

Copy from target to data array

Copy from data array to target

Neither, but may use extensions

```
    tlm_command      get_command() const ;  
    void             set_command( const tlm_command command ) ;
```

# GP Attributes, Address

- Shall be set by the **initiator**, but may be overwritten by one or more interconnect components.

```
sc_dt::uint64    get_address() const;  
void            set_address( const sc_dt::uint64 address );
```

# GP Attributes, Data Pointer

- Is a pointer to the data array
- Shall be set by the **initiator**, and shall not be overwritten by any interconnect component or target
- Methods `get_data_ptr()` and `set_data_ptr()` get or set the value of the pointer, not the contents of the array

```
unsigned char* get_data_ptr() const;  
void set_data_ptr( unsigned char* data );
```

# GP Attributes, Data Length

- Shall be set by the **initiator**, and shall not be overwritten by any interconnect component or target
- Shall not be set to 0
  - To transfer zero bytes, the command attribute should be set to `TLM_IGNORE_COMMANDS`
- Data length  $\leq$  BUSWIDTH / 8: single-word transfer
- Data length  $>$  BUSWIDTH / 8: burst transfer

```
unsigned int      get_data_length() const;  
void             set_data_length( const unsigned int length );
```

# GP Attributes, Byte Enable Pointer

- Shall be set by the **initiator**, and shall not be overwritten by any interconnect component or target
- may be used to create burst transfers where the address increment between each beat is greater than the number of significant bytes transferred on each beat, or to place words in selected byte lanes of a bus
- The number of elements in the byte enable array shall be given by the byte enable length attribute
- A value of 0
  - Indicates that corresponding byte is disabled
- A value of 0xff
  - Indicates that the corresponding byte is enabled

```
unsigned char* get_byte_enable_ptr() const;  
void set_byte_enable_ptr( unsigned char* );
```

# GP Attributes, Byte Enable Length

- Shall be set by the **initiator**, and shall not be overwritten by any interconnect component or target
- Shall be calculated using the formula

`byte_enable_array_index = data_array_index % byte_enable_length`

```
unsigned int          get_byte_enable_length() const;  
void                set_byte_enable_length( const unsigned int );
```

# GP Attributes, Streaming Width

- Determines the number of bytes transferred on each beat
- The bytes within the data array have a corresponding local addresses within the component accessing the GP
  - The lowest address: the value of the address attribute
  - The highest address:  
**address\_attribute + streaming\_width – 1**

unsigned int  
void

```
get_streaming_width() const;  
set_streaming_width( const unsigned int );
```

# GP Attributes, DMI Allowed

- Provides a hint to an initiator that it may try to obtain a direct memory pointer
- The **target** should set this attribute to true if the transaction could have been done through DMI

```
void  
bool
```

```
set_dmi_allowed( bool );  
is_dmi_allowed() const;
```

# GP Attributes, Response Status

- Shall be set to **TLM\_INCOMPLETE\_RESPONSE** by the **initiator**
- May be overwritten by the **target**, but should not be overwritten by any **interconnect component**

tlm\_response\_status  
Void  
std::string  
bool  
bool

get\_response\_status() const;  
set\_response\_status( const tlm\_response\_status );  
get\_response\_string();  
is\_response\_ok();  
is\_response\_error();

# GP Attributes, Response Status, Cont.

- The target may set the response status attribute:
  - **TLM\_OK\_RESPONSE**: Indicates that target was able to execute the command successfully
  - One of the responses listed in the following table to indicate status

| enum tlm_response_status       | Meaning                                         |
|--------------------------------|-------------------------------------------------|
| TLM_OK_RESPONSE                | Successful                                      |
| TLM_INCOMPLETE_RESPONSE        | Transaction not delivered to target. (Default)  |
| TLM_ADDRESS_ERROR_RESPONSE     | Unable to act on address                        |
| TLM_COMMAND_ERROR_RESPONSE     | Unable to execute command                       |
| TLM_BURST_ERROR_RESPONSE       | Unable to act on data length or streaming width |
| TLM_BYTE_ENABLE_ERROR_RESPONSE | Unable to act on byte enable                    |
| TLM_GENERIC_ERROR_RESPONSE     | Any other error                                 |

# GP Extensions

- Generic payload has an array-of-pointers to extensions
- One pointer per extension type
- Every transaction can potentially carry every extension type
- Flexible mechanism

```
template <typename T> T* set_extension ( T* ext );
```

Freed by ref counting

```
template <typename T> T* set_auto_extension ( T* ext );
```

Clears pointer, not  
extension object

```
template <typename T> T* get_extension() const;
```

mm => convert to auto  
no mm => free extension object

```
template <typename T> void clear_extension ();
```

```
template <typename T> void release_extension ();
```

# Transaction Level Modeling

Introduction

+ Processing Elements

+ Data

+ IP Cores

+ System

– Abstract Communications

+ Coding Styles

- Loosely Timed
- Approximately Timed

+ TLM-2.0 Transport Interfaces

• Blocking Transport

- Path (Forward & Return)
- Socket (Simple Socket)

• Simple Transport Example

• Non-blocking Transport

- Path (Forward, Backward & Return)
- Socket (Tagged & Multi Socket)
- Phases & Base Protocol

+ Complete System

# Coding Styles

## ○ Loosely-timed

- Each transaction has 2 timing points: *begin* and *end*
- Uses **blocking transport interface**

## ○ Approximately-timed

- Also referred as: cycle-approximate or cycle-count-accurate
- Sufficient for architectural exploration
- Each transaction has 4 timing points (extensible)
- Uses **non-blocking transport interface**

# Transaction Level Modeling

Introduction

+ Processing Elements

+ Data

- Abstract Communications

+ Coding Styles

- Loosely Timed
- Approximately Timed

+ TLM-2.0 Transport Interfaces

• Blocking Transport

- Path (Forward & Return)
- Socket (Simple Socket)

• Simple Transport Example

• Non-blocking Transport

- Path (Forward, Backward & Return)
- Socket (Tagged & Multi Socket)
- Phases & Base Protocol

+ Complete System

# Transport Interfaces

**Initiator**  
Calls **b\_transport**  
Calls **nb\_transport\_fw**  
Implements **nb\_transport\_bw**

**Target**  
Implements **b\_transport**  
Implements **nb\_transport\_fw**  
Calls **nb\_transport\_bw**

**tlm\_blocking\_transport\_if**

```
void b_transport( TRANS& , sc_time& );
```

**tlm\_fw\_nonblocking\_transport\_if**

```
tlm_sync_enum nb_transport_fw( TRANS& , PHASE& , sc_time& );
```

**tlm\_bw\_nonblocking\_transport\_if**

```
tlm_sync_enum nb_transport_bw( TRANS& , PHASE& , sc_time& );
```

returns a value  
indicating  
whether or not  
the return path  
was used

# Blocking Transport

- Blocking transport interface

- Includes timing annotation
- Typically used with loosely-timed coding style
- Forward path only

```
template < typename TRANS = tlm_generic_payload >

class tlm_blocking_transport_if : public virtual sc_core::sc_interface {
public:
    virtual void b_transport( TRANS& trans , sc_core::sc_time& t ) = 0;
};
```

# Blocking Transport: Path

## ○ Forward path

- Is the calling path by which an initiator or interconnect component makes interface method calls forward in the direction of another interconnect component or the target

## ○ Return path

- Is the return path by which an initiator, target or interconnect component returns immediately from the interface method that has been called



# Blocking Transport: Path

## ○ Timing Diagram

Initiator is blocked until return from b\_transport



# Transaction Level Modeling

## Introduction

### + Processing Elements

### + Data

## - Abstract Communications

### + Coding Styles

### + TLM-2.0 Transport Interfaces

- Blocking Transport
  - Path (Forward & Return)
  - [blockingBusbasedMemoryRW Example](#)
  - Socket (Simple Socket)
  - Simple Transport Example
  - Non-blocking Transport

### + Complete System



# Blocking Transport: Path

- Example 1: blockingBusbasedMemoryRW

Generic payload

Initiator  
(MemoryWriter  
Reader)

Target  
(memoryUnit)

# Blocking Transport: Path

- Example 1: blockingBusbasedMemoryRW: Initiator, memWriterReader



```
memWriterReader.h ✘ X memWriteRead_TB.h      memoryUnit.cpp*      memWriteRead.cpp
blockingBusbasedMemoryRW      (Global Scope)

1 #include <systemc.h>
2 #include "tlm.h"
3
4 class memWriterReader : public sc_module {
5 public:
6     sc_port<tlm::tlm_blocking_transport_if<>> memWriterReaderBus;
7
8     SC_CTOR(memWriterReader) :blockWriteRead(0) {
9         blockWriteRead = new tlm::tlm_generic_payload;
10
11         for (int i = 0; i < 4; i++) *(data+i) = i+192;
12
13         SC_THREAD(memWritingReading);
14     }
15
16     void memWritingReading();
17     tlm::tlm_generic_payload* blockWriteRead;
18
19     sc_lv<8> data[5];
20 };
```

# Blocking Transport: Path

- Example 1:  
blockingBusbasedMemoryRW:  
Initiator, memWriterReader

```
memWriteRead.h      memWriteRead_TB.h    memoryUnit.cpp*    memWriteRead.cpp + X
blockingBusbasedMemoryRW   (Global Scope)
```

memWriterReader.cpp

```
1 #include "memWriteRead.h"
2
3 void memWriterReader::memWritingReading(){
4     sc_time blockedTime = sc_time(13, SC_NS);
5     sc_time pauseTime = sc_time(15, SC_NS);
6
7     for (int i = 0; i < 111; i += 11) {
8         tlm::tlm_command cmd = (tlm::tlm_command)(rand() % 2);
9         if (cmd == tlm::TLM_WRITE_COMMAND) {
10             data[0] = (sc_lv<8>) (i+5);
11             data[1] = (sc_lv<8>) (i+6);
12             data[2] = (sc_lv<8>) (i+7);
13             data[3] = (sc_lv<8>) (i+8);
14             data[4] = (sc_lv<8>) (i+9);
15         }
16
17         blockWriteRead->set_command( cmd );
18         blockWriteRead->set_address( i );
19         blockWriteRead->set_data_ptr( (unsigned char*) data );
20         blockWriteRead->set_data_length( 5 );
21         blockWriteRead->set_streaming_width( 5 );
22         blockWriteRead->set_byte_enable_ptr( 0 );
23         blockWriteRead->set_dmi_allowed( false );
24         blockWriteRead->set_response_status( tlm::TLM_INCOMPLETE_RESPONSE );
25
26         memWriterReaderBus->b_transport( *blockWriteRead, blockedTime );
27
28         cout << "WR: " << (cmd ? 'W' : 'R') << ", @" << i << " data:";
29         sc_lv<8> vv;
30         for(int j=0; j<5; j++) {vv=data[j]; cout << vv << " ;"} cout << '\n';
31
32         wait(pauseTime);
33     }
34 }
```

Set GP parameters

= data\_length to indicate no streaming

Calling blocking transport

initial value

On Level Modeling

# Blocking Transport: Path

- Example 1: blockingBusbasedMemoryRW: Target, *memoryUnit*



memoryUnit.h

```
memWriteRead_TB.cpp      memWriteRead_Main.cpp      Bus.h      memoryUnit.h + X
blockingBusbasedMemoryRW      (Global Scope)
```

```
1 #include <systemc.h>
2 #include "tlm.h"
3 #include "Bus.h"

4

5 SC_MODULE (memoryUnit) {
6     sc_export<tlm::tlm_blocking_transport_if>> memSideBus;
7     bus *membus;
8     static const int SIZE=256;

9
10    SC_CTOR(memoryUnit) : memSideBus("memorySideBus")
11    {
12        membis = new bus();
13        memSideBus(*membus); // Can go in the TB
14        for (int i = 0; i < SIZE; i++)
15            memArray[i] = (sc_lv<8>) (i%256 + 192);
16    }
17    public:
18        virtual void b_transport( tlm::tlm_generic_payload&, sc_time& );
19        sc_lv<8> memArray[SIZE];
20    };
21
```

# Blocking Transport: Path

```

memWriteRead.h      memWriteRead_TB.h      memoryUnit.cpp*  memWriteRead.cpp
blockingBusbasedMemoryRW  (Global Scope)
1 #include "memoryUnit.h"
2
3 void memoryUnit::b_transport( tlm::tlm_generic_payload& gotThis,
4                               sc_time& delayValue )
5 {
6     tlm::tlm_command cmd = gotThis.get_command();
7     uint64 adr = gotThis.get_address();
8     unsigned char* ptr = gotThis.get_data_ptr();
9     unsigned int len = gotThis.get_data_length();
10    unsigned char* byt = gotThis.get_byte_enable_ptr();
11    unsigned int wid = gotThis.get_streaming_width();
12
13    if (adr >= uint64(SIZE) || byt != 0 || len > 5 || wid < len)
14        SC_REPORT_ERROR("TLM-2.0: ","Inconsistent Generic Payload");
15
16    unsigned int i;
17    if ( cmd == tlm::TLM_READ_COMMAND ){
18        for(i=0; i<len; i=i+1) {
19            *(ptr+i) = *((unsigned char*) (memArray+adr+i));
20        }
21    }
22    else if ( cmd == tlm::TLM_WRITE_COMMAND ){
23        for(i=0; i<len; i=i+1) {
24            *((unsigned char*) (memArray+adr+i)) = *(ptr+i);
25        }
26    }
27
28    gotThis.set_response_status( tlm::TLM_OK_RESPONSE );
29    wait(delayValue);
30 }

```

**memoryUnit.cpp**

- Example 1:  
**blockingBusbasedMemoryRW:**  
**Target, *memoryUnit***

Obliged to check address range and  
check for unsupported features, using  
the SystemC report handler is an  
acceptable way of signalling an error

Obliged to implement read  
and write commands

Obliged to set response  
status to indicate successful  
completion

# Blocking Transport: Path

- Example 1: blockingBusbasedMemoryRW: Interface Defenition & Implementation , Bus



Bus.h

```
Bus.h* ✘ X
blockingBusbasedMemoryRW (Global Scope)
1 #include <systemc.h>
2 #include "tlm.h"
3
4 class bus : public tlm::tlm_blocking_transport_if<>
5 {
6 public:
7     bus() {};
8     ~bus() {};
9
10    virtual void b_transport(tlm::tlm_generic_payload&, sc_time&){};
11};
```

# Blocking Transport: Path

- Example 1: blockingBusbasedMemoryRW: Testbench



```
memWriteRead_TB.cpp      Bus.h      memoryUnit.h      memWriteRead.h      memWriteRead_TB.h ✘ X ▾
blockingBusbasedMemoryRW  (Global Scope)
```

```
1 #include "memWriteRead.h"
2 #include "memoryUnit.h"
3
4 SC_MODULE (memWriteRead_TB) {
5     memWriterReader* MW1;
6     memoryUnit* MU1;
7
8     SC_CTOR(memWriteRead_TB)
9     {
10        MU1 = new memoryUnit("memoryUnit");
11        // MU1->memSideBus(*MU1->membus); // This is in memoryUnit
12        MW1 = new memWriterReader("memoryWriter");
13        MW1->memWriterReaderBus(*MU1->membus);
14    }
15}
16
```

memoryWrite\_TB.h

# Blocking Transport: Path

- Example 1: blockingBusbasedMemoryRW: Main

memWriteRead\_Main.cpp



```
memoryUnit.cpp*      memWriteRead_TB.cpp      memWriteRead_TB.h      memWriteRead_Main.cpp X
blockingBusbasedMemoryRW (Global Scope)
1 #include "memWriteRead_TB.h"
2
3 int sc_main(int argc, char* argv[])
4 {
5     memWriteRead_TB TB1("memoryWriteRead_TB");
6     sc_start();
7     return 0;
8 }
```

# Blocking Transport: Path

- Example 1: blockingBusbasedMemoryRW: Output

```
SystemC 2.3.2-Accellera --- Nov 16 2018 05:36:55  
Copyright (c) 1996-2017 by all Contributors,  
ALL RIGHTS RESERVED
```

```
WR: W, @0 data:00000101 00000110 00000111 00001000 00001001  
WR: W, @11 data:00010000 00010001 00010010 00010011 00010100  
WR: R, @22 data:00010000 00010001 00010010 00010011 00010100  
WR: R, @33 data:00010000 00010001 00010010 00010011 00010100  
WR: W, @44 data:00110001 00110010 00110011 00110100 00110101  
WR: R, @55 data:00110001 00110010 00110011 00110100 00110101  
WR: R, @66 data:00110001 00110010 00110011 00110100 00110101  
WR: R, @77 data:00110001 00110010 00110011 00110100 00110101  
WR: R, @88 data:00110001 00110010 00110011 00110100 00110101  
WR: R, @99 data:00110001 00110010 00110011 00110100 00110101  
WR: W, @110 data:01110011 01110100 01110101 01110110 01110111
```

# Blocking Transport: Socket (Simple)

- Combine a port with an export
- Provide methods to bind port and export of the paths
- Offer strong type checking when binding sockets parameterized with incompatible protocol types
- The classes **tlm\_initiator\_socket** and **tlm\_target\_socket** are typically used directly by applications
  - Belong to the interoperability layer of the TLM-2.0 standard



# Blocking Transport: Socket (Concepts)

## tlm\_initiator\_socket.h

```

24  namespace tlm {
25      template <unsigned int BUSWIDTH = 32,
26                  typename FW_IF = tlm_fw_transport_if<>,
27                  typename BW_IF = tlm_bw_transport_if<>>
28      class tlm_base_initiator_socket_b
29      {
30      public:
31          virtual ~tlm_base_initiator_socket_b() { ... }
32          virtual sc_core::sc_port_b<FW_IF> & get_base_port() = 0;
33          virtual BW_IF & get_base_interface() = 0;
34          virtual sc_core::sc_export<BW_IF> & get_base_export() = 0;
35      };
36
37
38      template <unsigned int BUSWIDTH,
39                  typename FW_IF,
40                  typename BW_IF> class tlm_base_target_socket_b;
41
42      template <unsigned int BUSWIDTH,
43                  typename FW_IF,
44                  typename BW_IF,
45                  int N> class tlm_base_target_socket;
46
47      template <unsigned int BUSWIDTH = 32,
48                  typename FW_IF = tlm_fw_transport_if<>,
49                  typename BW_IF = tlm_bw_transport_if<>,
50                  int N = 1>>
51      class tlm_base_initiator_socket { ... };
52
53      /* ... */
54
55      template <unsigned int BUSWIDTH = 32,
56                  typename TYPES = tlm_base_protocol_types,
57                  int N = 1 ... >
58      class tlm_initiator_socket { ... };
59
60  } // namespace tlm

```

## ◎ Class Definition

### Port and export

# Blocking Transport: Socket (Concepts)

```

59 class tlm_base_initiator_socket : public tlm_base_initiator_socket_b<BUSWIDTH, FW_IF, BW_IF>,
60   public sc_core::sc_port<FW_IF, N...>
61 {
62 public:
63   typedef FW_IF           fw_interface_type;
64   typedef BW_IF           bw_interface_type;
65   typedef sc_core::sc_port<fw_interface_type, N...> port_type;
66   typedef sc_core::sc_export<bw_interface_type> export_type;
67   typedef tlm_base_target_socket_b<BUSWIDTH,
68                                 fw_interface_type,
69                                 bw_interface_type> base_target_socket_type;
70   typedef tlm_base_initiator_socket_b<BUSWIDTH,
71                                     fw_interface_type,
72                                     bw_interface_type> base_type;
73
74   template <unsigned int, typename, typename, int...>
75     friend class tlm_base_target_socket;
76
77 public:
78   tlm_base_initiator_socket() { ... }
79   explicit tlm_base_initiator_socket(const char* name) { ... }
80   virtual const char* kind() const { ... }
81   unsigned int get_bus_width() const { ... }
82
83   void bind(base_target_socket_type& s) { ... }
84   void operator() (base_target_socket_type& s)...
85   void bind(base_type& s) { ... }
86   void operator() (base_type& s)...
87   void bind(bw_interface_type& ifs) { ... }
88   void operator() (bw_interface_type& s) { ... }
89
90   // Implementation of pure virtual functions of base class
91   virtual sc_core::sc_port_b<FW_IF> & get_base_port() { return *this; }
92   virtual BW_IF & get_base_interface() { return m_export; }
93   virtual sc_core::sc_export<BW_IF> & get_base_export() { ... }
94
95 protected:
96   export_type m_export;
97 };
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169

```

tlm\_initiator\_socket.h:  
base class

○ Class Definition, Cont.

Defining fw and bw interfaces

Defining port and export

Binding and overloading of ()

Base class for  
initiator sockets,  
providing binding  
methods

# Blocking Transport: Socket (Concepts)

## ○ Class Definition, Cont.

tlm\_initiator\_socket.h:  
the convenience class

Is inherited from the  
*tlm\_initiator\_socket\_base* class

```
182 [+] class tlm_initiator_socket :  
183 [+] {  
184 [+]     public tlm_base_initiator_socket <BUSWIDTH,  
185 [+]         tlm_fw_transport_if<TYPES>,  
186 [+]         tlm_bw_transport_if<TYPES>,  
187 [+]         N...>  
188 [+]     {  
189 [+]         public:  
190 [+]             tlm_initiator_socket() :  
191 [+]                 tlm_base_initiator_socket<BUSWIDTH,  
192 [+]                     tlm_fw_transport_if<TYPES>,  
193 [+]                     tlm_bw_transport_if<TYPES>,  
194 [+]                     N... >() { ... }  
195 [+]         explicit tlm_initiator_socket(const char* name) :  
196 [+]             tlm_base_initiator_socket<BUSWIDTH,  
197 [+]                 tlm_fw_transport_if<TYPES>,  
198 [+]                 tlm_bw_transport_if<TYPES>,  
199 [+]                 N... >(name)  
200 [+]         {  
201 [+]             ...  
202 [+]         }  
203 [+]     }  
204 [+]     virtual const char* kind() const { ... }  
205 [+] }  
206 [+] ;
```

# Blocking Transport: Socket (Concepts)

## ○ Class Hierarchy



# Blocking Transport: Socket (Simple)

## ◦ Simple Sockets

- Simple to use
- Derived from the interoperability layer sockets **tlm\_initiator\_socket** and **tlm\_target\_socket**
- Provide methods for registering callback methods instead of binding sockets to objects

# Blocking Transport: Socket (Simple)

```

70 class process : public tlm::tlm_bw_transport_if<TYPES>
71 {
72 public:
73     typedef sync_enum_type (MODULE::*TransportPtr)(transaction_type&,
74                                         phase_type&,
75                                         sc_core::sc_time&);
76     typedef void (MODULE::*InvalidateDirectMemPtr)(sc_dt::uint64,
77                                         sc_dt::uint64);
78
79     process(const std::string& name) { ... }
86
87     void set_transport_ptr(MODULE* mod, TransportPtr p) { ... }
100    void set_invalidate_direct_mem_ptr(MODULE* mod, InvalidateDirectMemPtr p) { ... }
113    sync_enum_type nb_transport_bw(transaction_type& trans, phase_type& phase, sc_core::sc_time& t) { ... }
128    void invalidate_direct_mem_ptr(sc_dt::uint64 start_range,
129                                    sc_dt::uint64 end_range) { ... }
138
113    sync_enum_type nb_transport_bw(transaction_type& trans, phase_type& phase, sc_core::sc_time& t)
139    {
140        if (m_transport_ptr) {
141            // forward call
142            assert(m_mod);
143            return (m_mod->*m_transport_ptr)(trans, phase, t);
144
145        } else {
146            std::stringstream s;
147            s << m_name << ": no transport callback registered";
148            SC_REPORT_ERROR("/OSCI_TLM-2/simple_socket", s.str().c_str());
149        }
150        return tlm::TLM_ACCEPTED; // unreachable code
151    }
152
153 };

```

- Simple Initiator  
Socket: Class Definition

# Transaction Level Modeling

## Introduction

### + Processing Elements

### + Data

## - Abstract Communications

### + Coding Styles

### + TLM-2.0 Transport Interfaces

- Blocking Transport

- Path (Forward & Return)
  - Socket (Simple Socket)

- [blockingSocketbasedMemoryRW Example](#)

- Simple Transport Example

- Non-blocking Transport

### + Complete System



# Blocking Transport: Socket (Simple)

- Example 2: blockingSocketbasedMemoryRW: Initiator, *memWriterReader*

The screenshot shows a code editor with two files open:

- memWriterReader.h**: A header file defining an initiator socket for the *memWriterReader* class.
- memWriteRead\_Main.cpp**: A source file containing the implementation of the *memWriterReader* class and a main function for memory initialization.

**memWriterReader.h** content:

```
#include <systemc.h>
#include "tlm.h"
#include "tlm_utils/simple_target_socket.h"
#include "tlm_utils/simple_initiator_socket.h"

class memWriterReader : public sc_module {
public:
    tlm_utils::simple_initiator_socket<memWriterReader> memWriterReaderSocket;

    SC_CTOR(memWriterReader) : memWriterReaderSocket("Writer-Socket"),
                                blockWriteRead(0) {
        blockWriteRead = new tlm::tlm_generic_payload;

        for (int i = 0; i < 4; i++) *(data+i) = i+192;
    }

    SC_THREAD(memWritingReading);
    tlm::tlm_generic_payload* blockWriteRead;

    sc_lv<8> data[5];
};
```

**memWriteRead\_Main.cpp** content:

```
1 #include <systemc.h>
2 #include "memWriterReader.h"
3 #include "memWriteRead.h"
4 #include "memoryUnit.h"
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
```

Annotations on the code:

- An annotation points to the `memWriterReaderSocket` definition in **memWriterReader.h** with the text: "Defining the Initiator socket and specializing it for *memWriterReader*".
- A green box highlights the memory initialization loop in **memWriteRead\_Main.cpp** with the text: "Initializing memory".

# Blocking Transport: Socket (Simple)

```

memWriteRead_Main.cpp      memWriteRead.cpp  memoryUnit.h      memWriteRead.h
blockingSocketbasedMemoryRW  (Global Scope)
1  #include "memWriteRead.h"
2
3 void memWriterReader::memWritingReading(){
4     sc_time blockedTime = sc_time(13, SC_NS);
5     sc_time pauseTime = sc_time(15, SC_NS);
6
7     for (int i = 0; i < 111; i += 11) {
8         tlm::tlm_command cmd = (tlm::tlm_command)(rand() % 2);
9         if (cmd == tlm::TLM_WRITE_COMMAND) {
10             data[0] = (sc_lv<8>) (i+5);
11             data[1] = (sc_lv<8>) (i+6);
12             data[2] = (sc_lv<8>) (i+7);
13             data[3] = (sc_lv<8>) (i+8);
14             data[4] = (sc_lv<8>) (i+9);
15         }
16
17         blockWriteRead->set_command( cmd );
18         blockWriteRead->set_address( i );
19         blockWriteRead->set_data_ptr( (unsigned char*) data );
20         blockWriteRead->set_data_length( 5 );
21         blockWriteRead->set_streaming_width( 5 );
22         blockWriteRead->set_byte_enable_ptr( 0 );
23         blockWriteRead->set_dmi_allowed( false );
24         blockWriteRead->set_response_status( tlm::TLM_INCOMPLETE_RESPONSE );
25
26         memWriterReaderSocket->b_transport( *blockWriteRead, blockedTime );
27
28         if ( blockWriteRead->is_response_error() )
29             SC_REPORT_ERROR("TLM-2", "Error in memory handling of b_transport");
30
31         cout << " At time: " << sc_time_stamp() << "WR: "
32             << (cmd ? 'W' : 'R') << ", Iteration:" << i << " data:";
33         sc_lv<8> vv;
34         for(int j=0; j<5; j++) {vv=data[j]; cout << vv << " ";} cout << '\n';
35         // wait(pauseTime);
36     }
37 }

```

memWriterReader.cpp

- Example 2:  
blockingSocketbasedMemoryRW:  
Initiator, *memWriterReader*

Set GP parameters

= data\_length to indicate no streaming

Mandatory initial value

Mandatory initial value  
Initiator changed to check response status

action Level Modeling

# Blocking Transport: Socket (Simple)

- Example 2: blockingSocketbasedMemoryRW: Target, *memoryUnit*

The screenshot shows a code editor with two files open:

- memoryUnit.h**: A header file containing the declaration of the `memoryUnit` module.
- memWriteRead\_Main.cpp**: A source file containing the implementation of the `memoryUnit` module.

The `memWriteRead_Main.cpp` file contains the following code:

```
#include <systemc.h>
#include "tlm.h"
#include "tlm_utils/simple_target_socket.h"
#include "tlm_utils/simple_initiator_socket.h"

SC_MODULE (memoryUnit) {
    tlm_utils::simple_target_socket<memoryUnit> memSideSocket;

    static const int SIZE=256;

    SC_CTOR(memoryUnit) : memSideSocket("memorySideSocket")
    {
        memSideSocket.register_b_transport(this, &memoryUnit::b_transport);

        for (int i = 0; i < SIZE; i++)
            memArray[i] = (sc_lv<8>) (i%256 + 192);
    }

    public:
        virtual void b_transport( tlm::tlm_generic_payload&, sc_time& );
        sc_lv<8> memArray[SIZE];
};


```

Annotations explain specific parts of the code:

- Defining the target socket and specializing it for *memoryUnit***: Points to the line `tlm_utils::simple_target_socket<memoryUnit> memSideSocket;`.
- Register callback for incoming *b\_transport* interface method call**: Points to the line `memSideSocket.register_b_transport(this, &memoryUnit::b_transport);`.
- Initializing memory**: Points to the line `for (int i = 0; i < SIZE; i++) memArray[i] = (sc_lv<8>) (i%256 + 192);`.

# Blocking Transport: Socket (Simple)

```

memWriteRead.cpp      memoryUnit.h      memWriteRead.h      memoryUnit.cpp + X
blockingSocketbasedMemoryRW  (Global Scope)
1 #include "memoryUnit.h"
2
3 void memoryUnit::b_transport( tlm::tlm_generic_payload& gotThis,
4                               sc_time& delayValue )
5 {
6     tlm::tlm_command cmd = gotThis.get_command();
7     uint64 adr = gotThis.get_address();
8     unsigned char* ptr = gotThis.get_data_ptr();
9     unsigned int len = gotThis.get_data_length();
10    unsigned char* byt = gotThis.get_byte_enable_ptr();
11    unsigned int wid = gotThis.get_streaming_width();
12
13    if (adr >= uint64(SIZE) || byt != 0 || len > 5 || wid < len)
14        SC_REPORT_ERROR("TLM-2.0: ","Inconsistent Generic Payload");
15
16    unsigned int i;
17
18    if ( cmd == tlm::TLM_READ_COMMAND ){
19        for(i=0; i<len; i=i+1) {
20            *(ptr+i) = *((unsigned char*) (memArray+adr+i));
21        }
22    }
23    else if ( cmd == tlm::TLM_WRITE_COMMAND ){
24        for(i=0; i<len; i=i+1) {
25            *((unsigned char*) (memArray+adr+i)) = *(ptr+i);
26        }
27    }
28
29    gotThis.set_response_status( tlm::TLM_OK_RESPONSE );
30    wait(delayValue);
31 }

```

memoryUnit.cpp

○ Example 2:  
**blockingSocketbasedMemoryRW:**  
 Target, *memoryUnit*

Obliged to check address range and  
 check for unsupported features, using  
 the SystemC report handler is an  
 acceptable way of signalling an error

Obliged to implement read  
 and write commands

The *b\_transport*  
 method may call wait,  
 directly or indirectly

Obliged to set response  
 status to indicate successful  
 completion

# Blocking Transport: Socket (Simple)

- Example 2: blockingSocketbasedMemoryRW: Testbench

The screenshot shows a simulation environment with a memory unit component. A memWriterReader and memoryUnit component are instantiated and connected to the memory unit's sockets.

```

memWriteRead.cpp      memoryUnit.h      memoryUnit.cpp      memWriteRead_TB.h
blockingSocketbasedMemoryRW  (Global Scope)
1 #include "memWriteRead.h"
2 #include "memoryUnit.h"
3
4 SC_MODULE (memWriteRead_TB) {
5     memWriterReader* MW1;
6     memoryUnit* MU1;
7
8     SC_CTOR(memWriteRead_TB)
9     {
10        MW1 = new memWriterReader("memoryWriter");
11        MU1 = new memoryUnit("memoryUnit");
12        MW1->memWriterReaderSocket.bind( MU1->memSideSocket );
13    }
14}

```

A yellow callout box labeled "memoryWrite\_TB.h" points to the memWriteRead\_TB.h file. An orange arrow labeled "Instantiate components" points to the instantiation code in the constructor. A green arrow points from a green callout box containing the following text to the bind statement in the code:

- One initiator is bound directly to one target with no intervening bus
- Bind initiator socket to target socket

- One initiator is bound directly to one target with no intervening bus
- Bind initiator socket to target socket

# Blocking Transport: Socket (Simple)

- Example 2: blockingSocketbasedMemoryRW: Main

memWriteRead\_Main.cpp



```
memoryUnit.h      memoryUnit.cpp      memWriteRead_TB.h      memWriteRead_Main.cpp + X
blockingSocketbasedMemoryRW      (Global Scope)
1  #include "memWriteRead_TB.h"
2
3  int sc_main(int argc, char* argv[])
4  {
5      memWriteRead_TB TB1("memoryWriteRead_TB");
6      sc_start();
7      return 0;
8 }
```

# Blocking Transport: Socket (Simple)

- Example 2: blockingSocketbasedMemoryRW: Output

```
SystemC 2.3.2-Accellera --- Nov 16 2018 05:36:55
Copyright (c) 1996-2017 by all Contributors,
ALL RIGHTS RESERVED

At time: 13 nsWR: W, Iteration:0 data:00000101 00000110 00000111 00001000 00001001
At time: 26 nsWR: W, Iteration:11 data:00010000 00010001 00010010 00010011 00010100
At time: 39 nsWR: R, Iteration:22 data:00010000 00010001 00010010 00010011 00010100
At time: 52 nsWR: R, Iteration:33 data:00010000 00010001 00010010 00010011 00010100
At time: 65 nsWR: W, Iteration:44 data:00110001 00110010 00110011 00110100 00110101
At time: 78 nsWR: R, Iteration:55 data:00110001 00110010 00110011 00110100 00110101
At time: 91 nsWR: R, Iteration:66 data:00110001 00110010 00110011 00110100 00110101
At time: 104 nsWR: R, Iteration:77 data:00110001 00110010 00110011 00110100 00110101
At time: 117 nsWR: R, Iteration:88 data:00110001 00110010 00110011 00110100 00110101
At time: 130 nsWR: R, Iteration:99 data:00110001 00110010 00110011 00110100 00110101
At time: 143 nsWR: W, Iteration:110 data:01110011 01110100 01110101 01110110 01110111
```

# Transaction Level Modeling

## Introduction

### + Processing Elements

### + Data

### + System

## - Abstract Communications

### + Coding Styles

- Loosely Timed
- Approximately Timed

### + TLM-2.0 Transport Interfaces

#### • Blocking Transport

- Path (Forward & Return)
- Socket (Simple Socket)

#### • Simple Transport Example

#### • Non-blocking Transport

- Path (Forward, Backward & Return)
- Socket (Tagged & Multi Socket)
- Phases & Base Protocol

## + Complete System

# Transaction Level Modeling

## Introduction

### + Processing Elements

### + Data

### - Abstract Communications

#### + Coding Styles

#### + TLM-2.0 Transport Interfaces

- Blocking Transport
- Simple Transport Example
- Non-blocking Transport

### + Complete System



# Simple Transport Example

## Example 3: Simple Transport: *interfaceClasses*



The image shows a code editor window with four tabs: `simpleTransport_Main.cpp*`, `interfaceClasses.cpp`, `channelTransport.h`, and `interfaceClasses.h`. The `interfaceClasses.h` tab is highlighted with a yellow box. The code in `interfaceClasses.h` defines a transport interface:

```
#include <systemc.h>

class transport_if : virtual public sc_interface
{
public:
    bool loading;
    sc_event INCOMPLETE;
    sc_event COMPLETE;
    virtual void transport(sc_lv<8> &) = 0;
};
```

# Simple Transport Example

- Example 3: Simple Transport: *channelTransport*



The background of the slide features a complex TLM-2.0 transport interface diagram. It consists of several vertical columns of components, primarily red and purple rectangles representing FIFO buffers or registers. These components are interconnected by a dense network of lines and arrows, representing data paths and handshaking signals. The interface is organized into multiple channels, each with its own set of inputs and outputs. The overall complexity suggests a high-level system-on-chip (SoC) architecture where multiple functional blocks communicate via a shared bus-like structure.

channelTransport.h

```
simpleTransport_Main.cpp*      simpleTransport.cpp      interfaceClasses.cpp      channelTransport.h ✎ X
Simple Transport      (Global Scope)
1   #include "interfaceClasses.h"
2
3   class channelTransport : public transport_if
4   {
5       sc_lv<8> saved;
6       sc_event put_event, get_event;
7       public:
8           channelTransport() {};
9           ~channelTransport() {};
10          void transport (sc_lv<8> &data);
11      };

```

# Simple Transport Example

## ○ Example 3: Simple Transport: *channelTransport*

```
interfaceClasses.h      simpleTransport.h      simpleTransport_TB.h      channelTransport.cpp + X
Simple Transport          (Global Scope)
1  #include "channelTransport.h"
2
3  void channelTransport::transport(sc_lv<8> &data) {
4      if (loading == true) {
5          saved = data;
6          wait(10, SC_NS);
7      }
8      else {
9          data = saved;
10         wait(15, SC_NS);
11     }
12 }
```

# Simple Transport Example

## Example 3: Simple Transport: *simpleTransport*



The screenshot shows a software development environment with a code editor window. The code editor has tabs for "simpleTransport\_Main.cpp\*", "channelTransport.h", "interfaceClasses.h", and "simpleTransport.h". The "simpleTransport.h" tab is highlighted with a yellow callout box. The code in "simpleTransport.h" defines two classes: "transmitter" and "receiver", both derived from "sc\_module". The "transmitter" class has a public port "out" of type "transport\_if". The "SC\_CTOR(transmitter)" constructor starts a thread named "transportingOut". The "void transportingOut()" function is defined. The "receiver" class has a public export "in" of type "transport\_if" and a pointer "insideTargetChannel" to a "channelTransport" object. The "SC\_CTOR(receiver)" constructor initializes "insideTargetChannel" to a new "channelTransport" object and connects its "in" port to "insideTargetChannel". It also starts a thread named "transportingIn". The "void transportingIn()" function is defined.

```
#include "channelTransport.h"

class transmitter : public sc_module {
public:
    sc_port<transport_if> out;

    SC_CTOR(transmitter) {
        SC_THREAD (transportingOut);
    }
    void transportingOut();
};

class receiver : public sc_module {
public:
    sc_export<transport_if> in;
    channelTransport* insideTargetChannel;

    SC_CTOR(receiver) {
        insideTargetChannel = new channelTransport;
        in(*insideTargetChannel);
        SC_THREAD(transportingIn);
    }
    void transportingIn();
};
```

simpleTransport.h

# Simple Transport Example



```
simpleTransport_TB.h    channelTransport.cpp    interfaceClasses.cpp    simpleTransport.cpp + X
Simple Transport    (Global Scope)

1 #include "simpleTransport.h"
2
3 void transmitter::transportingOut() {
4     int i;
5     sc_lv<8> dataToPut;
6     out->loading = false;
7     for (i=0; i<27; i++)
8     {
9         dataToPut = (sc_lv<8>) i;
10        out->loading = true;
11        out->transport(dataToPut);
12        out->INCOMPLETE.notify();
13        cout << "Data: (" << dataToPut << ") was transmitted at: "
14            << sc_time_stamp() << '\n';
15        wait(out->COMPLETE);
16    }
17 }
18
19 void receiver::transportingIn() {
20     sc_lv<8> dataThatGot;
21     int i; for (i=0; i<27; i++)
22     while (1)
23     {
24         wait(in->INCOMPLETE);
25         in->loading = false;
26         in->transport(dataThatGot);
27         in->COMPLETE.notify();
28         cout << "Data: (" << dataThatGot << ") was received at: "
29             << sc_time_stamp() << '\n';
30     }
31 }
```

- Example 3:  
Simple Transport:  
*simpleTransport*

# Simple Transport Example

## ○ Example 3: Simple Transport: Testbench



A screenshot of a software interface showing a code editor window. The window has tabs at the top: "channelTransport.h", "interfaceClasses.h", "simpleTransport.h", and "simpleTransport\_TB.h". The "simpleTransport\_TB.h" tab is active and highlighted with a yellow box. A yellow arrow points from the text "simpleTransport\_TB.h" in the title bar to the tab. The code in the editor is:

```
1 #include "simpleTransport.h"
2
3 SC_MODULE (simpleTransport_TB) {
4
5     transmitter* TRS1;
6     receiver* RCV1;
7
8     SC_CTOR(simpleTransport_TB) {
9         RCV1 = new receiver("receiver");
10        TRS1 = new transmitter("transmitter");
11        TRS1->out(*RCV1->insideTargetChannel);
12    }
13}
14
```

# Simple Transport Example

## Example 3: Simple Transport: Main



```
#include "simpleTransport_TB.h"
int sc_main(int argc, char *argv[]) {
    simpleTransport_TB SPG1("simpleTransport_TB");
    sc_start(900, SC_NS);
    return 0;
}
```

simpleTransport\_Main.cpp

# Simple Transport Example

## ○ Example 3: Simple Transport: Output

```
SystemC 2.3.2-Accellera --- Nov 16 2018 05:36:55
Copyright (c) 1996-2017 by all Contributors,
ALL RIGHTS RESERVED

Data: (00000000) was transmitted at: 10 ns
Data: (00000000) was received at: 25 ns
Data: (00000001) was transmitted at: 35 ns
Data: (00000001) was received at: 50 ns
Data: (00000010) was transmitted at: 60 ns
Data: (00000010) was received at: 75 ns
Data: (00000011) was transmitted at: 85 ns
Data: (00000011) was received at: 100 ns
Data: (00000100) was transmitted at: 110 ns
Data: (00000100) was received at: 125 ns
Data: (00000101) was transmitted at: 135 ns
Data: (00000101) was received at: 150 ns
Data: (00000110) was transmitted at: 160 ns
Data: (00000110) was received at: 175 ns
Data: (00000111) was transmitted at: 185 ns
Data: (00000111) was received at: 200 ns
Data: (00001000) was transmitted at: 210 ns
Data: (00001000) was received at: 225 ns
Data: (00001001) was transmitted at: 235 ns
Data: (00001001) was received at: 250 ns
Data: (00001010) was transmitted at: 260 ns
Data: (00001010) was received at: 275 ns
Data: (00001011) was transmitted at: 285 ns
Data: (00001011) was received at: 300 ns
Data: (00001100) was transmitted at: 310 ns
Data: (00001100) was received at: 325 ns
Data: (00001101) was transmitted at: 335 ns
```

# Transaction Level Modeling

Introduction

+ Processing Elements

+ Data

+ Memory

+ External Interfaces

– Abstract Communications

+ Coding Styles

- Loosely Timed
- Approximately Timed

+ TLM-2.0 Transport Interfaces

• Blocking Transport

- Path (Forward & Return)
- Socket (Simple Socket)

• Simple Transport Example

• Non-blocking Transport

- Path (Forward, Backward & Return)
- Socket (Tagged & Multi Socket)
- Phases & Base Protocol

+ Complete System

# Non-blocking

- Non-blocking transaction

- Includes timing

- Typically used

Called on **forward**

- Is appropriate when

interactions between **transaction** and target during the course of each transaction

```
enum tlm_sync_enum { TLM_ACCEPTED, TLM_UPDATED, TLM_COMPLETED };

template < typename TRANS= tlm_generic_payload, typename PHASE = tlm_phase>

class tlm_fw_nonblocking_transport_if : public virtual sc_core::sc_interface {
public:
    virtual tlm_sync_enum nb_transport(  

        TRANS& trans,  

        PHASE& phase,  

        sc_core::sc_time& t ) = 0;
};
```

# Non-blocking Transport: Path

- **Forward path**

- Is the calling path by which an initiator or interconnect component makes interface method calls forward in the direction of another interconnect component or the target

- **Backward path**

- Is the calling path by which a target or interconnect component makes interface method calls back in the direction of another interconnect component or the initiator

- **Return path**

- Is the return path by which an initiator, target or interconnect component returns immediately from the interface method that has been called



# Non-blocking Transport: Path

- # ○ Timing Diagram: Using The Backward Path

Transaction  
accepted now,  
caller asked to  
wait



# Non-blocking Transport: Path

- Timing Diagram:  
Using The Return Path

Callee annotates delay to next transition, caller waits



# Non-blocking Transport: Sockets

- Combine a port with an export
- Provide methods to bind port and export of both the forward and backward paths in a single call
- Group core interfaces for both the forward and backward paths together into a single object
- Offer strong type checking when binding sockets parameterized with incompatible protocol types
- The classes **tlm\_initiator\_socket** and **tlm\_target\_socket** are typically used directly by applications
  - Belong to the interoperability layer of the TLM-2.0 standard



# Non-blocking Transport: Sockets

## ○ Class Hierarchy



Standard sockets

Convenience sockets

# Non-blocking Transport: Convenience Sockets

- A family of derived socket classes provided in the utilities
  - Are derived from the classes *tlm\_initiator\_socket* and *tlm\_target\_socket*
- Additional functionalities
  - Register callbacks: Provides methods to register callbacks for incoming interface method calls
  - Tagged: Incoming interface method calls are tagged with an *id* to indicate the socket through which they arrived
  - Multi-ports: A single initiator socket can be bound to multiple target sockets and vice versa

# Non-blocking Transport: Convenience Sockets

## Simple Socket

- See Slides 52 and 53

# Non-blocking Transport: Convenience Sockets

## ○ Tagged Sockets

Incorporates a numerical *tag*

- To identify the socket in use
- To allow the callback to identify through which socket the incoming call arrived
  - Allows a single callback function to handle multiple sockets, with the tag identifying the socket which caused the callback to be invoked
  - Useful in the case where the same callback method is registered with multiple initiator sockets or multiple target sockets
- Is specified when the callback is registered
- Is an argument of the callback method

# Non-blocking Transport: Convenience Sockets

```

151 template <typename MODULE,
152     unsigned int BUSWIDTH = 32,
153     typename TYPES = tlm::tlm_base_protocol_types>
154 class simple_initiator_socket_tagged :
155     public tlm::tlm_initiator_socket<BUSWIDTH, TYPES>
156 {
157     public:
158         typedef typename TYPES::tlm_payload_type
159         typedef typename TYPES::tlm_phase_type
160         typedef tlm::tlm_sync_enum
161         typedef tlm::tlm_fw_transport_if<TYPES>
162         typedef tlm::tlm_bw_transport_if<TYPES>
163         typedef tlm::tlm_initiator_socket<BUSWIDTH, TYPES>
164
165     public:
166         simple_initiator_socket_tagged() { ... }
167         explicit simple_initiator_socket_tagged(const char* n) { ... }
168         void register_nb_transport_bw(MODULE* mod,
169             sync_enum_type (MODULE::*cb)(int,
170                                         transaction_type&,
171                                         phase_type&,
172                                         sc_core::sc_time&),
173             int id) { ... }
174
175         void register_invalidate_direct_mem_ptr(MODULE* mod,
176             void (MODULE::*cb)(int, sc_dt::uint64, sc_dt::uint64),
177             int id) { ... }
178
179     private:
180         class process { ... };
181     private:
182         process m_process;
183     };

```

Simple\_initiator\_socket.h:  
tagged socket class

- Tagged Socket: Class Definition

The tag that is used for identifying the socket of the incoming transaction

# Non-blocking Transport: Convenience Sockets

- Tagged Socket:



Distinguish origin of incoming transactions using socket id

# Non-blocking Transport: Convenience Sockets

## Multi Socket

- A variation on the tagged simple sockets
- Permit a single socket to be bound to multiple sockets on other components
- Use multi-port index number as the tag
  - Is able to identify from which socket on another component an incoming interface method call arrives

# Non-blocking Transport: Convenience Sockets

## ○ Multi Socket:

multi\_passthrough\_initiator\_socket      multi\_passthrough\_target\_socket



Many-to-Many socket binding  
Method calls tagged with multi-port index value

# Non-blocking Transport: Phase & Base Protocol

## ○ Phase

- Class **tlm\_phase** is the default phase type used by the non-blocking transport interface class templates and the base protocol
  - Base protocol phases:
    - **BEGIN\_REQ**, **END\_REQ**, **BEGIN\_RESP**, and **END\_RESP**
  - Extending the set of four phases provided by **tlm\_phase\_enum**:
    - Using the macro **DECLARE\_EXTENDED\_PHASE**
  - For maximal interoperability, an application should only use the four phases of **tlm\_phase\_enum**

# Non-blocking Transport: Phase & Base Protocol

## ○ Phases: Class Definition

tlm\_phase.h

The enumeration having values corresponding to the four phases of the base protocol: BEGIN\_REQ, END\_REQ, BEGIN\_RESP, and END\_RESP

```

25: namespace tlm {
26:
27: //enum tlm_phase { BEGIN_REQ, END_REQ, BEGIN_RESP, END_RESP };
28:
29: enum tlm_phase_enum { UNINITIALIZED_PHASE=0, BEGIN_REQ=1, END_REQ, BEGIN_RESP, END_RESP };
30:
31: inline unsigned int create_phase_number() { ... }
32: inline std::vector<const char*>& get_phase_name_vec() { ... }
33:
34: class tlm_phase{
35: public:
36:     tlm_phase(): m_id(0) {}
37:     tlm_phase(unsigned int id): m_id(id){}
38:     tlm_phase(const tlm_phase_enum& standard): m_id((unsigned int) standard){}
39:     tlm_phase& operator=(const tlm_phase_enum& standard){m_id=(unsigned int)standard; return *this;}
40:     operator unsigned int() const{return m_id;}
41:
42: private:
43:     unsigned int m_id;
44: };
45: ...
46: #define DECLARE_EXTENDED_PHASE(name_arg) \
47: ...
48:
49: } // namespace tlm
    
```

Is used to extend the four declared phases

# Non-blocking Transport: Phase & Base Protocol

## ○ Base Protocol

- The base protocol = tlm\_generic\_payload + tlm\_phase
- consists of a set of rules to ensure maximal interoperability between transaction level models of components

```
namespace tlm {  
    struct tlm_base_protocol_types  
    {  
        typedef tlm_generic_payload tlm_payload_type;  
        typedef tlm_phase tlm_phase_type;  
    };  
} // namespace tlm
```

For base protocol scenarios,  
there is not necessarily a unique  
hardware implementation,  
however, for every scenario,  
there is always some hardware  
correspondences

# Non-blocking Transport: Phase & Base Protocol

- The Initiator: Calls nb\_transport\_fw
  - The Target
    - Executes nb\_transport\_fw
    - Modifies the phase and put the transaction in the peq
    - Return TLM\_ACCEPTED to the initiator
  - The Initiator: Waits for receiving END\_REQ
  - In the Target:
    - Transaction is expired
    - Peq\_cb is executed: Calls nb\_transport\_bw
  - The Initiator:
    - Executes nb\_transport\_bw
    - Modify the phase and put the transaction in the peq
    - Return TLM\_ACCEPTED to the target
  - The Target:
    - Put the transaction with BEGIN\_RESP in the peq
    - Waits for expiration of the transaction
      - Peq\_cb is executed: Calls nb\_transport\_bw to send BEGIN\_RESP
  - The Initiator:
    - Executes nb\_transport\_bw
    - Modify the phase and put the transaction in the peq
    - Return TLM\_ACCEPTED to the target
    - Waits for expiration of the transaction
      - Peq\_cb is executed: Calls nb\_transport\_fw to send END\_REQ
  - The Target:
    - Executes nb\_transport\_fw
    - Return TLM ACCEPTED to the initiator



## **Nomenclature**

## Blue: Initiator

## Red: Target

## Call:

**nb transport(GP, phase, delay)**

## Return path from target to initiator:

## **tlm\_sync\_enum**

## Return path from initiator to target:

### tlm sync enum

## Phase Transition:

# Non-blocking Transport: Phase & Base Protocol



## Nomenclature

Blue: Initiator

Red: Target

Call:

`nb_transport(GP, phase, delay)`

Implement:

`nb_transport(GP, phase, delay){...}`

Return path from target to initiator:

`tlm_sync_enum`

Return path from initiator to target:

`tlm_sync_enum`

Phase Transition:

# Non-blocking Transport: Phase & Base Protocol

## ○ Abstract Communication All Scenarios



| TLM Comm. Scenario | TLM-2.0 base protocol flow: Edge id. |   |   |   |   |   |   |   |  |  |
|--------------------|--------------------------------------|---|---|---|---|---|---|---|--|--|
| SUDT2              | c                                    | h | j | o |   |   |   |   |  |  |
|                    | a                                    | f | g | i | j | o |   |   |  |  |
|                    | c                                    | h | l | n | q | s |   |   |  |  |
|                    | a                                    | f | g | i | l | n | q | s |  |  |
| SUDT3              | a                                    | e | j | o |   |   |   |   |  |  |
|                    | a                                    | e | l | n | q | s |   |   |  |  |
|                    | a                                    | e | l | n | q | s |   |   |  |  |
|                    | a                                    | e | l | n | p | r |   |   |  |  |
| SUDT4              | b                                    | m | q | s |   |   |   |   |  |  |
|                    | b                                    | m | p | r |   |   |   |   |  |  |
|                    | d                                    | r |   |   |   |   |   |   |  |  |
| SUDT5              | c                                    | h | k | r |   |   |   |   |  |  |
|                    | a                                    | f | g | i | k | r |   |   |  |  |
| SUDT7              | a                                    | e | k | r |   |   |   |   |  |  |
|                    | a                                    | e | l | n | q | s |   |   |  |  |
| SUDT8              | a                                    | e | k | r |   |   |   |   |  |  |
|                    | b                                    | m | q | s |   |   |   |   |  |  |
|                    | b                                    | m | p | r |   |   |   |   |  |  |
| SUDT9              | c                                    | h | l | n | p | r |   |   |  |  |
|                    | a                                    | f | g | i | l | n | p | r |  |  |
| SUDT10             |                                      |   |   |   |   |   |   |   |  |  |

# Non-blocking Transport: Phase & Base Protocol

## ○ Abstract Communication SUDT2 Scenario



**SUDT2**

| TLM Comm. Scenario | TLM-2.0 base protocol flow: Edge id. |   |   |   |   |   |   |   |
|--------------------|--------------------------------------|---|---|---|---|---|---|---|
| SUDT2              | c                                    | h | j | o |   |   |   |   |
|                    | a                                    | f | g | i | j | o |   |   |
|                    | c                                    | h | l | n | q | s |   |   |
|                    | a                                    | f | g | i | l | n | q | s |
| SUDT3              | a                                    | e | j | o |   |   |   |   |
|                    | a                                    | e | l | n | q | s |   |   |
|                    | a                                    | e | l | n | q | s |   |   |
|                    | a                                    | e | l | n | p | r |   |   |
| SUDT4              | b                                    | m | q | s |   |   |   |   |
|                    | b                                    | m | p | r |   |   |   |   |
|                    | d                                    | r |   |   |   |   |   |   |
|                    | c                                    | h | k | r |   |   |   |   |
| SUDT5              | a                                    | f | g | i | k | r |   |   |
|                    | a                                    | e | k | r |   |   |   |   |
|                    | a                                    | e | l | n | q | s |   |   |
| SUDT7              | a                                    | e | k | r |   |   |   |   |
|                    | a                                    | e | l | n | q | s |   |   |
|                    | a                                    | e | k | r |   |   |   |   |
| SUDT8              | b                                    | m | q | s |   |   |   |   |
|                    | b                                    | m | p | r |   |   |   |   |
|                    | c                                    | h | l | n | p | r |   |   |
|                    | a                                    | f | g | i | l | n | p | r |
| SUDT9              | a                                    | e | k | r |   |   |   |   |
|                    | b                                    | m | q | s |   |   |   |   |
|                    | b                                    | m | p | r |   |   |   |   |
| SUDT10             | c                                    | h | l | n | p | r |   |   |
|                    | a                                    | f | g | i | l | n | p | r |

# Non-blocking Transport: Phase & Base Protocol

## ○ Base Protocol Timing Parameters and Flow Control

BEGIN\_REQ must wait  
for previous  
END\_REQ,  
BEGIN\_RESP for  
END\_RESP



# Transaction Level Modeling

## Introduction

### + Processing Elements

### + Data

### - Abstract Communications

#### + Coding Styles

#### + TLM-2.0 Transport Interfaces

- Blocking Transport
- Simple Transport Example
- Non-blocking Transport
  - Path (Forward, Backward & Return)
  - Socket (Tagged & Multi Socket)
  - Phases & Base Protocol
  - nbFwdMemoryRW Example

### + Complete System



# Non-blocking Transport: Forward

- Example 4: nbFwdMemoryRW: Base Protocol Flow Control



## Nomenclature

Blue: Initiator  
Red: Target

Call:

`nb_transport(GP, phase, delay)`

Implement:

`nb_transport(GP, phase, delay){...}`

Return path from target to initiator:

`tlm_sync_enum`

Return path from initiator to target:

`tlm_sync_enum`

Phase Transition:

# Non-blocking Transport: Forward

## Example 4: nbFwdMemoryRW: Initiator, *memWriterReader*

The screenshot shows a code editor with two files open:

- nbFwdMemoryRW.cpp**: The implementation file containing the definition of the *memWriterReader* module.
- memWriterReader.h**: The header file for the *memWriterReader* module.

The code in **nbFwdMemoryRW.cpp** is as follows:

```
1 #include <systemc.h>
2
3 #include "tlm.h"
4 #include "tlm_utils/simple_initiator_socket.h"
5 #include "tlm_utils/simple_target_socket.h"
6
7 class memWriterReader : public sc_module {
8 public:
9     tlm_utils::simple_initiator_socket<memWriterReader, 32> memWRSocket;
10
11 SC_CTOR(memWriterReader) : memWRSocket("mem_WR_socket"), nBlockWriteRead(0)
12 {
13     nBlockWriteRead = new tlm::tlm_generic_payload();
14     for (int i = 0; i < 4; i++) *(data+i) = i+192;
15     SC_THREAD(nbMemWR);
16 }
17
18 tlm::tlm_generic_payload* nBlockWriteRead;
19 void nbMemWR();
20 void doSomethingGood(tlm::tlm_generic_payload&);
21
22 sc_lv<8> data[5];
23};
```

Annotations explain specific parts of the code:

- A yellow callout points to the line `tlm_utils::simple_initiator_socket<memWriterReader, 32> memWRSocket;` with the text: **Defining the Initiator socket and specializing it for memWriterReader**.
- A green callout points to the line `for (int i = 0; i < 4; i++) *(data+i) = i+192;` with the text: **Initializing memory**.

# Non-blocking Transport: Forward

- Example 4: nbFwdMemoryRW: Initiator, memWriterReader

```
memWriterReader_TB.h      memWriterReader.h      memoryUnit.h      memWriterReader.cpp + X
nbFwdMemoryRW          (Global Scope)
1 #include "memWriterReader.h"
2
3 void memWriterReader::nbMemWR()
4 {
5     tlm::tlm_phase forwardPhase;
6     sc_time processTime; // Processing time of initiator prior to call
7
8     processTime = sc_time(0, SC_PS);
9     for (int i = 0; i < 111; i = i + 11){
10         tlm::tlm_command cmd = (tlm::tlm_command)(rand() % 2);
11         if (cmd == tlm::TLM_WRITE_COMMAND) {
12             data[0] = (sc_lv<8>) (i+5);
13             data[1] = (sc_lv<8>) (i+6);
14             data[2] = (sc_lv<8>) (i+7);
15             data[3] = (sc_lv<8>) (i+8);
16             data[4] = (sc_lv<8>) (i+9);
17         }
18
19         nBlockWriteRead->set_command( cmd );
20         nBlockWriteRead->set_address( i );
21         nBlockWriteRead->set_data_ptr( (unsigned char*) data );
22         nBlockWriteRead->set_data_length( 5 );
23         nBlockWriteRead->set_streaming_width( 5 );
24         nBlockWriteRead->set_byte_enable_ptr( 0 );
25         nBlockWriteRead->set_dmi_allowed( false );
26         nBlockWriteRead->set_response_status( tlm::TLM_INCOMPLETE_RESPONSE )
27
28         forwardPhase = tlm::BEGIN_REQ;
29
30         cout << (cmd ? 'W' : 'R') << ", @" << i << " data:";
31         sc_lv<8> vv;
32         for(int j=0; j<5; j++) {vv=data[j]; cout << vv << " ;"}
33         cout << " @time " << sc_time_stamp() << " delay=" << processTime
34
35         tlm::tlm_sync_enum returnStatus;
36         returnStatus = memWRSocket->
37             nb_transport_fw(*nBlockWriteRead, forwardPhase, processTime);
38
39         if (returnStatus == tlm::TLM_COMPLETED)
40             doSomethingGood( *nBlockWriteRead, processTime );
41
42     }
43 }
```

memWriterReader.cpp

Set GP parameters

Setting the forward phase for starting a new transaction

Defining the *returnStatus* variable

Calling the *nb\_transport\_fw* method

The initiator can process the data of the incoming transaction in the case of receiving TLM\_COMPLETED

# Non-blocking Transport: Forward

- Example 4: nbFwdMemoryRW: Initiator, *memWriterReader*, Cont.

The data is ready,  
an appropriate  
processing can be  
started on the data

memWriterReader\_TB.h

memWriterReader.h

memoryUnit.h

memWriterReader.cpp

43

```
void memWriterReader::doSomethingGood( tlm::tlm_generic_payload& completeTrans,
                                         sc_time totalTime )
{
    if ( completeTrans.is_response_error() )
        SC_REPORT_ERROR("TLM-2", "error...\n");

    tlm::tlm_command cmd = completeTrans.get_command();
    uint64 adr = completeTrans.get_address();
    int* ptr = reinterpret_cast<int*>( completeTrans.get_data_ptr() );

    cout << "Above was completed @time " << totalTime << '\n';
}
```

memWriterReader.cpp

Initiator obliged to  
check response status

# Non-blocking Transport: Forward

- Example 4: nbFwdMemoryRW: Target, *memoryUnit*

**memoryUnit.h**

```

memWriterReader.cpp      memWriterReader_TB.h      memWriterReader.h      memoryUnit.h
nbFwdMemoryRW          (Global Scope)           memoryUnit.h
1 #include <systemc.h>
2
3 #include "tlm.h"
4 #include "tlm_utils/simple_initiator_socket.h"
5 #include "tlm_utils/simple_target_socket.h"
6
7 class memoryUnit : public sc_module {
8 public:
9     tlm_utils::simple_target_socket<memoryUnit, 32> memSocket;
10
11     static const int SIZE=256;
12
13     SC_CTOR(memoryUnit) : memSocket("memory_side_socket") {
14         memSocket.register_nb_transport_fw(this, &memoryUnit::nb_transport_fw);
15
16         // Initialize memory
17         for (int i = 0; i < SIZE; i++)
18             memArray[i] = (sc_lv<8>) (i%256 + 192);
19     }
20
21     virtual tlm::tlm_sync_enum nb_transport_fw( tlm::tlm_generic_payload&,
22                                               tlm::tlm_phase&, sc_time& );
23
24     sc_lv<8> memArray[SIZE];
25 };

```

Defining the target socket and specializing it for *memoryUnit*

Register callback for incoming *nb\_transport\_fw* interface method call

Initializing memory

# Non-blocking Transport: Forward

memWriterReader.h      memoryUnit.h      memWriterReader.cpp      memoryUnit.cpp

(Global Scope)

```

1 #include "memoryUnit.h"
2
3 tlm::tlm_sync_enum memoryUnit::nb_transport_fw(
4     tlm::tlm_generic_payload& receivedTrans,
5     tlm::tlm_phase& phase, sc_time& delay ){
6
7     tlm::tlm_command cmd = receivedTrans.get_command();
8     uint64_t          adr = receivedTrans.get_address();
9     unsigned char*    ptr = receivedTrans.get_data_ptr();
10    unsigned int      len = receivedTrans.get_data_length();
11    unsigned char*    byt = receivedTrans.get_byte_enable_ptr();
12    unsigned int      wid = receivedTrans.get_streaming_width();
13
14    if (byt != 0) {
15        receivedTrans.set_response_status( tlm::TLM_BYTE_ENABLE_ERROR_RESPONSE );
16        return tlm::TLM_COMPLETED;
17    }
18    if (len > 5 || wid < len) {
19        receivedTrans.set_response_status( tlm::TLM_BURST_ERROR_RESPONSE );
20        return tlm::TLM_COMPLETED;
21    }
22    unsigned int i;
23    if ( cmd == tlm::TLM_READ_COMMAND ) {
24        for(i=0; i<len; i=i+1) {
25            *(ptr+i) = *((unsigned char*) (memArray+adr+i));
26        }
27    } else if ( cmd == tlm::TLM_WRITE_COMMAND ) {
28        for(i=0; i<len; i=i+1) {
29            *((unsigned char*) (memArray+adr+i)) = *(ptr+i);
30        }
31
32    receivedTrans.set_response_status( tlm::TLM_OK_RESPONSE );
33    delay = delay + sc_time(123, SC_NS);
34    return tlm::TLM_COMPLETED;
35 }
```

**memoryUnit.cpp**

- Example 4: nbFwdMemoryRW: Target, *memoryUnit*

The *nb\_transport* methods shall not call wait, directly or indirectly

Obliged to implement read and write commands

Obliged to set response status to indicate successful completion

Should return TLM\_COMPLETED according to the protocol

# Non-blocking Transport: Forward

- Example 4: nbFwdMemoryRW: Testbench

```

memWriterReader.cpp      memWriterReader_TB.h  memWriterReader.h  memoryUnit.h
nbFwdMemoryRW          (Global Scope)        memoryUnit.h
1 #include "memWriterReader.h"
2 #include "memoryUnit.h"
3
4 SC_MODULE(memWriterReader_TB)
5 {
6     memWriterReader *WR1;
7     memoryUnit    *MU1;
8
9     SC_CTOR(memWriterReader_TB)
10 {
11     WR1 = new memWriterReader("WR");
12     MU1 = new memoryUnit("memory");
13     WR1->memWSocket.bind(MU1->memSocket);
14 }
15

```

memWriterReader\_TB.h

Instantiate components

- One initiator is bound directly to one target with no intervening bus
- Bind initiator socket to target socket

# Non-blocking Transport: Forward

- Example 4: nbFwdMemoryRW: Main



```
#include "memWriterReader_TB.h"

int sc_main(int argc, char* argv[])
{
    memWriterReader_TB TB1("memWriterReader_TB");
    sc_start();
    return 0;
}
```

memWriterReader\_Main.cpp

# Non-blocking Transport: Forward

- Example 4: nbFwdMemoryRW: Output

```
SystemC 2.3.2-Accellera --- Nov 18 2018 08:59:55
Copyright (c) 1996-2017 by all Contributors,
ALL RIGHTS RESERVED
W, @0 data:00000101 00000110 00000111 00001000 00001001 @time 0 s delay=0 s
Above was completed @time 123 ns
W, @11 data:00010000 00010001 00010010 00010011 00010100 @time 0 s delay=123 ns
Above was completed @time 246 ns
R, @22 data:00010000 00010001 00010010 00010011 00010100 @time 0 s delay=246 ns
Above was completed @time 369 ns
R, @33 data:00010000 00010001 00010010 00010011 00010100 @time 0 s delay=369 ns
Above was completed @time 492 ns
W, @44 data:00110001 00110010 00110011 00110100 00110101 @time 0 s delay=492 ns
Above was completed @time 615 ns
R, @55 data:00110001 00110010 00110011 00110100 00110101 @time 0 s delay=615 ns
Above was completed @time 738 ns
R, @66 data:00110001 00110010 00110011 00110100 00110101 @time 0 s delay=738 ns
Above was completed @time 861 ns
R, @77 data:00110001 00110010 00110011 00110100 00110101 @time 0 s delay=861 ns
Above was completed @time 984 ns
R, @88 data:00110001 00110010 00110011 00110100 00110101 @time 0 s delay=984 ns
Above was completed @time 1107 ns
R, @99 data:00110001 00110010 00110011 00110100 00110101 @time 0 s delay=1107 ns
Above was completed @time 1230 ns
W, @110 data:01110011 01110100 01110101 01110110 01110111 @time 0 s delay=1230 ns
Above was completed @time 1353 ns
```

# Transaction Level Modeling

## Introduction

### + Processing Elements

### + Data

### - Abstract Communications

#### + Coding Styles

#### + TLM-2.0 Transport Interfaces

- Blocking Transport
- Simple Transport Example
- Non-blocking Transport
  - Path (Forward, Backward & Return)
  - Socket (Tagged & Multi Socket)
  - Phases & Base Protocol
  - DMIImemoryRW Example

### + Complete System



# DMI Transport

- Direct Memory Interface (DMI) provides a direct access to an area of memory for an initiator
- The DMI offers a large increase in simulation speed
- DMI uses both **forward** and **backward** interface
  - The **forward** DMI interface is used to request a DMI access (e.g., read, write or both) to a given address. `get_direct_mem_ptr` interface method is called on the forward path from initiator to target
  - The **backward** DMI interface is used to invalidate DMI pointers previously established using the forward path. `invalidate_direct_mem_ptr` is called on the backward path from target to initiator
- The DMI interfaces use DMI transaction object, `tlm_dmi`
- Only makes sense with loosely-timed models

## TLM-2.0 Transport Interfaces

# DMI Transaction Object

```
enum dmi_access_e
{
    DMI_ACCESS_NONE      = 0x00,
    DMI_ACCESS_READ      = 0x01,
    DMI_ACCESS_WRITE     = 0x02,
    DMI_ACCESS_READ_WRITE = DMI_ACCESS_READ | DMI_ACCESS_WRITE
};

class tlm_dmi {
public:
    enum dmi_access_e;
    tlm_dmi(void);
    void init(void);
private:
    unsigned char* m_dmi_ptr;
    sc_dt::uint64 m_dmi_start_address;
    sc_dt::uint64 m_dmi_end_address;
    dmi_access_e m_dmi_access;
    sc_core::sc_time m_dmi_read_latency;
    sc_core::sc_time m_dmi_write_latency;
};
```

Granted access

Define the latency of read/write transactions

```
class tlm_dmi {
public:
    enum dmi_access_e;
    tlm_dmi(void);
    void init(void);
    unsigned char* get_dmi_ptr();
    sc_dt::uint64 get_start_address();
    sc_dt::uint64 get_end_address();
    sc_core::sc_time get_read_latency();
    sc_core::sc_time get_write_latency();
    dmi_access_e get_granted_access();
    bool is_none_allowed();
    bool is_read_allowed();
    bool is_write_allowed();
    bool is_read_write_allowed();
    void set_dmi_ptr(unsigned char* p);
    void set_start_address(sc_dt::uint64 addr);
    void set_end_address(sc_dt::uint64 addr);
    void set_read_latency(sc_core::sc_time t);
    void set_write_latency(sc_core::sc_time t);
    void set_granted_access(dmi_access_e a);
    void allow_none();
    void allow_read();
    void allow_write();
    void allow_read_write();
private:
    unsigned char* m_dmi_ptr;
    sc_dt::uint64 m_dmi_start_address;
    sc_dt::uint64 m_dmi_end_address;
    dmi_access_e m_dmi_access;
    sc_core::sc_time m_dmi_read_latency;
    sc_core::sc_time m_dmi_write_latency;
};
```

Different access types for indicating the access granted to the initiator

Invalidate all attributes

The attributes are used by target

The attributes are used by initiator

If the forward call is successful, the target returns the *dmi\_ptr*, which must point to the data element corresponding to the *dmi\_start\_address*

The absolute start and end addresses of the DMI region

# DMI Transport



# DMI Transport

- Example 5 scenario:

- The target grants both read and write accesses
- The entire memory contents have been granted as the DMI region
- The *b\_transport* method calls *set\_dmi\_allowed* to set the DMI hint. Because there is no point in the initiator making repeated calls to *get\_direct\_mem\_ptr* if it can be told in advance that such calls are going to fail
- The initiator requests a DMI pointer from the target, knowing the request is likely to succeed
- For implementing *invalidate\_direct\_mem\_ptr*, the initiator ignores the bounds of the direct memory region, and simply invalidates the DMI pointer whatever

# DMI Transport

- Example 5: DMImemoryRW: Initiator, *memWriterReader*

**memWriterReader.h**

```

(Global Scope)
1 #include <systemc.h>
2 #include "tlm.h"
3 #include "tlm_utils/simple_initiator_socket.h"
4
5 class memWriterReader : public sc_module {
6 public:
7     tlm_utils::simple_initiator_socket<memWriterReader> memWRSocket;
8
9     SC_CTOR(memWriterReader) : memWRSocket("mem_WR_socket"), DMIptrValid(false)
10 {
11     memWRSocket.register_invalidate_direct_mem_ptr(this, &memWriterReader::invalidate_direct_mem_ptr);
12     WriteRead = new tlm::tlm_generic_payload;
13 }
14
15 SC_THREAD(DMImemWR);
16
17 void DMImemWR();
18
19 virtual void invalidate_direct_mem_ptr(sc_dt::uint64, sc_dt::uint64);
20
21 tlm::tlm_generic_payload* WriteRead;
22 int data;
23 bool DMIptrValid;
24 tlm::tlm_dmi DMIdata;
25
26 };

```

Defining a transaction with Generic Payload type

Defining a DMI transaction object

Implement the functionality of the initiator, i.e., generating transactions and sending them

Register callback for incoming *invalidate\_direct\_mem\_ptr* interface method call

Defining the initiator socket and specializing it for *memWriterReader*

defaults to 32-bits wide

# DMI Transport

memWriterReader.h    memWriterReader.cpp    memoryUnit.h

```

memWriterReader.cpp
(Global Scope)
1 #include "memWriterReader.h"
2
3 void memWriterReader::DMIMemWR()
4 {
5     sc_time processTime = sc_time(10, SC_NS);
6     for (int i = 0; i < 128; i += 4) {
7         tlm::tlm_command cmd = (tlm::tlm_command)(rand() % 2);
8         if (cmd == tlm::TLM_WRITE_COMMAND) data = 0xFF000000 | i;
9
10        if (DMIptrValid) {
11            if (cmd == tlm::TLM_READ_COMMAND) {
12                assert(DMIdata.is_read_allowed());
13                memcpy(&data, DMIdata.get_dmi_ptr() + i, 4);
14                wait(DMIdata.get_read_latency());
15            }
16            else if (cmd == tlm::TLM_WRITE_COMMAND) {
17                assert(DMIdata.is_write_allowed());
18                memcpy(DMIdata.get_dmi_ptr() + i, &data, 4);
19                wait(DMIdata.get_write_latency());
20            }
21
22            cout << "DMI   = { " << (cmd ? 'W' : 'R') << ", " << hex << i
23              << " } , data = " << hex << data << " at time " << sc_time_stamp() << endl;
24        }
25
26        else {
27            WriteRead->set_command(cmd);
28            WriteRead->set_address(i);
29            WriteRead->set_data_ptr((unsigned char*)&data);
30            WriteRead->set_data_length(4);
31            WriteRead->set_streaming_width(4);
32            WriteRead->set_byte_enable_ptr(0);
33            WriteRead->set_dmi_allowed(false);
34            WriteRead->set_response_status(tlm::TLM_INCOMPLETE_RESPONSE);
35        }
36    }
37
38    cout << "trans = { " << (cmd ? 'W' : 'R') << ", " << hex << i
39      << " } , data = " << hex << data << " at time " << sc_time_stamp()
40      << " delay = " << processTime << endl;
41
42    DMIptrValid = false;
43}

```

**memWriterReader.cpp**

Generate a random sequence of reads and writes

Bypass transport interface and use DMI if it is available

If DMI is not available, set GP parameters

Reused across calls to b\_transport and DMI

○ Example 5: DMImemoryRW: Initiator, *memWriterReader*

memWriterReader.h    memWriterReader.cpp    memoryUnit.h    memoryUnit.cpp    memWriterReader.cpp

```

memWriterReader.cpp
memWRSocket->b_transport(*WriteRead, processTime);

if (WriteRead->is_response_error()) {
    char txt[100];
    sprintf(txt, "Error from b_transport, response status = %s",
           WriteRead->get_response_string().c_str());
    SC_REPORT_ERROR("TLM-2", txt);
}

if (WriteRead->is_dmi_allowed()) {
    DMIdata.init();
    DMIptrValid = memWRSocket->get_direct_mem_ptr(*WriteRead, DMIdata);
}

cout << "trans = { " << (cmd ? 'W' : 'R') << ", " << hex << i
      << " } , data = " << hex << data << " at time " << sc_time_stamp()
      << " delay = " << processTime << endl;

void memWriterReader::invalidate_direct_mem_ptr(sc_dt::uint64 start_range,
                                                sc_dt::uint64 end_range) {
    DMIptrValid = false;
}

```

Calling the *b\_transport* method

Initiator obliged to check response status

Check DMI hint

Implement backward DMI method

# DMI Transport

- Example 5: DMImemoryRW: Target, *memoryUnit*

```

memWriterReader.h      memWriterReader.cpp      memoryUnit.h      memoryUnit.cpp      memWriterReader.h
(Global Scope)
1 #include <systemc.h>
2 #include "tlm.h"
3 #include "tlm_utils/simple_target_socket.h"
4
5 class memoryUnit : public sc_module {
6 public:
7     tlm_utils::simple_target_socket<memoryUnit> memSocket;
8
9     static const int SIZE = 256;
10    int memArray[SIZE];
11    const sc_time Latency = sc_time(10, SC_NS);
12
13    SC_CTOR(memoryUnit) : memSocket("memory_side_socket"){
14        memSocket.register_b_transport(this, &memoryUnit::b_transport);
15        memSocket.register_get_direct_mem_ptr(this, &memoryUnit::get_direct_mem_ptr);
16
17        for (int i = 0; i < SIZE; i++)
18            memArray[i] = 0xAA000000 | (rand() % 256);
19
20        SC_THREAD(DMImem);
21    }
22
23    virtual void b_transport(tlm::tlm_generic_payload&, sc_time&);
24
25    virtual bool get_direct_mem_ptr(tlm::tlm_generic_payload&, tlm::tlm_dmi&);
26    void DMImem();
27
28};

```

**memoryUnit.h**

Defining the target socket and specializing it for *memoryUnit*

Register callback for incoming *b\_transport* interface method call

Register callback for incoming *get\_direct\_mem\_ptr* interface method call

Initializing memory

Calling *invalidate\_direct\_mem\_ptr*

# DMI Transport

memoryUnit.h      memoryUnit.cpp      memoryUnit.h      memoryUnit.cpp      memWriterReader.h

→ memoryUnit

```

1 #include "memoryUnit.h"
2
3
4 void memoryUnit::b_transport(tlm::tlm_generic_payload& gotThis, sc_time& delayValue)
5 {
6     tlm::tlm_command cmd = gotThis.get_command();
7     uint64 adr = gotThis.get_address()/4;
8     unsigned char* ptr = gotThis.get_data_ptr();
9     unsigned int len = gotThis.get_data_length();
10    unsigned char* byt = gotThis.get_byte_enable_ptr();
11    unsigned int wid = gotThis.get_streaming_width();
12
13    if (adr >= uint64(SIZE) || byt != 0 || len > 4 || wid < len)
14        SC_REPORT_ERROR("TLM-2.0: ", "Inconsistent Generic Payload");
15
16    unsigned int i;
17    if (cmd == tlm::TLM_READ_COMMAND){
18        memcpy(ptr, &memArray[adr], len);
19    }
20    else if (cmd == tlm::TLM_WRITE_COMMAND){
21        memcpy(&memArray[adr], ptr, len);
22    }
23    wait(delayValue);
24    gotThis.set_dmi_allowed(true);
25    gotThis.set_response_status(tlm::TLM_OK_RESPONSE);
26 }

```

## memoryUnit.cpp

### Example 5: DMI<sub>MemoryRW</sub>: Target, *memoryUnit*

Getting GP parameters

Obliged to check address range  
and check for unsupported features

Obliged to implement  
read and write commands

Set DMI hint to indicated that DMI is supported

Obliged to set response status to  
indicate successful completion

# DMI Transport

- Example 5: DMImemoryRW: Target, *memoryUnit* (Cont.)

```

memWriterReader.h      memWriterReader.cpp      memoryUnit.h      memoryUnit.cpp      memWriterReader_TB.h
(Global Scope)
28     bool memoryUnit::get_direct_mem_ptr(tlm::tlm_generic_payload& gotThis, tlm::tlm_dmi& dmi_data)
{
    dmi_data.allow_read_write();
    dmi_data.set_dmi_ptr((unsigned char*)&memArray[0]);
    dmi_data.set_start_address( 0 );
    dmi_data.set_end_address( SIZE*4-1 );
    dmi_data.set_read_latency(Latency);
    dmi_data.set_write_latency(Latency);
    return true;
}

void memoryUnit::DMIImem()
{
    for (int i = 0; i < 4; i++) {
        wait(Latency * 8);
        memSocket->invalidate_direct_mem_ptr(0, SIZE - 1);
    }
}

```

Permit read and write access

Set other attributes of the DMI object to describe the details of the access

Invalidate DMI pointers periodically

The *dmi\_ptr* is the actual direct memory pointer. The target is free to grant any DMI region that encloses the requested address. *start\_address* and *end\_address* describe the bounds of the DMI region from the point of view of the target

# DMI Transport

## Example 5: DMImemoryRW: Testbench

```
memWriterReader.cpp      memWriterReader_TB.h  X  memWriterReader.h  memoryUnit.h
nbFwdMemoryRW          (Global Scope)          

1 #include "memWriterReader.h"
2 #include "memoryUnit.h"
3
4 SC_MODULE(memWriterReader_TB)
5 {
6     memWriterReader *WR1;
7     memoryUnit    *MU1;
8
9     SC_CTOR(memWriterReader_TB)
10 {
11     WR1 = new memWriterReader("WR");
12     MU1 = new memoryUnit("memory");
13     WR1->memWSocket.bind(MU1->memSocket);
14 }
15 }
```

Instantiate components

Bind initiator socket to target socket

# DMI Transport

## Example 5: DMImemoryRW: Main

```
memWriterReader_Main.cpp ✘ X memWriterReader_TB.cpp memWriterReader.h  
(Global Scope)  
1 #include "memWriterReader_TB.h"  
2  
3 int sc_main(int argc, char* argv[]){  
4     memWriterReader_TB TB1("memWriterReader_TB");  
5     sc_start();  
6     return 0;  
7 }  
8 }
```

memWriterReader\_Main.cpp

# DMI Transport

- Example 5:
- DMImemoryRW: Output

The transaction is done by *b\_transport*

After the completion of a transaction,  
DMI access is provided for 80 NS

DMI access is invalidated after 80 NS. So,  
The transaction is done by *b\_transport*

32 (128/4) transactions  
are generated and sent

```
C:\Windows\system32\cmd.exe
SystemC 2.3.1-Accellera --- Oct 21 2019 11:59:41
Copyright <c> 1996-2014 by all Contributors,
ALL RIGHTS RESERVED

trans = < W, 0 >, data = ff000000 at time 10 ns delay = 10 ns
DMI = < W, 4 >, data = ff000004 at time 20 ns
DMI = < R, 8 >, data = aa0000be at time 30 ns
DMI = < R, c >, data = aa000084 at time 40 ns
DMI = < W, 10 >, data = ff000010 at time 50 ns
DMI = < R, 14 >, data = aa00006c at time 60 ns
DMI = < R, 18 >, data = aa0000d6 at time 70 ns
DMI = < R, 1c >, data = aa0000ae at time 80 ns
trans = < R, 20 >, data = aa000052 at time 90 ns delay = 10 ns
DMI = < R, 24 >, data = aa000090 at time 100 ns
DMI = < W, 28 >, data = ff000028 at time 110 ns
DMI = < W, 2c >, data = ff00002c at time 120 ns
DMI = < W, 30 >, data = ff000030 at time 130 ns
DMI = < W, 34 >, data = ff000034 at time 140 ns
DMI = < W, 38 >, data = ff000038 at time 150 ns
DMI = < W, 3c >, data = ff00003c at time 160 ns
trans = < W, 40 >, data = ff000040 at time 170 ns delay = 10 ns
DMI = < R, 44 >, data = aa0000a6 at time 180 ns
DMI = < W, 48 >, data = ff000048 at time 190 ns
DMI = < R, 4c >, data = aa00003c at time 200 ns
DMI = < W, 50 >, data = ff000050 at time 210 ns
DMI = < R, 54 >, data = aa00000c at time 220 ns
DMI = < R, 58 >, data = aa00003e at time 230 ns
DMI = < W, 5c >, data = ff00005c at time 240 ns
trans = < R, 60 >, data = aa000024 at time 250 ns delay = 10 ns
DMI = < R, 64 >, data = aa00005e at time 260 ns
DMI = < W, 68 >, data = ff000068 at time 270 ns
DMI = < R, 6c >, data = aa00001c at time 280 ns
DMI = < R, 70 >, data = aa000006 at time 290 ns
DMI = < W, 74 >, data = ff000074 at time 300 ns
DMI = < W, 78 >, data = ff000078 at time 310 ns
DMI = < R, 7c >, data = aa0000de at time 320 ns

Press any key to continue . . .
```

# Transaction Level Modeling

## Introduction

### + Processing Elements

### + Data

### - Abstract Communications

#### + Coding Styles

#### + TLM-2.0 Transport Interfaces

- Blocking Transport
- Simple Transport Example
- Non-blocking Transport
  - Path (Forward, Backward & Return)
  - Socket (Tagged & Multi Socket)
  - Phases & Base Protocol
  - nbFwdBwdMemoryRW Example

### + Complete System

#### The concepts covered:

- ✓ Generic payload
- ✓ Simple sockets
- ✓ Non-blocking transport
  - ✓ Forward path
  - ✓ Backward path
  - ✓ 4 calls
- ✓ Payload Event Queue



# Payload Event Queue (PEQ)

---

- Is a class that maintains a queue of SystemC event notifications, where each notification carries an associated transaction object
- Each transaction is written into the PEQ annotated with a delay, and each transaction emerges from the back of the PEQ at a time calculated from the current simulation time plus the annotated delay
- Will schedule the timing point associated with the `nb_transport` call to occur at the correct simulation time
- There are two ways to implement PEQ
  - `peq_with_get`
  - `peq_with_cb_and_phase`

# Non-blocking Transport: Forward & Backward



# Non-blocking Transport: Forward & Backward

**memWriterReader**  
SC\_MODULE

nbMemWR()

nb\_transport\_bw()

memWR\_peq()

doSomethingGood()

Notation: *return, phase, delay*

-, BEGIN\_REQ, 0

TLM\_ACCEPTED, -, -

-, END\_REQ, 20

TLM\_ACCEPTED, -, -

-, BEGIN\_RESP, 0

TLM\_ACCEPTED, -, -

-, END\_RESP, 10

TLM\_ACCEPTED, -, -

**memoryUnit**  
SC\_MODULE

nb\_transport\_fw()

send\_ENDREQ()

send\_BEGINRESP()

mem\_peq()

# Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: Base Protocol Flow Control



## Nomenclature

Blue: Initiator  
Red: Target

Call:

`nb_transport(GP, phase, delay)`

Implement:

`nb_transport(GP, phase, delay){...}`

Return path from target to initiator:

`tlm_sync_enum`

Return path from initiator to target:

`tlm_sync_enum`

Phase Transition:

# Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: 4 calls



# Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: Initiator, *memWriterReader*

**memWriterReader.h**

```

memWriterReader.cpp      memWriterReader.h  X memoryUnit.cpp    memory
(Global Scope)
1 #include <systemc.h>
2 #include "tlm.h"
3 #include "tlm_utils/simple_initiator_socket.h"
4 #include "tlm_utils/peq_with_cb_and_phase.h"
5
6 class memWriterReader : public sc_module {
7 public:
8     tlm_utils::simple_initiator_socket<memWriterReader, 32> memWRSocket;
9
10 SC_CTOR(memWriterReader) : memWRSocket("mem WR socket"), nBlockWriteRead(0),
11                           m_peq(this, &memWriterReader::memWR_peq)
12 {
13     nBlockWriteRead = new tlm::tlm_generic_payload();
14     SC_THREAD(nbMemWR);
15     memWRSocket.register_nb_transport_bw(this, &memWriterReader::nb_transport_bw);
16 }
17 tlm::tlm_generic_payload* nBlockWriteRead;
18 void nbMemWR();
19 void doSomethingGood(tlm::tlm_generic_payload& );
20 tlm_utils::peq_with_cb_and_phase<memWriterReader> m_peq;
21 void memWR_peq(tlm::tlm_generic_payload&, const tlm::tlm_phase& );
22 virtual tlm::tlm_sync_enum nb_transport_bw(tlm::tlm_generic_payload&, tlm::tlm_phase&, sc_time&);
23
24 int data[5];
25 };

```

memWriterReader.h

Register callback for incoming *nb\_transport\_bw* interface method call

Non-blocking transport, so the initiator can process the data of the incoming transaction

Declared a Payload Event Queue, then implement it

**memWriterReader  
SC\_MODULE**

**nbMemWR()**

**nb\_transport\_bw()**

**memWR\_peq()**

**doSomethingGood()**

# Non-blocking Transport: Forward & Backward

memWriterReader.cpp memWriterReader.h memoryUnit.cpp memory

memWriterReader

```

1 #include "memWriterReader.h"
2
3 void memWriterReader::nbMemWR(){
4     tlm::tlm_phase forwardPhase;
5     sc_time delay;
6     for (int i = 0; i < 25; i = i + 5){
7         tlm::tlm_command cmd = (tlm::tlm_command)(rand() % 2);
8         if (cmd == tlm::TLM_WRITE_COMMAND) {
9             data[0] = (i+1); data[1] = (i+2); data[2] = (i+3);
10            data[3] = (i+4); data[4] = (i+5);
11        }
12        nBlockWriteRead->set_command( cmd );
13        nBlockWriteRead->set_address( i );
14        nBlockWriteRead->set_data_ptr( (unsigned char*) data );
15        nBlockWriteRead->set_data_length( 5 );
16        nBlockWriteRead->set_streaming_width( 5 );
17        nBlockWriteRead->set_byte_enable_ptr( 0 );
18        nBlockWriteRead->set_dmi_allowed( false );
19        nBlockWriteRead->set_response_status( tlm::TLM_INCOMPLETE_RESPONSE );
20
21        forwardPhase = tlm::BEGIN_REQ;
22        delay = sc_time(0, SC_NS);
23        cout << "*****" << endl;
24        cout << "=> The initiator sends BEGIN_REQ for " << (cmd ? 'W' : 'R');
25        if (cmd == 1) {cout << " data: ";
26        for (int j = 0; j<5; j++) cout << data[j] << " ";
27        cout << " at " << sc_time_stamp() << " delay=" << delay << '\n';
28
29        tlm::tlm_sync_enum returnStatus;
30        returnStatus = memWRSocket->
31            nb_transport_fw(*nBlockWriteRead, forwardPhase, delay);
32        cout << "<- Return path for BEGIN_REQ is " << returnStatus << " at " << sc_time_stamp() << endl;
33        wait(130 , SC_NS);
34    }
35}

```

Generates a generic payload

Call `nb_transport_fw` for `BEGIN_REQ`

Do some operations  
Wait enough for avoiding pipeline transactions

- Example 6 - `nbFwdBwdMemoryRW`: Initiator, `memWriterReader`



## Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: Initiator, *memWriterReader*

memWriterReader.cpp memWriterReader.h memoryUnit.cpp memoryUnit.n

```

memWriterReader.cpp → X memWriterReader.h
→ memWriterReader
68     tlm::tlm_sync_enum memWriterReader::nb_transport_bw(tlm::tlm_generic_payload& trans,
69     tlm::tlm_phase& phase, sc_time& delay){
70     m_peq.notify(trans, phase, delay);
71     return tlm::TLM_ACCEPTED;
72 }
```

Move the incoming transactions to PEQ, then return TLM ACCEPTED

memWriterReader.cpp

**memWriterReader**  
SC\_MODULE

nbMemWR()

nb\_transport\_bw()

memWR\_peq()

doSomethingGood()



# Non-blocking Transport: Forward & Backward

### Example 6 – nbFwdBwdMemoryRW: Initiator, *memWriterReader*

Payload event queue callback to handle transactions from target  
Transaction could have arrived through return path or backward path

**memWriterReader.cpp**

```

52 void memWriterReader::memWR_peq(tlm::tlm_generic_payload& trans, const tlm::tlm_phase& phase){
53
54     if (phase == tlm::BEGIN_REQ || phase == tlm::END_RESP)
55         SC_REPORT_FATAL("TLM-2", "Illegal transaction phase received by initiator");
56     else if (phase == tlm::BEGIN_RESP){
57         tlm::tlm_phase forwardPhase = tlm::END_RESP;
58         sc_time delay = sc_time(10, SC_NS); Response accept delay = 10ns
59         cout << "=> The initiator sends END_RESP at " << sc_time_stamp() << " delay=" << delay << endl;
60         tlm::sync_enum returnStatus =
61             memWRSocket->nb_transport_fw(trans, forwardPhase, delay); Call nb_transport_fw for END_RES
62         cout << "<-- Return path for END_RESP is " << returnStatus << " at " << sc_time_stamp() << endl;
63         doSomethingGood(*nBlockWriteRead);
64     }
65 }
```

The initiator can process the data of the incoming transaction



memWriterReader memoryUnit



## Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: Initiator, *memWriterReader*

memWriterReader.cpp → X memWriterReader.h

```

memWriterReader
37 void memWriterReader::doSomethingGood( tlm::tlm_generic_payload& completeTrans )
38 {
    if ( completeTrans.is_response_error() )
        SC_REPORT_ERROR("TLM-2", "error...\n");
    tlm::tlm_command cmd = completeTrans.get_command();
    uint64 adr = completeTrans.get_address();
    int* ptr = reinterpret_cast<int*>( completeTrans.get_data_ptr() );
    if (cmd == tlm::TLM_READ_COMMAND){
        cout << "The incoming data from memArray is ready to process: ";
        for (int j = 0; j < 5; j++) cout << *(ptr + j) << " ";
        cout << "at " << sc_time_stamp() << '\n';
    }
}

```

Initiator obliged to check response status on receiving *BEGIN\_RESP* or *TLM\_COMPLETED*

46  
47  
48  
49 }

The data is ready, an appropriate processing can be started on the data

memWriterReader.cpp



# Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: Target, *memoryUnit*

**memoryUnit.h**

```

memoryUnit.cpp      memoryUnit.h + x
(Global Scope)
1 #include <systemc.h>
2 #include "tlm.h"
3 #include "tlm_utils/simple_target_socket.h"
4 #include "tlm_utils/peq_with_cb_and_phase.h"
5
6 class memoryUnit : public sc_module {
7 public:
8     tlm_utils::simple_target_socket<memoryUnit, 32> memSocket;
9     static const int SIZE=32;
10
11 SC_CTOR(memoryUnit) : memSocket("memory_side_socket"),
12                         m_peq(this, &memoryUnit::mem_peq) {
13
14     memSocket.register_nb_transport_fw(this, &memoryUnit::nb_transport_fw);
15     for (int i = 0; i < SIZE; i++)
16         memArray[i] = (i + 82);
17 }
18 virtual tlm::tlm_sync_enum nb_transport_fw( tlm::tlm_generic_payload&, tlm::tlm_phase&, sc_time& );
19 tlm_utils::peq_with_cb_and_phase<memoryUnit> m_peq;
20 void mem_peq(tlm::tlm_generic_payload&, const tlm::tlm_phase& );
21 void send_ENDREQ(tlm::tlm_generic_payload& );
22 void send_BEGINRESP(tlm::tlm_generic_payload& );
23 int memArray[SIZE];
24 };

```

**memoryUnit SC\_MODULE**

- Register callback for incoming *nb\_transport\_fw* interface method call
- Declared a Payload Event Queue, then implement it
- Function to call *nb\_transport\_bw* for *END\_REQ*
- Function to call *nb\_transport\_bw* for *END\_REQ*

# Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: Target, *memoryUnit*

**memoryUnit.cpp**

```

memoryUnit.cpp + X memoryUnit.h
→ memoryUnit

1 #include "memoryUnit.h"
2
3 tlm::tlm_sync_enum memoryUnit::nb_transport_fw(
4     (tlm::tlm_generic_payload& receivedTrans,
5      tlm::tlm_phase& phase, sc_time& delay ){
6
7     tlm::tlm_command cmd = receivedTrans.get_command();
8     uint64 adr = receivedTrans.get_address();
9     unsigned char* ptr = receivedTrans.get_data_ptr();
10    unsigned int len = receivedTrans.get_data_length();
11    unsigned char* byt = receivedTrans.get_byte_enable_ptr();
12    unsigned int wid = receivedTrans.get_streaming_width();
13    if (byt != 0) {
14        receivedTrans.set_response_status( tlm::TLM_BYTE_ENABLE_ERROR_RESPONSE );
15        return tlm::TLM_COMPLETED;
16    if (len > 5 || wid < len) {
17        receivedTrans.set_response_status( tlm::TLM_BURST_ERROR_RESPONSE );
18        return tlm::TLM_COMPLETED;
19
20    m_peq.notify(receivedTrans, phase, delay);
21    return tlm::TLM_ACCEPTED;
22}

```

Move the incoming transactions to PEQ, then return TLM\_ACCEPTED



# Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: Target, *memoryUnit*



# Non-blocking Transport: Forward & Backward

## Example 6 – nbFwdBwdMemoryRW: Target, *memoryUnit*

**memoryUnit.cpp** ➔ **memoryUnit.h**

→ **memoryUnit**

```

63 void memoryUnit::send_BEGINRESP(tlm::tlm_generic_payload& receivedTrans){
64     tlm::tlm_sync_enum status;
65     tlm::tlm_phase backwardPhase;
66     sc_time delay;
67
68     tlm::tlm_command cmd = receivedTrans.get_command();
69     uint64 adr = receivedTrans.get_address();
70     unsigned char* ptr = receivedTrans.get_data_ptr();
71     unsigned int len = receivedTrans.get_data_length();
72
73     unsigned int i;
74     if (cmd == tlm::TLM_READ_COMMAND)
75         for (i = 0; i < len; i = i + 1)
76             *(ptr + 4*i) = *((unsigned char*)(memArray + adr + i));
77     else if (cmd == tlm::TLM_WRITE_COMMAND)
78         for (i = 0; i < len; i = i + 1)
79             *((unsigned char*)(memArray + adr + i)) = *(ptr + 4*i);
80
81     receivedTrans.set_response_status(tlm::TLM_OK_RESPONSE);
82     backwardPhase = tlm::BEGIN_RESP;
83     delay = SC_ZERO_TIME;
84     cout << "<= The target sends BEGIN_RESP at " << sc_time_stamp() << " delay=" << delay << endl;
85     status = memSocket->nb_transport_bw(receivedTrans, backwardPhase, delay);
86     cout << "--> Return path for BEGIN_RESP is " << status << " at " << sc_time_stamp() << endl;
87     if (cmd == tlm::TLM_READ_COMMAND) cout << "The data is read from memArray: ";
88     else cout << "The data is written in memArray: ";
89     for (int j = 0; j < 5; j++) cout << static_cast<int>(*((unsigned char*)(memArray + adr + j))) << " ";
90     cout << "at " << sc_time_stamp() << endl;
91 }

```

**memoryUnit.cpp**

Implement read and write commands

Call `nb_transport_bw` for `BEGIN_RESP`



# Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: Target, *memoryUnit*



# Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: Testbench



```

memWriterReader.cpp      memWriterReader_TB.h  memWriterReader.h
nbFwdMemoryRW          (Global Scope)        m
1 #include "memWriterReader.h"
2 #include "memoryUnit.h"
3
4 SC_MODULE(memWriterReader_TB)
5 {
6     memWriterReader *WR1;
7     memoryUnit    *MU1;
8
9     SC_CTOR(memWriterReader_TB)
10 {
11     WR1 = new memWriterReader("WR");
12     MU1 = new memoryUnit("memory");
13     WR1->memWRSocket.bind(MU1->memSocket);
14 }
15 };

```

memWriterReader\_TB.h

## Testbench



# Non-blocking Transport: Forward & Backward

- Example 6 – nbFwdBwdMemoryRW: Main



```
#include "memWriterReader_TB.h"

int sc_main(int argc, char* argv[])
{
    memWriterReader_TB TB1("memWriterReader_TB");
    sc_start();
    return 0;
}
```

memWriterReader\_Main.cpp

## Non-blocking Transport: Forward & Backward

```
C:\> C:\WINDOWS\system32\cmd.exe
SystemC 2.3.1-Accellera --- Sep 29 2019 11:41:37
Copyright (c) 1996-2014 by all Contributors,
ALL RIGHTS RESERVED
=====
=> The initiator sends BEGIN_REQ for W data: 1 2 3 4 5 at 0 s delay=0 s
<-- Return path for BEGIN_REQ is 0 at 0 s
<=> The target sends END_REQ at 0 s delay=20 ns
--> Return path for END_REQ is 0 at 0 s
<=> The target sends BEGIN_RESP at 100 ns delay=0 s
--> Return path for BEGIN_RESP is 0 at 100 ns
The data is written in memArray: 1 2 3 4 5 at 100 ns
=> The initiator sends END_RESP at 100 ns delay=10 ns
<-- Return path for END_RESP is 0 at 100 ns
The current transaction was completed at 110 ns
=====
=> The initiator sends BEGIN_REQ for W data: 6 7 8 9 10 at 130 ns delay=0 s
<-- Return path for BEGIN_REQ is 0 at 130 ns
<=> The target sends END_REQ at 130 ns delay=20 ns
--> Return path for END_REQ is 0 at 130 ns
<=> The target sends BEGIN_RESP at 230 ns delay=0 s
--> Return path for BEGIN_RESP is 0 at 230 ns
The data is written in memArray: 6 7 8 9 10 at 230 ns
=> The initiator sends END_RESP at 230 ns delay=10 ns
<-- Return path for END_RESP is 0 at 230 ns
The current transaction was completed at 240 ns
=====
=> The initiator sends BEGIN_REQ for R at 260 ns delay=0 s
<-- Return path for BEGIN_REQ is 0 at 260 ns
<=> The target sends END_REQ at 260 ns delay=20 ns
--> Return path for END_REQ is 0 at 260 ns
<=> The target sends BEGIN_RESP at 360 ns delay=0 s
--> Return path for BEGIN_RESP is 0 at 360 ns
The data is read from memArray: 92 93 94 95 96 at 360 ns
=> The initiator sends END_RESP at 360 ns delay=10 ns
<-- Return path for END_RESP is 0 at 360 ns
The incoming data from memArray is ready to process: 92 93 94 95 96 at 360 ns
The current transaction was completed at 370 ns
```

Writing in the memory

Reading from the memory

### Example 6 – nbFwdBwdMemoryRW: Result



# Transaction Level Modeling

## Introduction

### + Processing Elements

### + Data

### - Abstract Communications

#### + Coding Styles

#### + TLM-2.0 Transport Interfaces

- Blocking Transport
- Simple Transport Example
- Non-blocking Transport
  - Path (Forward, Backward & Return)
  - Socket (Tagged & Multi Socket)
  - Phases & Base Protocol
  - [nbFwdBwdCPUAcc Example](#)

### + Complete System

#### The concepts covered:

- ✓ Generic payload
- ✓ Simple sockets
- ✓ Non-blocking transport
  - ✓ Forward path
  - ✓ Backward path



# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Scenario



# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Scenario

2

Communicating

Communicating

Accelerator

inReg

Reg.

+

\*

outReg

Overall view of Example 7

CPU

IR

PC

SR

ALU

R0

R1

R2

R3

R4

....

# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Scenario



# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Scenario

4

Communicating  
(Interrupt)

Overall  
view of  
Example 7



e.g. by ISR

Communicating

# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Scenario

5



# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Scenario (More details)



**Overall view of Example 7**

# Non-blocking Transport: Forward & Backward

### ○ Example 7 – nbFwdBwdCPUAcc: Scenario

- Processor (CPU) needs to perform intensive computations (for example calculating exponential or floating-point multiplication). It decides to entrust its computations to an accelerator (exponential accelerator that designed in Topic 4, or FP multiplier accelerator)
  - The processor generates a generic payload and calls *nb\_transport\_fw* to send the input data to the accelerator
  - The transfer latency from CPU to Accelerator has been considered 50ns, which includes the request accept delay, the latency of the target, and the response accept delay
  - The accelerator receives the input data after 50ns and then starts the calculations while the processor executes other instructions
  - When the calculations are done, the accelerator generates a generic payload and interrupts the CPU for the provided result by calling *nb\_transport\_bw* method
  - The transfer latency from CPU to Accelerator is 57ns
  - The processor receives the result after 57ns

**Interrupt** creates a temporary halt during program execution and allows peripheral devices to access the processor. The processor responds to that interrupt with an Interrupt Service Routine (ISR)

# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Initiator, Processor

**Processor.h**

```

Processor.h + X Processor.cpp Accelerator.h Accelerator.cpp
(Global Scope)
1 #include <systemc.h>
2 #include "tlm.h"
3 #include "tlm_utils/simple_initiator_socket.h"
4
5 class Processor : public sc_module {
6 public:
7     tlm_utils::simple_initiator_socket<Processor, 32> procSocket;
8
9     SC_CTOR(Processor) : procSocket("Processor_Socket"), inData(0)
10    {
11        inData = new tlm::tlm_generic_payload();
12        SC_THREAD(exeInstructions);
13        procSocket.register_nb_transport_bw(this, &Processor::nb_transport_bw);
14    }
15    tlm::tlm_generic_payload* inData;
16    void exeInstructions();
17    virtual tlm::tlm_sync_enum nb_transport_bw(tlm::tlm_generic_payload&, tlm::tlm_phase&, sc_time&);
18    int data;
19    int result;
20};

```

Register callback for incoming *nb\_transport\_bw* interface method call

SC\_THREAD for starting communication

**Processor SC\_MODULE**

```

exeInstructions()
procSocket->nb_transport_fw()

nb_transport_bw()

```

## Non-blocking Transport: Forward & Backward

**Processor.h**    **Processor.cpp**    **Accelerator.h**    **Accelerator.cpp**

**Processor.cpp**

```

1 #include "Processor.h"
2
3 void Processor::exeInstructions(){
4     tlm::tlm_phase forwardPhase;
5     sc_time delay;
6
7     for (int i = 64; i < 104; i = i + 8){
8         cout << "*****" << endl;
9         tlm::tlm_command cmd = tlm::TLM_WRITE_COMMAND;
10        if (cmd == tlm::TLM_WRITE_COMMAND) data = i;
11
12        inData->set_command( cmd );
13        inData->set_address( 0 );
14        inData->set_data_ptr( (unsigned char*) &data );
15        inData->set_data_length( 1 );
16        inData->set_streaming_width( 1 );
17        inData->set_byte_enable_ptr( 0 );
18        inData->set_dmi_allowed( false );
19        inData->set_response_status( tlm::TLM_INCOMPLETE_RESPONSE );
20
21        forwardPhase = tlm::BEGIN_REQ;
22        delay = sc_time(0, SC_NS);
23        cout << "The CPU sends an input data: " << (sc_lv<8>)data;
24        cout << " at " << sc_time_stamp() << " delay=" << delay << '\n';
25        tlm::tlm_sync_enum returnStatus;
26
27        returnStatus = procSocket->nb_transport_bw(*inData, forwardPhase, delay);
28        wait(delay);
29        if (returnStatus == tlm::TLM_COMPLETED){
30            if (inData->is_response_error())
31                SC_REPORT_ERROR("TLM-2", "Error in memory handling of b_transport");
32            cout << "The CPU is done with communication at " << sc_time_stamp() << endl;
33        }
34        cout << "The CPU is processing some instruction at " << sc_time_stamp() << '\n';
35        wait(1300, SC_NS);
36    }
}

```

To execute the current instruction (writing the input data in the accelerator)

Generates a generic payload

Call `nb_transport_bw` to send the input data to the accelerator

For doing some instructions and for the interrupt time including 57ns for communication

- Example 7 – nbFwdBwdCPUAcc: Initiator, Processor



# Non-blocking Transport: Forward & Backward

Processor.h    Processor.cpp    X Accelerator.h    Accelerator.cpp

Processor

```

39  tlm::sync_enum Processor::nb_transport_bw(tlm::tlm_generic_payload& receivedTrans,
40  tlm::tlm_phase& phase, sc_time& delay){
41      tlm::tlm_command cmd = receivedTrans.get_command();
42      uint64 adr = receivedTrans.get_address();
43      unsigned char* ptr = receivedTrans.get_data_ptr();
44      unsigned int len = receivedTrans.get_data_length();
45      unsigned char* byt = receivedTrans.get_byte_enable_ptr();
46      unsigned int wid = receivedTrans.get_streaming_width();
47
48      if (byt != 0) {
49          receivedTrans.set_response_status(tlm::TLM_BYTE_ENABLE_ERROR_RESPONSE);
50          return tlm::TLM_COMPLETED;
51      }
52      if (len != 1 || wid != len) {
53          receivedTrans.set_response_status(tlm::TLM_BURST_ERROR_RESPONSE);
54          return tlm::TLM_COMPLETED;
55      }
56
57      if (cmd == tlm::TLM_READ_COMMAND)
58          *(ptr) = *((unsigned char*)&data + adr);
59
60      else if (cmd == tlm::TLM_WRITE_COMMAND)
61          *((unsigned char*)&result + adr) = *(ptr);
62
63      receivedTrans.set_response_status(tlm::TLM_OK_RESPONSE);
64      delay = delay + sc_time(57, SC_NS);
65      cout << "The result data: " << (sc_lv<8>)((unsigned char*)&result + adr));
66      cout << " was received by the CPU at " << sc_time_stamp() << endl;
67
68  }
```

- Example 7 – nbFwdBwdCPUAcc: Initiator, Processor



# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Target, Accelerator

**Accelerator.h**

```

Processor.h Processor.cpp Accelerator.h ✘ Accelerator.cpp
(Global Scope)
5 class Accelerator : public sc_module {
6 public:
7     tlm_utils::simple_target_socket<Accelerator, 32> accSocket;
8
9     SC_CTOR(Accelerator) : accSocket("Accelerator_socket") {
10         startCal = SC_LOGIC_0;
11
12         accSocket.register_nb_transport_fw(this, &Accelerator::nb_transport_fw);
13         SC_THREAD(expCalculation);
14         sensitive << startCal;
15
16     }
17
18     virtual tlm::tlm_sync_enum nb_transport_fw( tlm::tlm_generic_payload&,
19                                                 tlm::tlm_phase&, sc_time& );
20
21     void expCalculation();
22     int inReg;
23     int outReg;
24     sc_signal <sc_logic> startCal;
};

```

Function for accelerator calculation and calling *nb\_transport\_fw* to interrupt the CPU for communicating back

Register callback for incoming *nb\_transport\_fw* interface method call

**Accelerator SC\_MODULE**

*nb\_transport\_fw()*

*expCalculation()*

*procSocket->nb\_transport\_bw()*

# Non-blocking Transport: Forward & Backward

Processor.h    Processor.cpp    Accelerator.h    Accelerator

→ Accelerator

Accelerator.cpp

```
1 #include "Accelerator.h"
2
3 tlm::tlm_sync_enum Accelerator::nb_transport_fw
4     (tlm::tlm_generic_payload& receivedTrans,
5      tlm::tlm_phase& phase, sc_time& delay ){
6
7     tlm::tlm_command cmd = receivedTrans.get_command();
8     uint64 adr = receivedTrans.get_address();
9     unsigned char* ptr = receivedTrans.get_data_ptr();
10    unsigned int len = receivedTrans.get_data_length();
11    unsigned char* byt = receivedTrans.get_byte_enable_ptr();
12    unsigned int wid = receivedTrans.get_streaming_width();
13
14    if (byt != 0) {
15        receivedTrans.set_response_status( tlm::TLM_BYTE_ENABLE_ERROR_RESPONSE );
16        return tlm::TLM_COMPLETED;
17    }
18    if (len != 1 || wid != len) {
19        receivedTrans.set_response_status( tlm::TLM_BURST_ERROR_RESPONSE );
20        return tlm::TLM_COMPLETED;
21    }
22
23    if ( cmd == tlm::TLM_READ_COMMAND )
24        *(ptr) = *((unsigned char*) (&outReg));
25    else if ( cmd == tlm::TLM_WRITE_COMMAND )
26        *((unsigned char*)(&inReg + adr)) = *(ptr);
27
28    startCal = ~startCal.read();
29    receivedTrans.set_response_status( tlm::TLM_OK_RESPONSE );
30    delay = delay + sc_time(50, SC_NS);
31
32 }
```

- Example 7 – nbFwdBwdCPUAcc:  
*Target, Accelerator*



# Non-blocking Transport: Forward & Backward

**Accelerator.cpp**

```

Processor.h      Processor.cpp      Accelerator.h      Accelerator.cpp
  Accelerator
34 void Accelerator::expCalculation(){
35     while (true){
36         wait(50, SC_NS);
37         cout << "The input data: " << (sc_lv<8>)inReg;
38         cout << " was received by the accelerator at " << sc_time_stamp() << endl;
39         // do some calculations
40         cout << "The accelerator starts the calculations at " << sc_time_stamp() << endl;
41         wait((rand() % 200) + 400, SC_NS);
42
43         tlm::tlm_generic_payload* outData = new tlm::tlm_generic_payload;
44         tlm::tlm_command cmd = tlm::TLM_WRITE_COMMAND;
45         if (cmd == tlm::TLM_WRITE_COMMAND) outReg = rand();
46         outData->set_command(cmd);
47         outData->set_address(3);
48         outData->set_data_ptr((unsigned char*)&outReg);
49         outData->set_data_length(1);
50         outData->set_streaming_width(1);
51         outData->set_byte_enable_ptr(0);
52         outData->set_dmi_allowed(false);
53         outData->set_response_status(tlm::TLM_INCOMPLETE_RESPONSE);
54
55         tlm::tlm_phase backwardPhase = tlm::BEGIN_REQ;
56         sc_time delay = sc_time(0, SC_NS);
57         cout << "The accelerator sends the result: " << (sc_lv<8>)outReg;
58         cout << " at " << sc_time_stamp() << " delay=" << delay << endl;
59         tlm::tlm_sync_enum returnStatus;
60         returnStatus = accSocket->nb_transport_bw(*outData, backwardPhase, delay);
61         wait(delay);
62         if (returnStatus == tlm::TLM_COMPLETED){
63             if (outData->is_response_error())
64                 SC_REPORT_ERROR("TLM-2", "Error in memory handling of b_transport");
65             cout << "The accelerator is done and ready for the next execution at " << sc_time_stamp() << endl;
66         }
67     }
68 }
69 }
```

**Transfer latency = 50 ns**

**Calculation time**

**Generates a generic payload**

**Transfer latency = 57 ns**

**Call nb\_transport\_bw to interrupt the processor for the provided result**

- Example 7 – nbFwdBwdCPUAcc: Target, Accelerator



# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Testbench

**Testbench.h**

```

Testbench.h ✎ X Processor.h Processor.cpp
(Global Scope)
1 #include "Processor.h"
2 #include "Accelerator.h"
3
4 SC_MODULE(Testbench)
{
5     Processor *CPU;
6     Accelerator *Acc;
7
8     SC_CTOR(Testbench)
9     {
10         CPU = new Processor("Processor");
11         Acc = new Accelerator("Accelerator");
12         CPU->procSocket.bind(Acc->accSocket);
13     }
14 };
15 
```

## Testbench

### Processor SC\_MODULE

exeInstructions()

nb\_transport\_bw()

### Accelerator SC\_MODULE

nb\_transport\_fw()

expCalculation()

# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Main



Main.cpp

```
(Global Scope)
1 #include "Testbench.h"
2
3 int sc_main(int argc, char* argv[])
4 {
5     Testbench TB("Testbench");
6     sc_start();
7     return 0;
8 }
```

# Non-blocking Transport: Forward & Backward

- Example 7 – nbFwdBwdCPUAcc: Result

```
C:\> C:\WINDOWS\system32\cmd.exe

SystemC 2.3.1-Accellera --- Sep 29 2019 11:41:37
Copyright (c) 1996-2014 by all Contributors,
ALL RIGHTS RESERVED
*****
The CPU sends an input data: 01000000 at 0 s delay=0 s
The CPU is done with communication at 50 ns
The CPU is processing some instruction at 50 ns
The input data: 01000000 was received by the accelerator at 50 ns
The accelerator starts the calculations at 50 ns
The accelerator sends the result: 00100011 at 491 ns delay=0 s
The result data: 00100011 was received by the CPU at 491 ns
The accelerator is done and ready for the next execution at 548 ns
*****
The CPU sends an input data: 01001000 at 1350 ns delay=0 s
The CPU is done with communication at 1400 ns
The CPU is processing some instruction at 1400 ns
The input data: 01001000 was received by the accelerator at 1400 ns
The accelerator starts the calculations at 1400 ns
The accelerator sends the result: 10000100 at 1934 ns delay=0 s
The result data: 10000100 was received by the CPU at 1934 ns
The accelerator is done and ready for the next execution at 1991 ns
*****
```

