

# The Definitive Guide to SystemC: The SystemC Language

David C Black, Doulos



## The SystemC Language



THURSDAY IS  
TRAINING DAY



- Introduction to SystemC
  - Overview and background
  - Central concepts
  - The SystemC World
  - Use cases and benefits
- Core Concepts and Syntax
- Bus Modeling
- Odds and Ends

# What is SystemC?

System-level modeling language

- Network of communicating processes (c.f. HDL)
- Supports heterogeneous models-of-computation
- Models hardware and software

C++ class library

- Open source proof-of-concept simulator
- Owned by Accellera Systems Initiative



# Features of SystemC



THURSDAY IS  
TRAINING DAY

- Modules (structure)
- Ports (structure)
- Processes (computation, concurrency)
- Channels (communication)
- Interfaces (communication refinement)
- Events (time, scheduling, synchronization)
- Data types (hardware, fixed point)



# Modules and Channels



THURSDAY IS  
TRAINING DAY



# Architecture of SystemC



THURSDAY IS  
TRAINING DAY





Formed from Open-SystemC Initiative (OSCI) and Accellera  
Website <http://www.accellera.org/community/systemc>

## National Users Groups

- <http://www-ti.informatik.uni-tuebingen.de/~systemc>
- <http://www.nascug.org>
- <http://www.iscug.in>
- Others...

## Language Working Group LWG

- IEEE 1666™ -2005
  - SystemC 2.2 released Mar 2006 (IEEE 1666 compliant)
- IEEE 1666™ -2011
  - SystemC 2.3.1 released Apr 2014 (IEEE 1666 compliant)
  - Includes TLM-2

# Other Working Groups



THURSDAY IS  
TRAINING DAY

## Verification Working Group VWG

- SystemC Verification (SCV) library released 2003

## Synthesis Working Group SWG

- Synthesisable subset of SystemC

## Transaction-Level Modeling Working Group TLMWG

- TLM-1.0 released May 2005
- TLM-2.0 released June 2008
- TLM-2.0 part of IEEE 1666-2011

## Analog and Mixed Signal Working Group AMSWG

## Control, Configuration & Inspection Working Group CCIWG

# What can you do with SystemC?



THURSDAY IS  
TRAINING DAY

## Discrete Event Simulation

- Register Transfer Level (events, time)
- Transaction Level (delta delays, bus resolution)
- Kahn Process Networks (communication using function calls)
- Dataflow (infinite fifos, reads block when empty)
- CSP (input-execution-output stages)
- CSP (rendezvous, blocking reads & writes)

## Continuous Time or Frequency Domain (Analog) SystemC AMS

NOT gate level

## NOT abstract RTOS modeling

(process scheduling, priorities, pre-emption)  
(originally planned as version 3)

# Typical Use Case: Virtual Platform



THURSDAY IS  
TRAINING DAY

Multiple software stacks



Multiple buses and bridges



TLM-2.0

Digital and analog hardware IP blocks

# What is SystemC Used For?



- Virtual Platform
  - Architectural exploration, performance modeling
  - Software development
  - Reference model for functional verification
  - Available before RTL - **early!**
  - Simulates much faster than RTL - **fast!**
- High-level synthesis
- Used by
  - Architects, software, hardware, and verification engineers

# SystemC is Glue!

- Transaction-level modeling is communication-centric



# Why SystemC in Particular?



- Industry standard IEEE 1666™
- Open-source C++ simulator
  - Available across multiple platforms (Linux, Windows, Mac)
  - No vendor lock-in, no licensing issues
- Easy integration with the C/C++ world
- Easy to put SystemC wrappers around existing HDL or C models
- Wide availability of tools, models and know-how
- Mature tool vendor support for co-simulation and debug

## The SystemC Language



THURSDAY IS  
TRAINING DAY

- Introduction to SystemC
- Bus Modeling
- Odds and Ends
- Core Concepts and Syntax
  - Data
  - Modules and connectivity
  - Processes & Events
  - Channels and Interfaces
  - Ports

# SystemC Data Types



THURSDAY IS  
TRAINING DAY

In namespace `sc_dt::`

| Template                        | Base class                | Description                          |
|---------------------------------|---------------------------|--------------------------------------|
| <code>sc_int&lt;W&gt;</code>    | <code>sc_int_base</code>  | Signed integer, W < 65               |
| <code>sc_uint&lt;W&gt;</code>   | <code>sc_uint_base</code> | Unsigned integer, W < 65             |
| <code>sc_bigint&lt;W&gt;</code> | <code>sc_signed</code>    | Arbitrary precision signed integer   |
| <code>sc_bignum&lt;W&gt;</code> | <code>sc_unsigned</code>  | Arbitrary precision unsigned integer |
|                                 |                           | (intermediate results unbounded)     |
| <code>sc_logic</code>           |                           | 4-valued logic: '0' '1' 'X' 'Z'      |
| <code>sc_bv&lt;W&gt;</code>     | <code>sc_bv_base</code>   | Bool vector                          |
| <code>sc_lv&lt;W&gt;</code>     | <code>sc_lv_base</code>   | Logic vector                         |
|                                 |                           |                                      |
| <code>sc_fixed&lt;&gt;</code>   | <code>sc_fix</code>       | Signed fixed point number            |
| <code>sc_ufixed&lt;&gt;</code>  | <code>sc_ufix</code>      | Unsigned fixed point number          |

# Limited Precision Integer sc\_int



THURSDAY IS  
TRAINING DAY

```
int      i;  
sc_int<8> j;  
  
i = 0x123;  
sc_assert( i == 0x123 );  
  
  
j = 0x123;  
sc_assert( j == 0x23 );  
  
  
sc_assert( j[0] == 1 );  
sc_assert( j.range(7,4) == 0x2 );  
sc_assert( concat(j,j) == 0x2323 );
```

Truncated to 8 bits

Bit select

Part select

Concatenation

- Other useful operators: arithmetic, relational, bitwise, reduction, assignment  
`length()` `to_int()` `to_string()` *implicit-conversion-to-64-bit-int*

# Logic and Vector Types



THURSDAY IS  
TRAINING DAY

sc\_logic and sc\_lv<W>

- Values SC\_LOGIC\_0, SC\_LOGIC\_1, SC\_LOGIC\_X, SC\_LOGIC\_Z
- Initial value is SC\_LOGIC\_X

No arithmetic operators

Can write values as chars and strings, i.e. '0' '1' 'X' 'Z'

```
sc_logic R, S;  
R = '1';  
S = 'Z';  
S = S & R;
```

```
sc_int<4> n = "0b1010";  
bool boo = n[3];  
sc_lv<4> lv = "01XZ";  
sc_assert( lv[0] == 'Z' );  
n += lv.to_int();  
cout << n.to_string(SC_HEX);
```

# Fixed Point Types



THURSDAY IS  
TRAINING DAY

```
sc_fixed <wl, iwl, q_mode, o_mode, n_bits> a;  
sc_ufixed<wl, iwl, q_mode, o_mode, n_bits> b;  
sc_fix c(wl, iwl, q_mode, o_mode, n_bits);  
sc_ufix d(wl, iwl, q_mode, o_mode, n_bits);
```

- |                          |                                        |
|--------------------------|----------------------------------------|
| Word length              | - number of stored bits - no limit     |
| Integer word length      | - number of bits before binary point   |
| Quantization mode        | - behavior when insufficient precision |
| Overflow mode            | - behavior when result too big         |
| Number of saturated bits | - used with wrap overflow modes        |

Compiler flag -DSC\_INCLUDE\_FX

# Data Summary



THURSDAY IS  
TRAINING DAY

| C++       |                     | SystemC                 | SystemC               | SystemC               |
|-----------|---------------------|-------------------------|-----------------------|-----------------------|
| Unsigned  | <b>unsigned int</b> | <b>sc_bv, sc_lv</b>     | <b>sc_uint</b>        | <b>sc_biguint</b>     |
| Signed    | <b>int</b>          |                         | <b>sc_int</b>         | <b>sc_bigint</b>      |
| Precision | Host-dependent      | Limited precision       | Limited precision     | Unlimited precision   |
| Operators | C++ operators       | No arithmetic operators | Full set of operators | Full set of operators |
| Speed     | <b>Fastest</b>      | Faster                  | Slower                | <b>Slowest</b>        |

# Modules



THURSDAY IS  
TRAINING DAY



# SC\_MODULE



THURSDAY IS  
TRAINING DAY

Class

Ports

Constructor

Process

```
#include "systemc.h"

SC_MODULE(Mult)
{
    sc_in<int> a;
    sc_in<int> b;
    sc_out<int> f;

    void action() { f = a * b; }

    SC_CTOR(Mult)
    {
        SC_METHOD(action);
        sensitive << a << b;
    }
};
```

# SC\_MODULE or sc\_module?



THURSDAY IS  
TRAINING DAY

- Equivalent

```
:  
SC_MODULE (Name)  
{  
    ...  
};
```

```
struct Name: sc_module  
{  
    ...  
};
```

```
class Name: public sc_module  
{  
public:  
    ...  
};
```

# Separate Header File



```
// mult.h  
  
#include "systemc.h"  
  
SC_MODULE(Mult)  
{  
    sc_in<int> a;  
    sc_in<int> b;  
    sc_out<int> f;  
  
    void action();  
  
    SC_CTOR(Mult)  
    {  
        SC_METHOD(action);  
        sensitive << a << b;  
    }  
};
```

```
// mult.cpp  
  
#include "mult.h"  
  
void Mult::action()  
{  
    f = a * b;  
}
```

- Define constructor in .cpp?  
Yes - explained later



# Implicit read/write and Delta delays



THURSDAY IS  
TRAINING DAY

```
sc_in<int>      a, b;
```

```
SC_METHOD(action);  
    sensitive << a << b;
```

- Implicit method calls

For convenience

```
void action() { f = a * b; }
```

- Explicit method calls

write() schedules event with delta delay

```
f->write( a->read() * b->read() );
```

```
cout << f->read() << endl;
```

Old value!

# The Test Bench



# Top Level Module



THURSDAY IS  
TRAINING DAY

Header files

```
#include "systemc.h"
#include "stim.h"
#include "mult.h"
#include "mon.h"
```

Channels

```
sc_signal<int> asig, bsig, fsig;
sc_clock testclk;
```

Modules

```
Stim stim1;
Mult uut;
Mon mon1;
```

```
...
```

```
}
```

# Module Instantiation



THURSDAY IS  
TRAINING DAY

```
SC_MODULE(Top)
{
    sc_signal<int> asig, bsig, fsig;

    sc_clock testclk;

    Stim stim1;
    Mult uut;
    Mon mon1;           ← Name of data member

    SC_CTOR(Top)
    : testclk("testclk", 10, SC_NS),
      stim1("stim1"),
      uut ("uut"),
      mon1 ("mon1")
    {
        ...
    }
}
```

String name of instance (constructor argument)



# Port Binding

```
SC_CTOR(Top)
: testclk("testclk", 10, SC_NS),
stim1("stim1"),
uut("uut"),
mon1("mon1")
{
    stim1.a(asig);
    stim1.b(bsig);
    stim1.clk(testclk);

    uut.a(asig);
    uut.b(bsig);
    uut.f(fsig);

    mon1.a.bind(asig);
    mon1.b.bind(bsig);
    mon1.f.bind(fsig);
    mon1.clk.bind(testclk);
}
```

Alternative function

Port name

Channel name



- sc\_main is the entry point to a SystemC application

```
#include "systemc.h"
#include "top.h"

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

Called from main()

Instantiate one top-level module

End elaboration, run simulation

# An Alternative to sc\_main



THURSDAY IS  
TRAINING DAY

- A SystemC implementation is not obliged to support sc\_main

```
#include "top.h"

SC_MODULE_EXPORT (Top) ;

// NCSC_MODULE_EXPORT (Top) ;
```

QuestaSim

Incisive

Tool-specific macro



# Namespaces



```
#include "systemc.h"
```

```
SC_MODULE(Mod)
{
    sc_in<bool> clk;
    sc_out<int> out;

    ... cout << endl;
```

Old header - global namespace

```
#include "systemc"

SC_MODULE(Mod)
{
    sc_core::sc_in<bool> clk;
    sc_core::sc_out<int> out;

    ... std::cout << std::endl;
```

New header

```
#include "systemc"
using namespace sc_core;
using namespace sc_dt;
using std::cout;
using std::endl;

SC_MODULE(Mod) {
    sc_in<bool> clk;
    sc_out<int> out;
    ... cout << endl;
```

# Summary of Files



# Compilation and Simulation



THURSDAY IS  
TRAINING DAY



# Kinds of Process



THURSDAY IS  
TRAINING DAY

- Processes
  - Must be within a module (not in a function)
  - A module may contain many processes
- Three different kinds of process
  - Methods SC\_METHOD
  - Threads SC\_THREAD
  - Clocked threads SC\_CTHREAD (for synthesis)
- Processes can be *static* or *dynamic*

# SC\_METHOD Example



```
#include <systemc.h>

template<class T>
SC_MODULE(Register)
{
    sc_in<bool> clk, reset;
    sc_in<T> d;
    sc_out<T> q;

    void entry();

    SC_CTOR(Register)
    {
        SC_METHOD(entry);
        sensitive << reset;
        sensitive << clk.pos();
    }
};
```

```
template<class T>
void Register<T>::entry()
{
    if (reset)
        q = 0; // promotion
    else if (clk.posedge())
        q = d;
}
```

- SC\_METHODs execute in zero time
- SC\_METHODs cannot be suspended
- SC\_METHODs should not contain infinite loops

# SC\_THREAD Example



THURSDAY IS  
TRAINING DAY

```
#include "systemc.h"

SC_MODULE(Stim)
{
    sc_in<bool> Clk;
    sc_out<int> A;
    sc_out<int> B;

    void stimulus();

    SC_CTOR(Stim)
    {
        SC_THREAD(stimulus);
        sensitive << Clk.pos();
    }
}
```

```
#include "stim.h"

void Stim::stimulus()
{
    wait();
    A = 100;
    B = 200;
    wait();
    A = -10;
    B = 23;
    wait();
    A = 25;
    B = -3;
    wait();
    sc_stop(); ← Stop simulation
}
```

for Clk edge

- More general and powerful than an SC\_METHOD
- Simulation may be slightly slower than an SC\_METHOD
- Called once only: hence often contains an infinite loop

# SC\_HAS\_PROCESS



THURSDAY IS  
TRAINING DAY

```
#include "systemc.h"
class Counter: public sc_module
{
public:
```

```
    sc_in<bool> clock, reset;
    sc_out<int> q;
```

```
    Counter(sc_module_name _nm, int _mod)
        : sc_module(_nm), count(0) , modulus(_mod)
```

```
{
```

```
    SC_HAS_PROCESS(Counter);
```

```
    SC_METHOD(do_count);
```

```
        sensitive << clock.pos();
```

```
}
```

```
private:
```

```
    void do_count();
    int      count;
    int const modulus;
```

```
} ;
```

Constructor arguments

Needed if there's a process  
and not using SC\_CTOR

# Dynamic Sensitivity



THURSDAY IS  
TRAINING DAY

```
SC_CTOR(Module)
```

```
{
```

```
    SC_THREAD(thread) ;
```

```
    sensitive << a << b;
```

```
}
```

Static sensitivity list

```
void thread()
```

```
{
```

```
    for(;;)
```

```
{
```

```
    wait();
```

```
    ...
```

```
    wait(10, SC_NS);
```

```
    ...
```

```
    wait(e);
```

```
    ...
```

```
}
```

```
}
```

Wait for event on a or b

Wait for 10ns

Wait for event e

ignore a or b



# sc\_event and Synchronization

```
SC_MODULE (Test)
{
    int data;
    sc_event e;
    SC_CTOR (Test)
    {
        SC_THREAD (producer);
        SC_THREAD (consumer);
    }
    void producer()
    {
        wait(1, SC_NS);
        for (data = 0; data < 10; data++) {
            e.notify();
            wait(1, SC_NS);
        }
    }
    void consumer()
    {
        for (;;) {
            wait(e);
            cout << "Received " << data << endl;
        }
    }
};
```

Shared variable

Primitive synchronization object

Schedule event immediately

Resume when event occurs



- Simulation time is a 64-bit unsigned integer
- Time resolution is programmable - must be power of 10 x fs
- Resolution can be set once only, before use and before simulation
- Default time resolution is 1 ps

```
enum sc_time_unit {SC_FS, SC_PS, SC_NS, SC_US, SC_MS, SC_SEC};
```

```
sc_time(double, sc_time_unit);
```

Constructor

```
void      sc_set_time_resolution(double, sc_time_unit);  
sc_time  sc_get_time_resolution();
```

```
const sc_time& sc_time_stamp();
```

Get current simulation time

# sc\_clock

```
sc_clock clk("clk",      period, 0.4, firstedge, true);
```

```
sc_clock clk("clk", 10, SC_NS, 0.4, 12, SC_NS, true);
```

Duty cycle

1st edge rising



Defaults

```
("clock_N", 1, SC_NS, 0.5, 0, SC_NS, true);
```

Avoid clocks altogether for fast models!

# Dynamic Processes



THURSDAY IS  
TRAINING DAY

```
#define SC_INCLUDE_DYNAMIC_PROCESSES
```

PoC sim

```
void global_func()
{
    for (;;) {
        wait(10, SC_NS);
        ...
    }
}
```

```
void static_thread()
{
    wait(20, SC_NS);
    ...
    sc_spawn( &global_func );
    ...
}
```

```
SC_THREAD(static_thread);
```

# Spawning Member Functions



THURSDAY IS  
TRAINING DAY

```
struct Mod: sc_module
{
    SC_CTOR(Mod)
    {
        SC_THREAD(T);
        sc_spawn( sc_bind( &Mod::proc, this) );
    }

    void T()
    {
        sc_spawn( sc_bind( &Mod::proc, this ) );
    }

    void proc ()
    {
        ...
    }
};
```

Spawn a member function as a static process

Spawn a member function as a dynamic process

# Elaboration / Simulation Callbacks



THURSDAY IS  
TRAINING DAY



# Overriding the Callbacks



THURSDAY IS  
TRAINING DAY

- Called for each

- Module
- Primitive channel
- Port
- Export

```
SC_MODULE(Test)
{
    SC_CTOR(Test)  {}

    void before_end_of_elaboration() {...}
    void end_of_elaboration()           {...}
    void start_of_simulation()         {...}
    void end_of_simulation()          {...}
};
```

- Do nothing by default
- `before_end_of_elaboration()` may perform instantiation and port binding
- In PoC simulator, `end_of_simulation()` only called after `sc_stop()`

# The Scheduler in Detail



# Kinds of Channel



THURSDAY IS  
TRAINING DAY

- Primitive channels
  - Implement one or more interfaces
  - Derived from `sc_prim_channel`
  - Have access to the update phase of the scheduler
  - Examples - `sc_signal`, `sc_signal_resolved`, `sc_fifo`
- Hierarchical channels
  - Implement one or more interfaces
  - Derived from `sc_module`
  - Can instantiate ports, processes and modules
- Minimal channels - implement one or more interfaces

# Built-in Primitive Channels



THURSDAY IS  
TRAINING DAY

| Channel                                                               | Interfaces                                                                        | Events                                                              |
|-----------------------------------------------------------------------|-----------------------------------------------------------------------------------|---------------------------------------------------------------------|
| <code>sc_signal&lt;T&gt;</code>                                       | <code>sc_signal_in_if&lt;T&gt;</code><br><code>sc_signal inout_if&lt;T&gt;</code> | <code>value_changed_event()</code>                                  |
| <code>sc_buffer&lt;T&gt;</code>                                       | Same as <code>sc_signal</code>                                                    | On every write()                                                    |
| <code>sc_signal_resolved</code><br><code>sc_signal_rv&lt;W&gt;</code> | Same as<br><code>sc_signal&lt;sc_logic&gt;</code>                                 | Same as <code>sc_signal</code>                                      |
| <code>sc_clock</code>                                                 | Same as <code>sc_signal&lt;bool&gt;</code>                                        | posedge & negedge                                                   |
| <code>sc_fifo&lt;T&gt;</code>                                         | <code>sc_fifo_in_if&lt;T&gt;</code><br><code>sc_fifo_out_if&lt;T&gt;</code>       | <code>data_written_event()</code><br><code>data_read_event()</code> |
| <code>sc_mutex</code>                                                 | <code>sc_mutex_if</code>                                                          | n/a                                                                 |
| <code>sc_semaphore</code>                                             | <code>sc_semaphore_if</code>                                                      | n/a                                                                 |
| <code>sc_event_queue</code>                                           | n/a                                                                               | Every <code>notify()</code> invocation                              |

# Interface Method Call



THURSDAY IS  
TRAINING DAY



An interface declares a set of methods (pure virtual functions)

An interface is an abstract base class of the channel

A channel *implements* one or more interfaces (c.f. Java)

A module calls interface methods via a port

# Declare the Interface



THURSDAY IS  
TRAINING DAY

Important

```
#include "systemc"  
  
class queue_if : virtual public sc_core::sc_interface  
{  
public:  
    virtual void write(char c) = 0;  
    virtual char read() = 0;  
};
```

# Queue Channel Implementation



THURSDAY IS  
TRAINING DAY

```
#include "queue_if.h"
class Queue : public queue_if, public sc_core::sc_object
{
public:
    Queue(char* nm, int _sz)
        : sc_core::sc_object(nm), sz(_sz)
    { data = new char[sz]; w = r = n = 0; }

    void write(char c);
    char read();

private:
    char* data;
    int sz, w, r, n;
};
```

}

Implements interface methods

# Understanding Ports



THURSDAY IS  
TRAINING DAY



# Queue Ports



THURSDAY IS  
TRAINING DAY

```
class Producer : public sc_core::sc_module
{
public:
    sc_core::sc_port<queue_write_if> out;
    void do_writes();
    SC_CTOR(Producer)
    {
        SC_THREAD(do_writes);
    }
};
```



# Calling Methods via Ports



THURSDAY IS  
TRAINING DAY

```
#include <systemc>
#include "producer.h"
using namespace sc_core;

void Producer::do_writes()
{
    std::string txt = "Hallo World.";
    for (int i = 0; i < txt.size(); i++)
    {
        wait(SC_ZERO_TIME);
        out->write(txt[i]);
    }
}
```

```
out->write(txt[i]);
```

Note: -> overloaded

Interface Method Call

# Why Ports?



THURSDAY IS  
TRAINING DAY

- Ports allow modules to be independent of their environment
- Ports support elaboration-time checks (`register_port`, `end_of_elaboration`)
- Ports can have data members and member functions



# Exports



THURSDAY IS  
TRAINING DAY



Copyright © 2014-2015 by Doulos Ltd



## The SystemC Language



THURSDAY IS  
TRAINING DAY

- Introduction to SystemC
- Core Concepts and Syntax
- Bus Modeling
  - Master and slave interfaces
  - Blocking versus non-blocking
  - Multiports
- Odds and Ends



# Example Bus Model



THURSDAY IS  
TRAINING DAY

Multiple bus masters (modules), shared bus (channel), multiple slaves (channels)  
Bus arbitration and memory mapping built into the bus



# Master Interface Definition



```
class master_if : virtual public sc_interface
{
public:
    virtual void write(sc_uint<8> address, sc_uint<12> data,
                      int id) = 0;
    virtual void read (sc_uint<8> address, sc_uint<12> &data,
                      int id) = 0;
};
```

# Slave Interface Definition



THURSDAY IS  
TRAINING DAY



```
class slave_if : virtual public sc_interface
{
public:
    virtual void slave_write(sc_uint<8> address, sc_uint<12> data) = 0;
    virtual void slave_read (sc_uint<8> address, sc_uint<12> &data) = 0;
    virtual void get_map(unsigned int &start, unsigned int &size)      = 0;
};
```

Memory map managed within bus channel

# Master Write and Read



THURSDAY IS  
TRAINING DAY



Runs in the context of the caller

```
void Bus::write(sc_uint<8> address, sc_uint<12> data, int id)
{
    request[id] = true;           // request access to the bus
    wait(proceed[id]);          // wait for permission
    request[id] = false;         // clear the flag
    slave_port[find_port(address)]->slave_write(address, data);
}
```

Array of flags



Array of events



# Blocking and Non-blocking Calls



THURSDAY IS  
TRAINING DAY

An Interface Method Call runs in the context of the caller

Important!

ASI terminology:

- A *blocking* method may call wait
- A *blocking* method must be called from a thread process
  
- A *non-blocking* method must not call wait
- A *non-blocking* method may be called from a thread or method process
- Naming convention nb\_\*

# Bus Controller Process



THURSDAY IS  
TRAINING DAY

```
void Bus::control_bus()
{
    int highest;
    for (;;)
    {
        wait(clock->posedge_event());

        // Pick out a master that's made a request
        highest = -1;
        for (int i = 0; i < n_masters; i++)
            if (request[i])
                highest = i;

        // Notify the master with the highest id
        if (highest > -1)
            proceed[highest].notify();
    }
}
```

## The SystemC Language



THURSDAY IS  
TRAINING DAY

- [Introduction to SystemC](#)
- [Modules, Processes and Ports](#)
- [Bus Modeling](#)
- Odds and Ends



# Assignments in Multiple Processes



```
SC_CTOR(Test)
{
    SC_THREAD(p1);
    SC_THREAD(p2);
}

sc_logic v;

void p1()
{
    v = '0';
}

void p2()
{
    v = '1';
}
```

```
SC_CTOR(Test)
{
    SC_THREAD(p1);
    SC_THREAD(p2);
}

sc_signal<sc_logic> s;

void p1()
{
    s.write(sc_logic('0'));
}

void p2()
{
    s.write(SC_LOGIC_1);
}
```

Cast necessary

Built-in const

- In either case the value is indeterminate ('0' or '1')
- Indeed, writing a signal from > 1 process gives a run-time error

# Bus Resolution



THURSDAY IS  
TRAINING DAY

```
SC_CTOR(Test)
{
    SC_THREAD(p1);
    SC_THREAD(p2);
}

sc_signal_resolved S;
```

```
void p1()
{
    wait(10, SC_NS);
    S = sc_logic('0');
}

void p2()
{
    wait(20, SC_NS);
    S = SC_LOGIC_1;
    wait(10, SC_NS);
    S = SC_LOGIC_Z;
}
```

| Resolved | X | 0 | 1 | Z |
|----------|---|---|---|---|
| X        | X | X | X | X |
| 0        | X | 0 | X | 0 |
| 1        | X | X | 1 | 1 |
| Z        | X | 0 | 1 | Z |

| Time  | p1  | p2  | S   |
|-------|-----|-----|-----|
| 0 ns  | -   | -   | 'X' |
| 10 ns | '0' | -   | '0' |
| 20 ns | '0' | '1' | 'X' |
| 30 ns | '0' | 'Z' | '0' |

# Multiple Bindings



THURSDAY IS  
TRAINING DAY



Unsynchronised access to  
shared memory

sc\_signal\_resolved



`p[0] ->method();  
p[1] ->method();`

Array ports

# Resolved Channels and Ports



THURSDAY IS  
TRAINING DAY

## Resolved Channels

```
class sc_signal_resolved
: public sc_signal<sc_logic>
{ ... }
```

## Ports

```
sc_in<>
sc_inout<>
sc_out<>
```

```
sc_in_resolved
sc_inout_resolved
sc_out_resolved
```

```
template <int W>
class sc_signal_rv
: public sc_signal<sc_lv<W> >
{ ... }
```

```
sc_in<>
sc_inout<>
sc_out<>
```

```
sc_in_rv<>
sc_inout_rv<>
sc_out_rv<>
```

Can only bind to resolved signals

# Event Methods



THURSDAY IS  
TRAINING DAY

```
sc_in<bool> clock;
```

Event finder needed - port not yet bound

```
sensitive << clock.pos();
```

```
sc_event_finder& pos() const;
```

```
if (clock.posedge()) ...
```

```
bool posedge() const;
```

```
wait(clock.posedge_event());
```

```
const sc_event& posedge_event() const;
```

| Return type      | Method                | bool and sc_logic |                 |
|------------------|-----------------------|-------------------|-----------------|
| sc_event_finder& | value_changed()       | pos()             | neg()           |
| bool             | event()               | posedge()         | negedge()       |
| sc_event&        | value_changed_event() | posedge_event()   | negedge_event() |

# Accessing the command line



THURSDAY IS  
TRAINING DAY

- Not always able to access command-line from sc\_main
- sc\_argc() and sc\_argv()[] provide universal access

```
for (int i = 1; i < sc_argc(); ++i) {  
    string arg(sc_argv()[i]);  
    if (arg == "-flag") {  
        flag = true;  
    } else if (arg == "-num"  
               && i+1 < sc_argc()) {  
        number = atoi(sc_argv()[++i]);  
    }  
}
```

- Alternatives include
  - read from file
  - cstr = getenv("ENVIRONMENT\_VARIABLE"); //may be nullptr

# Debugging



THURSDAY IS  
TRAINING DAY

- Stream I/O: operator<< overloaded for built-in types

```
cout << "data=" << data << endl;
```

- Error reporting

```
SC_REPORT_INFO("msg_type", "message");
```

- Writing trace files for waveform viewers

```
sc_trace(trace_file, data, "data");
```

- Software debuggers and EDA tools

```
ddd run.x
```

# Error Reporting



THURSDAY IS  
TRAINING DAY

```
SC_REPORT_INFO    ("msg_type", "msg") ;  
  
SC_REPORT_WARNING ("msg_type", "msg") ;  
  
SC_REPORT_ERROR   ("msg_type", "msg") ;  
  
SC_REPORT_FATAL   ("msg_type", "msg") ;
```

Prints msg\_type + msg

also file, line, process, time

also throws exception

calls abort()

# Handling Reports



THURSDAY IS  
TRAINING DAY

- Use the macros

```
SC_REPORT_WARNING("", "Just a friendly warning");  
SC_REPORT_ERROR ("Me", "Big trouble");  
SC_REPORT_FATAL ("Me", "You're dead");
```

Message itself

Message type = origin of message

- Customize the report handler

```
sc_report_handler::set_actions(SC_WARNING, SC_DO NOTHING);  
sc_report_handler::set_actions("Me", SC_DISPLAY);  
sc_report_handler::set_actions("Me", SC_WARNING, SC_LOG);  
sc_report_handler::set_log_file_name("log.txt");  
sc_report_handler::stop_after(SC_WARNING, 5);
```



# Message and Message Type

```
#include <sstream>  
  
...  
  
ostringstream oss;  
  
oss << "Big trouble, address = "  
     << addr << " data = " << data << ends;  
  
SC_REPORT_ERROR("/DOULOS/SUPERBUS", oss.str().c_str());
```

Message type = hierarchical vendor string

- e.g. report from Proof-of-Concept simulator

```
Info: (I804) /IEEE_Std_1666/deprecated:  
sc_start(double) deprecated, use sc_start(sc_time) or sc_start()
```

- Suppress all such reports

```
sc_report_handler::set_actions("/IEEE_Std_1666/deprecated",  
                               SC_DO_NOTHING);
```



# Actions and Severity

## Actions

```
SC_UNSPECIFIED  
SC_DO_NOTHING  
SC_THROW  
SC_LOG  
SC_DISPLAY  
SC_CACHE_REPORT  
SC_INTERRUPT  
SC_STOP  
SC_ABORT
```

## Behaviour of the default handler

No action  
No action but inhibit lower priority actions  
Throw exception  
Write message to log file  
Write message to standard output  
Cache the report  
Call `sc_interrupt_here()`  
Call `sc_stop()`  
Call `abort()`

## Default actions

Each action is a bit mask – can be OR'd

```
SC_INFO      : SC_LOG | SC_DISPLAY  
SC_WARNING: SC_LOG | SC_DISPLAY  
SC_ERROR     : SC_LOG | SC_CACHE_REPORT | SC_THROW  
SC_FATAL     : SC_LOG | SC_DISPLAY | SC_CACHE_REPORT | SC_ABORT
```

# Accellera Systems Initiative



THURSDAY IS  
TRAINING DAY

## Current Accellera Board Members

- AMD
- ARM
- Cadence Design Systems Inc
- **Freescale Semiconductor**
- Intel Corporation
- Mentor Graphics
- NXP Semiconductors
- Qualcomm Inc
- Renesas Mobile
- ST Microelectronics
- Synopsys
- Texas Instruments

Website <http://www.accellera.org>

## European and North American SystemC Users Groups (and others)

- <http://www-ti.informatik.uni-tuebingen.de/~systemc>
- <http://www.nascug.org>

# For More FREE Information



THURSDAY IS  
TRAINING DAY

- IEEE 1666

[standards.ieee.org/getieee/1666/download/1666-2011.pdf](http://standards.ieee.org/getieee/1666/download/1666-2011.pdf)

- ASI SystemC 2.3.1

[www.accellera.org](http://www.accellera.org)

- On-line tutorials

[www.doulos.com/knowhow/systemc](http://www.doulos.com/knowhow/systemc)

### Hardware Design

» VHDL   » Verilog   » SystemVerilog

### Embedded Systems and ARM

» C   » C++   » UML   » RTOS   » Linux   » Yocto  
» ARM Cortex   » Python   » Android   » OpenCL   » OpenGL

### ESL & Verification

» SystemC   » TLM-2.0   » SystemVerilog  
» OVM/VMM/UVM   » Perl   » Tcl/Tk