Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Documentation for SST #305

Merged
merged 6 commits into from
May 18, 2023
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Binary file added docs/sphinx/assets/sst_mem_model.png
jj16791 marked this conversation as resolved.
Show resolved Hide resolved
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
10 changes: 10 additions & 0 deletions docs/sphinx/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,16 @@ The Simulation Engine - SimEng
user/configuring_simeng
user/creating_binaries

.. toctree::
:maxdepth: 4
:caption: SST Integration
:hidden:

sst/index
sst/building_simeng_with_sst
sst/understanding_sst
sst/running_simeng_with_sst

SimEng is a framework for building modern, cycle-accurate processor simulators. Its goals are to be:

- Fast, typically 4-5X faster than gem5
Expand Down
57 changes: 57 additions & 0 deletions docs/sphinx/sst/building_simeng_with_sst.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
Building SimEng with SST
========================

Prerequisites
*************
In addition to CMake and a compiler that supports C++17, version **12.0.x** of SST-Core and SST-Elements are required.

.. warning::
Please ensure that while installing SST-Elements, the initial configuration step should include the
``--with-sst-core=<sst-core_install_directory>`` flag so that all components present in **SST-Elements** are registered with the **SST-Core**.
This is important because **SimEng** relies on components present in **SST-Elements**.
However, if these components aren't registered, **SST-Core** will fail to locate the components to use during simulation.

.. note::
If the path: ``<sst-core_install_directory>/bin`` is added to the ``$PATH`` environment variable then all sst executables will be
available globally. i.e. you will be to use the commands ``sst`` , ``sst-info``, ``sst-register`` and ``sst-config`` without having to navigate
to the SST install directory.

FinnWilkinson marked this conversation as resolved.
Show resolved Hide resolved
Build Steps
***********
Two flags have been added to SimEng's CMake configuration step to enable integration with SST:

.. code-block:: text

-DSIMENG_ENABLE_SST={ON, OFF} // Defaults to OFF
-DSST_INSTALL_DIR=<sst-core_install_directory> // Path to the SST-Core install location

The rest of the steps for building and installing SimEng with SST integration remain the same as a standalone SimEng :ref:`installation<Building_SimEng>` i.e the build step and the install step.

Validation
**********
A successful SST installation should also install the ``sst-info`` executable. This executable is used to print all the components registered with SST-Core which
can be used in an SST simulation. To verify a successful installation of SimEng with SST integration enabled, the command ``sst-info --libs=libsstsimeng``
should output:

.. code-block:: text

================================================================================
ELEMENT 0 = sstsimeng()
Num Components = 1
Component 0: simengcore
CATEGORY: PROCESSOR COMPONENT
NUM STATISTICS = 0
NUM PORTS = 0
NUM SUBCOMPONENT SLOTS = 0
NUM PARAMETERS = 5
PARAMETER 0 = config_path (Path to Simeng YAML config file (string)) []
PARAMETER 1 = executable_path (Path to executable binary to be run by SimEng (string)) []
PARAMETER 2 = executable_args (argument to be passed to the executable binary (string)) []
PARAMETER 3 = clock (Clock rate of the SST clock (string)) []
PARAMETER 4 = max_addr_memory (Maximum address that memory can access (int)) [<required>]
simengcore: Simeng core wrapper for SST
Using ELI version 0.9.0
Compiled on: Aug 25 2022 11:55:05, using file: <path_to_simeng>/sst/include/SimengCoreWrapper.hh
Num SubComponents = 0
Num Modules = 0
Num SSTPartitioners = 0
10 changes: 10 additions & 0 deletions docs/sphinx/sst/index.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
Overview
========

The following provides information on how to build, run, and configure SimEng with SST. SimEng uses SST for simulating the memory heirarchy.

* :doc:`Building SimEng with the SST <building_simeng_with_sst>`

* :doc:`Understanding SST <understanding_sst>`

* :doc:`Running SimEng with SST <running_simeng_with_sst>`
263 changes: 263 additions & 0 deletions docs/sphinx/sst/running_simeng_with_sst.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,263 @@
.. _Running_SimEng_W_SST:

Running SimEng with SST
========================

Setup
*****
During the installation stage of SimEng with the SST integration enabled, ``libsstsimeng.so`` is installed in the
``<path_to_simeng_install>/sst`` directory and ``libsimeng.so`` is installed in the ``<path_to_simeng_install>/lib`` directory at the install location.
To make sure SimEng is able to run successfully inside a SST simulation, the ``$LD_LIBRARY_PATH`` environment variable should include the
path to ``libsimeng.so`` i.e ``<path_to_simeng_install>/lib``. For the SST Simulation, SimEng also relies on components present inside **SST-Elements**,
so the path to all **SST-Elements** needs to be included also i.e ``<sst-elements_install_directory>/lib/sst-elements-library``.


Configuration for SST Simulation
********************************
SST provides a python module to allow interaction with the simulation build system. SST simulations are configured through a ``config.py`` python file. The SST
core python module is defined in CPython and is only available in the python interpreter launched within a running SST executable. SST parses the python file and
creates a component graph, which it then uses to instantiate and configure components, links, and subcomponents.

All SST config files have to use the following import:

.. code-block:: python

import sst
# or
from sst import *

The following links can be used to access SST documentation regarding `SST Python Classes <http://sst-simulator.org/SSTPages/SSTUserPythonClasses>`_ and
`Global Functions in SST Python Module <http://sst-simulator.org/SSTPages/SSTUserPythonGlobalFunctions>`_

Within this module, there are a number of available classes and global functions. The available classes are **Component**, **SubComponent**, **Link**,
**StatisticOutput**, and **StatisticGroup**. The global functions are divided between general functions and functions operating on or returning one of the
available classes mentioned.

Declaring and configuring SST::Component(s)
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Declaring components in the SST configuration files can be achieved by declaring an ``sst.Component`` class. This constructor takes in a unique identifier string
called ``name`` and an ``element_type`` string used to identify the SST component to dynamically load during runtime. The ``element_type`` string is very
important and has a specific format i.e ``<componentLibrary>.<(sub)componentName>``.

.. warning::

``(sub)componentName`` is different from ``name``. ``(sub)componentName`` is used to locate the actual SST component or subcomponent from the SST Element
registry, whereas ``name`` is a just unique identifier string which can be used in the codebase to manually locate and instantiate components.

The ``componentLibrary`` string is the name of the dynamic library which contains components with the ``lib`` prefix stripped. During runtime, SST adds the
``lib`` prefix to the ``componentLibrary`` string, dynamically loads the library, and uses the ``(sub)componentName`` string to index into the Element registry and
instantiate the component or subcomponent.

**Example:**
SimEng core is wrapped in a class called ``SimengCoreWrapper``. The class is then registered as a custom SST component with ``(sub)componentName = simengcore``. The
class is then compiled into a shared library with the name ``libsstsimeng``. Finally, using the ``sst-register`` executable, the ``libsstsimeng`` library and its
path are registered into SST. This is all done automatically in the build and install steps of SST Simeng integration.

``simengcore`` can now be in a SST simulation using the following ``element_type`` string:

.. code-block:: python

import sst
cpu = sst.Component("mycore", "sstsimeng.simengcore")

Adding parameters to the SST component can be done through the ``addParams`` method of the SST Python module Component class.

.. code-block:: python

import sst
cpu = sst.Component("mycore", "sstsimeng.simengcore")
cpu.addParams({
"clock": "1GHz",
...
})

.. note::

* To find out about the ``componentLibrary`` or ``(sub)componentNane`` you can use the ``sst-info`` command.
* The ``sst-info`` command can be used to find out about all libraries, components and subcomponents registered with the SST Core
* The ``sst-info <componentLibrary>`` command can be used to find out about all components and subcomponents in a library.
* The ``sst-info <componentLibrary>.<(sub)componentName>`` command can be used to get information about a specific component or subcomponent in a library.

* Examples:
* ``sst-info sstsimeng``
* ``sst-info sstsimeng.simengcore``

Declaring and Configuring SST::Link(s)
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Links are present as a class in a Python SST Module as well. There are two ways to connect SST components using links. For both approaches, links first need to
be instantiated using ``sst.Link``, which takes in a user-defined name as its argument.

Approach 1
''''''''''
For the first approach, the link needs to be instantiated and then, using the ``sst.Link.connect`` function, the components need to be connected. The connect
function takes in 2 tuples, of the same format: ``(component, port_name, port_latency)``. ``port_name`` is the name of the port defined by the component. This
can be found in the SST component’s documented ports with the ``sst-info`` command.

.. code-block:: python

component_1 = sst.Component(...) # Assuming component_1 has a port called data_port
component_2 = sst.Component(...) # Assuming component_2 has a port called data_port

link = sst.Link("mylink") # Passing a user-defined link name

# (component, port_name, port_latency)
link.connect((component_1,"data_port","10ns"), (component_2,"data_port","10ns"))

Approach 2
''''''''''
For the second approach, the addLink method of each component can be used to establish communication. The addLink method takes in 3 arguments: ``link``,
``port_name``, and ``port_latency``.

.. code-block:: python

component_1 = sst.Component(...) # Assuming component_1 has a port called data_port
component_2 = sst.Component(...) # Assuming component_2 has a port called data_port

link = sst.Link("mylink") # Passing a user-defined link name

component_1.addLink(link, "data_port", "10ns")
component_2.addLink(link, "data_port", "10ns")

Declaring and Configuring SST::SubComponent(s)
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
If a component has a SubComponent slot, then it can be filled using the ``component.setSubComponent`` method. This method takes in 3 arguments: ``slot_name`` ,
``element_type`` and ``slot_index``. The ``slot_index`` is the index in which the SubComponent should be inserted. This defaults to 0 and is not required if only
one SubComponent is being loaded into the specified slot. Each SubComponent must be loaded into a unique ``slot_index`` and some SubComponents will require the
indexes to be incremental. SubComponents can also be configured using the ``addParams`` method.

.. code-block:: python

import sst
# Memory controller from memHierarchy library
memctrl = sst.Component("memory", "memHierarchy.MemController")
memctrl.addParams({
...
})
# Memory controller has a subcomponent slot for the memory backend.
memory = memctrl.setSubComponent("backend", "memHierarchy.simpleMem")
memory.addParams({
...
})

SimEng SST Configuration
************************

Configuring the SimEngCore
~~~~~~~~~~~~~~~~~~~~~~~~~~
As discussed earlier, the SimEng core has been wrapped in the ``SimengCoreWrapper`` class and registered as a custom SST Component. Instantiating and using this
component is fundamental to running the simulation. The component has the following parameters:

* ``config_path``: Path to YAML configuration file needed by SimEng for configuration of the core microarchitecture under simulation.
* ``executable_path``: Path to the executable to run inside SimEng.
* ``executable_args``: Arguments provided to the executable.
FinnWilkinson marked this conversation as resolved.
Show resolved Hide resolved
* ``clock``: The frequency of clock ticking the SimEng Core e.g. 1GHz (S.I units accepted).
* ``max_addr_range``: Maximum address which can be accessed by SimEng.
* ``cache_line_width``: Width of the cache line (in bytes).

Configuring StandardInterface
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
After the ``simengcore`` has been instantiated, the ``StandardInterface`` has to be set into the ``memory`` slot of simengcore.

.. code-block:: python

iface = cpu.setSubComponent("memory", "memHierarchy.standardInterface")

Configuring the Cache
~~~~~~~~~~~~~~~~~~~~~~
Next, the L1 cache needs to be configured. All configuration parameters (and their documentation), ports and slots for the cache can be found using the
command ``sst-info memHierarchy.Cache``. The default cache parameters provided by the installation are defined below:

.. code-block:: python

l1cache = sst.Component("l1cache.mesi", "memHierarchy.Cache")
l1cache.addParams({
"access_latency_cycles" : "2",
"cache_frequency" : "2Ghz",
"replacement_policy" : "nmru",
"coherence_protocol" : "MESI",
"associativity" : "4",
"cache_line_size" : "64",
"debug" : 1,
"debug_level" : 1,
"L1" : "1",
"cache_size" : "200KiB",
})

Configuring the MemoryController and MemBackend
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
As discussed in the :ref:`Understanding SST<U_SST>` section, the ``MemoryController`` and the ``MemBackend`` are the last levels in the memory hierarchy. The ``MemBackend`` is a subcomponent of the
``MemoryController`` (``slot: backend``) and for the SimEng simulation the ``SimpleMem`` backend is used. Other backends are also supported.

All configuration parameters (and their documentation), ports, and slots for ``MemoryController`` and ``MemBackend`` can be found using the commands: ``sst-info memHierarchy.MemController``
and ``sst-info memHierarchy.simpleMem``. The default memory controller and backend parameters provided by the installation are defined below:

.. code-block:: python

# Memory controller
memctrl = sst.Component("memory", "memHierarchy.MemController")
memctrl.addParams({
"clock" : "1GHz",
"request_width" : "64",
"debug" : 1,
"debug_level" : 1,
"addr_range_end" : 1*1024*1024*1024-1,
})

# Memory model
memory = memctrl.setSubComponent("backend", "memHierarchy.simpleMem")
memory.addParams({
"access_time" : "1ns",
"mem_size" : "1GiB",
})

Connecting Components with links
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
In the last step, we need to connect all components defined above together so that they can communicate with each other and exchange memory requests. As mentioned
above, the ``high_network_0`` port of the cache needs to be connected to higher cache levels or the core. Since the SimEng core interfaces with the memory system
through ``StandardInterface``, the cache will be connected to it. The default core to memory hierarchy component link provided by the installation are defined
below:

.. code-block:: python

cpu_to_cache_link = sst.Link("link1")
cpu_to_cache_link.connect((iface, "port", "100ps"), (l1cache, "high_network_0", "100ps"))
# or
cpu_to_cache_link = sst.Link("link1")
iface.addLink(cpu_to_cache_link, "port", "100ps")
l1cache.addLink(cpu_to_cache_link, "high_network_0", "100ps")

Now that the SimEng core has been connected to the cache, the cache needs to connect to the lower levels of memory. For this configuration, it will be connected
to the memory controller. Otherwise it will be connected to lower levels of cache i.e. L2. As mentioned earlier, the cache connects to lower levels through the
``low_network_0`` port.

.. code-block:: python

l1_to_mem_link = sst.Link("link2")
l1_to_mem_link.connect((l1cache, "low_netowrk_0", "50ps"), (memctrl, "direct_link", "50ps"))
# or
l1_to_mem_link = sst.Link("link2")
l1cache.addLink(l1_to_mem_link, "low_network_0", "50ps")
memctrl.addLink(l1_to_mem_link, "direct_link", "50ps")

.. warning::

The SimEng core YAML configuration file defines ``stack_size`` and ``heap_size`` parameters which are used to determine the size of the process image.
In addition to these parameters, the process image size is also determined by the ELF binary header sections (which contain instructions and initialised data). To ensure
the simulation is very fast, SimEng internally initialises the process image as a large char array that can index the highest virtual address encountered in
the ELF binary header sections, eliminating the need for any address translation.

To make sure that the memory used by SST is consistent with the details mentioned above, the memory size of the backend i.e ``mem_size`` has been set to **2GiB**
in the default SST ``config.py`` file. The ``SimengCoreWrapper`` checks if the SST memory backend has been configured with enough memory to store the process
image. This check is done using the ``addr_range_end`` parameter of ``sstsimeng.simengcore``.

The user must also ensure that the maximum address accessible in the memory backend is consistent with ``addr_range_end`` parameter of the memory controller
i.e ``memHierarchy.MemController`` and ``max_addr_range`` parameter of SimEng core i.e ``sstsimeng.simengcore``.

.. note::
More examples of the SST ``config.py`` files are present in the **SST-Elements** code base, found `here <https://github.com/sstsimulator/sst-elements/tree/master/src/sst/elements/memHierarchy/tests>`_. Files starting with the prefix ``sdl`` contain different examples of memory hierarchy configurations which SST can simulate.

Running SST SimEng Simulation
*****************************
To run the simulation, navigate to the ``config.py`` file (the default configuration file can be found at the path ``<path-to-simeng-install>/sst/config``) and
use the command ``sst config.py`` to start the simulation.

Loading