# Cycle Model Studio Version 9.2

**SystemC User Manual** 

Non-Confidential



### Cycle Model Studio SystemC User Manual

Copyright © 2017 ARM Limited. All rights reserved.

#### **Release Information**

The following changes have been made to this document.

#### **Change History**

| Date          | Issue | Confidentiality  | Change                     |
|---------------|-------|------------------|----------------------------|
| November 2016 | A     | Non-Confidential | Restamp release with 9.0.0 |
| May 2017      | В     | Non-Confidential | Updated for v9.2           |

#### **Non-Confidential Proprietary Notice**

This document is protected by copyright and other related rights and the practice or implementation of the information contained in this document may be protected by one or more patents or pending patent applications. No part of this document may be reproduced in any form by any means without the express prior written permission of ARM Limited ("ARM"). No license, express or implied, by estoppel or otherwise to any intellectual property rights is granted by this document unless specifically stated.

Your access to the information in this document is conditional upon your acceptance that you will not use or permit others to use the information for the purposes of determining whether implementations infringe any patents.

THIS DOCUMENT IS PROVIDED "AS IS". ARM PROVIDES NO REPRESENTATIONS AND NO WARRANTIES, EXPRESS, IMPLIED OR STATUTORY, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF MERCHANTABILITY, SATISFACTORY QUALITY, NON-INFRINGEMENT OR FITNESS FOR A PARTICULAR PURPOSE WITH RESPECT TO THE DOCUMENT. For the avoidance of doubt, ARM makes no representation with respect to, and has undertaken no analysis to identify or understand the scope and content of, third party patents, copyrights, trade secrets, or other rights.

This document may include technical inaccuracies or typographical errors.

This document may be translated into other languages for convenience, and you agree that if there is any conflict between the English version of this document and any translation, the terms of the English version shall prevail.

TO THE EXTENT NOT PROHIBITED BY LAW, IN NO EVENT WILL ARM BE LIABLE FOR ANY DAMAGES, INCLUDING WITHOUT LIMITATION ANY DIRECT, INDIRECT, SPECIAL, INCIDENTAL, PUNITIVE, OR CONSEQUENTIAL DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY OF LIABILITY, ARISING OUT OF ANY USE OF THIS DOCUMENT, EVEN IF ARM HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.

This document consists solely of commercial items. You shall be responsible for ensuring that any use, duplication or disclosure of this document complies fully with any relevant export laws and regulations to assure that this document or any portion thereof is not exported, directly or indirectly, in violation of such export laws. Use of the word "partner" in reference to ARM's customers is not intended to create or refer to any partnership relationship with any other company. ARM may make changes to this document at any time and without notice.

If any of the provisions contained in these terms conflict with any of the provisions of any signed written agreement specifically covering this document with ARM, then the signed written agreement prevails over and supersedes the conflicting provisions of these terms.

Words and logos marked with \* or ™ are registered trademarks or trademarks of ARM Limited or its affiliates in the EU and/or elsewhere. All rights reserved. Other brands and names mentioned in this document may be the trademarks of their respective owners. You must follow the ARM trademark usage guidelines http://www.arm.com/about/trademarks/guidelines/index.php.

Copyright © ARM Limited or its affiliates. All rights reserved. ARM Limited. Company 02557590 registered in England. 110 Fulbourn Road, Cambridge, England CB1 9NJ.

In this document, where the term ARM is used to refer to the company it means "ARM or any of its subsidiaries as appropriate".

#### **Confidentiality Status**

This document is Non-Confidential. The right to use, copy and disclose this document may be subject to license restrictions in accordance with the terms of the agreement entered into by ARM and the party that ARM delivered this document to.

#### **Product Status**

The information in this document is final, that is for a developed product.

#### Web Address

http://www.arm.com

# Contents

| Chapter 1. Overview                                    |    |
|--------------------------------------------------------|----|
| Integration Benefits                                   |    |
| Accellera SystemC Installation                         |    |
| Example Design                                         |    |
| Chapter 2.                                             |    |
| Integration Example                                    |    |
| Setup                                                  |    |
| Creating the Cycle Model                               |    |
| Creating the Component for SystemC                     |    |
| Customizing the Component for SystemC                  |    |
| Compiling the Component for SystemC                    |    |
| Linking the Executable                                 |    |
| Running the Testbench                                  |    |
| Chapter 3.                                             |    |
| Command Line Reference                                 |    |
| Chapter 4.                                             |    |
| Accessing Internal Signals                             |    |
| Setting Directives During the Cycle Model Compiler Run | 11 |
| Accessing Signals During Simulation                    |    |

| Chapter 5. Waveforms      |  |
|---------------------------|--|
| Enabling Waveform Dumping |  |
| Chapter 6. Testbench      |  |

# Overview

1

This guide discusses how to integrate a Cycle Model into a SystemC<sup>TM</sup> development environment. Specifically, we examine the case of replacing an existing module in a design with a Cycle Model. This process includes building the Cycle Model with the Cycle Model Compiler, automatically generating a system module to include this Cycle Model into a SystemC environment, and compiling the design into an executable.

### 1.1 Integration Benefits

SystemC is a design environment that provides the user with an effective method for refining a design from a high level of abstraction to an implementation level model. A SystemC model can be used to verify the design and integrate system software. However, when a SystemC model is refined down to an implementation level it begins to suffer from performance bottlenecks and much of its value is diminished.

ARM products allow designers to compile their synthesizable RTL into an ultra-high performance Cycle Model that can then be linked directly into a SystemC design environment. This integration provides the speed necessary to perform software validation and performance modeling while maintaining the investment already made in the SystemC environment.

A traditional approach to integrate implementation-level RTL into SystemC typically involves integrating an RTL simulator via the PLI. While this approach will correctly model the implementation, it will do so at a relatively slow speed. In addition, PLI introduces a substantial amount of interconnect overhead.

### 1.2 Accellera SystemC Installation

Cycle Model Compiler code works with SystemC Version 2.3.1. To dowload it, visit http://accellera.org/downloads/standards/systemc.

# 1.3 Example Design

The design used to demonstrate this application is a simple FIFO with separate read and write clocks. A SystemC test-bench is included that randomly loads and unloads data into the FIFO.

To integrate this design into SystemC, the SystemC wrapper tool generates a component for SystemC (the wrapper) that invokes the Cycle Model at the correct times and makes sure that data is correctly transferred in and out of the Cycle Model. The following figure illustrates the design referenced in this guide.



# Integration Example

The basic integration procedure includes the following steps (details are included in the subsections that follow):

- 1. Identify the Verilog module to be used in a SystemC simulation and compile it with Cycle Model Studio. This generates the Cycle Model.
- 2. Run the SystemC wrapper tool (see "Creating the Component for SystemC" on page 2-4 for details):
  - If you are not using Cycle Model Studio:

```
carbon systemCWrapper libdesign.io.db
  [-portType <portName>=<systemCType>]
  [-moduleName name]
```

• If you are using Cycle Model Studio:

```
carbon systemCWrapper -ccfg design.ccfg
```

3. Link the Cycle Model and component for SystemC with the other testbench code used in the existing environment.

### 2.1 Setup

The general integration concepts discussed in the following steps assumes a fully compiled SystemC build area, as well as a working knowledge of the following:

- SystemC
- The Cycle Model Compiler command line and options

All of the following steps are executed automatically when you run make in the \$CARBON\_HOME/app\_notes/systemC directory. In order to compile the design correctly, the Makefile for the example needs to be modified.

- SYSTEMC\_ARCH should be set based on the architecture of the computer where the compile is taking place. It should be set to linux for Linux machines.
- SYSTEMC\_HOME should be set to the top directory in which the SystemC release has been compiled. This example has been validated against the SystemC shipped with the ARM tools.
- CARBON HOME must be set to the ARM installation directory.
- When running on a 64-bit Linux machine, CARBON\_HOST\_ARCH and CARBON\_TARGET\_ARCH must be set correctly depending on whether you want to create a 32- or 64-bit Cycle Model. See "Setting System Architecture Variables" in the *Cycle Model Studio Installation Guide* for more information.
- Optionally, if you want to dump waveforms during runtime, uncomment one of the following lines in the Makefile:

```
#WAVE_DUMP = -DCARBON_DUMP_VCD=1
#WAVE DUMP = -DCARBON DUMP FSDB=1
```

# 2.2 Creating the Cycle Model

Create the Cycle Model for the FIFO. This is done by invoking the Cycle Model Compiler and referencing the RTL for the FIFO module. The Verilog source and Makefile for this example are located in the app notes/systemC/ subdirectory.

```
> make libfifo.a
```

# 2.3 Creating the Component for SystemC

Use the SystemC wrapper tool to create the component for SystemC (a SystemC wrapper around the Cycle Model):

```
> carbon systemCWrapper libfifo.io.db
```

The SystemC wrapper tool generates two SystemC files: libfifo.systemc.cpp and libfifo.systemc.h.

See "Command Line Reference" on page 3-9 for more command line options.

### 2.3.1 Customizing the Component for SystemC

At this point, if you want to customize the component for SystemC while ensuring that the edits are reused if you rerun the SystemC wrapper tool, you can edit the "Carbon User Code" sections of the libfifo.systemc.cpp and libfifo.systemc.h files, as shown in the following example:

```
// from libdesign.systemc.cpp
void component::end_of_elaboration()
{
    // CARBON USER CODE [PRE componentEnd Of Elaboration] BEGIN
    /* code that goes here is preserved when the wrapper is regenerated
    */
    cout << "before end of elaboration";
    // CARBON USER CODE END
    ... generated code to initialize model outputs goes here ...
    // CARBON USER CODE [POST componentEnd Of Elaboration] BEGIN
    cout << "after end of elaboration";
    // CARBON USER CODE END
}</pre>
```

Empty "Carbon User Code" sections or sections containing white space are ignored. Any "Carbon User Code" sections that are unused during component generation are appended to the end of the generated source in an #if 0 block and a warning is generated.

Any changes made to the Carbon User Code section of the source files are retained when you re-run SystemC using the configuration file, assuming that you do not move the source files from the output directory.

You must use the generated Carbon User Code sections; if you insert your own Carbon User Code sections, they will be ignored.

### 2.4 Compiling the Component for SystemC

The SystemC module that encapsulates the Cycle Model needs to be compiled into an object so that it can be linked into an executable. The testbench and top-level files need to be compiled as well. In the commands below, be sure to replace <SYSTEMC\_INCLUDE\_PATH> with the correct path for your SystemC installation.

```
> g++ -c -I <SYSTEMC_INCLUDE_PATH> -I$CARBON_HOME/include libfifo.systemc.cpp
> g++ -c -I <SYSTEMC_INCLUDE_PATH> -I$CARBON_HOME/include fifo_tb.cpp
> g++ -c -I <SYSTEMC_INCLUDE_PATH> -I$CARBON_HOME/include main.cpp

or
> make libfifo.systemc.o fifo_tb.o main.o
```

In this example, the Makefile contains the comments which can be edited to turn on waveform dumping. For details about controlling waveform dumping, see "Enabling Waveform Dumping" on page 5-15.

Note: The CARBON\_LIB\_LIST make variable links the program so that LD\_LIBRARY\_PATH overrides -rpath, therefore, a single GCC version should be used within your environment to avoid library conflicts. While a Cycle Model itself has no dependencies on compiler libraries, custom code compiled with the ARM-provided GCC may. If this code is integrated into an environment that uses a different version of GCC (for example, a third-party tool), runtime errors may occur. In environments such as this, it is recommended that the GCC provided by the third-party tool be used to compile the custom code.

### 2.5 Linking the Executable

All of the files that were generated now need to be linked into a single executable. There are a number of required libraries that need to be linked to form the executable.

```
> make run.x
```

To compile a component for SystemC, the following are the required g++ options (using Makefile notation):

Non-SystemC includes, options, and sources required by the Cycle Model:

```
-I$(CARBON_HOME)/include -I$(CARBON_MODEL_DIR)
-c $(CARBON_MODEL_DIR)/lib<design>.systemc.cpp
-m32 # or -m64 for Linux64
```

SystemC-related includes, libraries, and options that are required:

```
-I$(SYSTEMC_HOME)/include
-fexceptions
```

To link a Cycle Model to SystemC, the following are the required g++ options (using Makefile notation):

Non-SystemC includes, library, options, and objects required by the Cycle Model:

```
-I$(CARBON_HOME)/include -I$(CARBON_MODEL_DIR)
-L$(CARBON_HOME)/Linux/lib/gcc/shared -lcarbon5
$(CARBON_MODEL_DIR)/lib<design>.a lib<design>.systemc.o
```

SystemC-related includes, libraries, and options that are required:

```
-I. -I$(SYSTEMC_HOME)/include
-L. -L$(SYSTEMC_HOME)/lib-$(SYSTEMC_ARCH)
-fexceptions
```

where user-defined environment variables are as follows:

```
CARBON_HOME — Cycle Models installation directory

SYSTEMC_HOME — SystemC installation directory

CARBON MODEL DIR — Location of Cycle Model and component for SystemC
```

To run the SystemC simulation with a Cycle Model, the following shared library *must* be accessible at link time location or LD LIBRARY PATH, etc.

```
libcarbon5 so
```

This file can be found in the following locations:

Linux: \$CARBON\_HOME/Linux/lib/gcc/shared/libcarbon5.so Linux64: \$CARBON\_HOME/Linux64/lib/gcc/shared/libcarbon5.so

# 2.6 Running the Testbench

Once the example is built using the Makefile, type:

```
> ./run.x
```

to execute the design with the embedded Cycle Model. The SystemC copyright should be displayed, followed a few seconds later by the following:

```
Simulation exited at time 50000000 No errors detected.
```

If any errors are detected during execution, a message is displayed that provides the timestamp, expected value, and actual value. Errors can be forced by modifying the initial value of x\_data\_out in the fifo\_tb.cpp file.

# Command Line Reference

The carbon systemCWrapper tool has the following command line options:

### -ccfg design.ccfg

If you are using Cycle Model Studio, you should designate the .ccfg file on the command line.

### -portType <portName>=<systemCType>

You can use the following <systemCType> declarations:

- int
- uint | unsigned | unsigned int
- sc\_int
- sc bigint
- sc\_uint
- sc biguint
- long | long int
- ulong | unsigned long | unsigned long int
- char
- uchar | unsigned char
- sc\_logic
- sc lv
- bool
- sc bv

The uint, unsigned, long, ulong, and uchar aliases remove the need for quotes in the port-Type declaration, as shown in the following equivalent declarations:

```
-portType "a=unsigned long"
-portType a=ulong
```

The -portType option is not necessary if you are using Cycle Model Studio, as the declarations can be made in the .ccfq file.

#### -moduleName <name>

Allows you to specify the name of the generated SC MODULE on the command line.

The -moduleName option is not necessary if you are using Cycle Model Studio, as the declarations can be made in the .ccfg file.

### -forceUpdate

If this option is specified, calls to sc\_prim\_channel::request\_update are forced for all input changes. If this is not specified, request\_update is called only for clock, reset, and feed-through inputs.

# Accessing Internal Signals

To access internal signals in an SC\_MODULE, you need to set directives and then access the correct member variables during simulation.

## 4.1 Setting Directives During the Cycle Model Compiler Run

There are two types of directives that can be used to access internal signals:

• The sc directives: scObserveSignal and scDepositSignal. These directives are similar to the observeSignal and depositSignal directives in that they allow external access into the design. These directives also add an sc\_signal variable to the generated SC\_MODULE. This allows the surrounding SystemC environment to directly access the internal signal via SystemC functions and data types. Nets declared with scObserveSignal and scDepositSignal are shadowed by System C member variables in the SC MODULE for the DUT.

Because the value of scObserveSignal and scDepositSignal nets are updated every time the Cycle Model is executed, you should use them only when necessary for optimal performance.

• The standard directives: observeSignal, depositSignal, and forceSignal.

Memories must be accessed with these directives. In addition, all signals that only need to be observed or deposited a few times should be accessed with these directives for best performance. These directives require that you access the signals from the testbench SC\_MODULE:

```
carbonNetID* popHandle; // works for
   // sc directives (e.g. scObserveSignal) and
   // standard directives (e.g. observeSignal)
```

Nets declared with observeSignal, depositSignal, and forceSignal are not automatically shadowed using SystemC member variables, however, the Cycle Model API can be used to access them via net handles in the testbench as shown in the next section. The API uses the underlying CarbonObjectID\*, which is available in the generated SC MODULE.

## 4.2 Accessing Signals During Simulation

After setting the directives during the Cycle Model Compiler run as outlined above, how you access those signals during simulation depends on whether you access them through the test-bench or through the component for SystemC.

1. Accessing signals via the component for SystemC.

To use the SystemC wrapper member variable to access your signals, the signals must have been marked with the sc directives, e.g. scobserveSignal.

An example of this is shown in the example run earlier. The internal signal num\_pops is marked with an scObserveSignal directive. As a result, its value can be seen directly in main.cpp:

```
cout << "Fifo was popped " << FIFO.num pops << " times" << endl;</pre>
```

2. Accessing signals via the testbench member variables.

You can use the testbench to access member variables created with either type of directive; e.g. scObserveSignal or observeSignal.

The following SystemC example, twocounter, creates an SC\_MODULE to encapsulate the testbench, storing internal RTL signals as CarbonNetIDs. The DUT is instantiated underneath the testbench. The twocounter example, twocounter.v, contains the DUT and is located in:

\$CARBON\_HOME/app\_notes/systemC/twocounter. The Makefile contains the build rules for the examples. To build the twocounter example, type:

make twocounter

The twocounter.cpp file contains the SystemC testbench; "carbon\_testbench" has declarations for member variables to access internal signals:

```
SC_MODULE( carbon_testbench ) {
public:
    // Member variable declaration for handles to internal
    // Carbon signals
    CarbonNetID* mInternalSignal1; // marked as forcible in directive
    CarbonNetID* mInternalSignal2; // marked as observable in directive
...
```

These signals are initialized in the constructor, right after the ports are hooked up between the testbench and the DUT:

```
// Constructor for the testbench
SC_CTOR( carbon_testbench ) : mTwoCounter("inst") {
....
// Member variable initialization for handles to internal
// Carbon signals
```

Using assertions forces SystemC to abort if the hierarchical name lookup fails. This could happen if the DUT signal names are changed and the DUT is recompiled without updating the test-bench code. Note that the Cycle Models C API also prints an error message to stderr if the lookup fails. The error message can be redirected with carbonAddMsqCB.

From this point forward, you can use the member variables mInternalSignal1 and mInternalSignal2 with Cycle Models C API functions to manipulate the internal signals:

To simplify use of some Cycle Model API functions, helper methods are included in the generated SC\_MODULE, see the *SystemC Wrapper Methods Reference Manual* for more information.

# Waveforms

There are two ways to affect waveforms for simulations:

- Waveform dumping
- Waveform, profile data flushing, and exit() calls

# 5.1 Enabling Waveform Dumping

The generated SystemC module exposes the following Cycle Model API functions as methods. Please see the *SystemC Wrapper Methods Reference Manual* for details about these functions.

- carbonSCWaveInitVCD(const char\* fileName, sc\_time\_unit timescale)
- carbonSCWaveInitFSDB(const char\* fileName, sc time unit timescale)
- carbonSCWaveInitFSDBAutoSwitch (const char\* fileNamePrefix=NULL, sc\_time\_unit timescale=SC\_PS, unsigned int limitMegs=1000, unsigned int maxFiles=0)
- carbonSCDumpVars(unsigned int depth = 0, const char\* listOfScopesNets = NULL)
- carbonSCDumpVar(CarbonNetID \*handle)
- carbonSCDumpStateIO(unsigned int depth = 0, const char\* listOfScopesNets = NULL)
- carbonSCDumpOn() carbonDumpOff()

There are two ways to turn on waveform dumping during the simulation run; you can dump all variables or you can control what is dumped.

### To dump all variables

Add one of the following to the gcc compile line (depending on the desired file type) of the C++ module that instantiates the design:

```
-DCARBON_DUMP_VCD=1
-DCARBON_DUMP_FSDB=1
```

These commands activate carbonSchedule on all clock edges and make all necessary API calls for you.

#### To control the variables that are dumped, the timescale, and the filename

1. Activate complete waveforms by adding the following macros to the gcc compile line for lib<design>.systemc.cpp:

```
-DCARBON SCHED ALL CLKEDGES=1
```

2. Turn on waveform dumping by calling these methods:

#### For example,

```
mTwoCounter.carbonSCWaveInitVCD(<myFile>.vcd, SC_NS);
mTwoCounter.carbonSCDumpVars(1);
```

To simplify usage even further, the following additional wave dumping logic is included with the component for SystemC:

If carbonSCDumpVar, carbonSCDumpVars, or carbonSCDumpStateIO is called before a call to carbonWaveInit{VCD|FSDB}, then a VCD wave file named carbon <module instance name>.vcd is automatically initialized.

This makes it possible to enable dumping of all signals in a given model instance with a single line of SystemC code:

```
modelInstance.carbonSCDumpVars();
```

### 5.2 Waveform, Profile Data Flushing, and exit() Calls

To improve performance, waveforms and profile data are cached, and written to their respective files 1) when the buffer is full, and 2) when the object is destroyed. However, if you call exit() prior to the object being destroyed, the program is killed before the cached data is flushed, and the data is lost.

### To ensure waveform dumping, use one of the following methods:

• Call sc stop() rather than exit():

```
sc_start(10000);
...
sc_stop();
```

• Explicitly flush the waveforms immediately before the exit() call. Note that this requires using a handle to the SystemC module:

```
Foo *foo = new Foo(...);
...
foo->carbonSCDumpFlush(); // explicitly flush waveforms
exit();
...
delete foo;
```

Note: The sc\_stop() approach writes out both waveforms and profiling data. The carbonSCDump-Flush approach only writes out waveforms.

| Cł         | ובר | nt  | اΩ | r R |
|------------|-----|-----|----|-----|
| <b>U</b> I | ıa  | IJι | CI | U   |

# **Testbench**

To simulate a SystemC Cycle model with data in a vector file, you can create a SystemC test-bench around an SC\_MODULE by executing the following command:

carbon systemCTestbench lib<design>.io.db <vectorFile>

The testbench file, called lib<design>.sctestbench.cpp, is placed in your run directory. Clocks and resets are included in the vectors, so they are treated as ordinary inputs.