authors: Bobby R. Bruce
last edited: 2024-11-15 22:30:44 +0000

Building a “Hello World” example with the gem5 standard library

In this tutorial we will cover how to create a very basic simulation using gem5 components. This simulation will setup a system consisting of a single-core processor, running in Atomic mode, connected directly to main memory with no caches, I/O, or other components. The system will run an X86 binary in syscall emulation (SE) mode. The binary will be obtained from gem5-resources and which will print a “Hello World!” string to stdout upon execution.

To start we must compile gem5 to simulate the X86 ISA:

# In the root of the gem5 directory
scons build/X86/gem5.opt -j <number of threads>

Then a new Python file should be created (we will refer to this as hello-world.py going forward). The first lines in this file should be the needed imports:

from gem5.components.boards.simple_board import SimpleBoard
from gem5.components.cachehierarchies.classic.no_cache import NoCache
from gem5.components.memory.single_channel import SingleChannelDDR3_1600
from gem5.components.processors.simple_processor import SimpleProcessor
from gem5.components.processors.cpu_types import CPUTypes
from gem5.resources.resource import Resource
from gem5.simulate.simulator import Simulator

All these libraries are included inside the compiled gem5 binary. Therefore, you will not need to obtain them from elsewhere. from gem5. indicates we are importing from the gem5 standard library, and the lines starting with from gem5.components are importing components from the gem5 components package. The from gem5.resources line means we are importing from the resources package, and from gem5.simulate, the simulate package. All these packages, components, resources, and simulate are part of the gem5 standard library.

Next we begin specifying the system. The gem5 library requires the user to specify four main components: the board, the cache hierarchy, the memory system, and the processor.

Let’s start with the cache hierarchy:

cache_hierarchy = NoCache()

Here we are using NoCache(). This means, for our system, we are stating there is no cache hierarchy (i.e., no caches). In the gem5 library the cache hierarchy is a broad term for anything that exists between the processor cores and main memory. Here we are stating the processor is connected directly to main memory.

Next we declare the memory system:

memory = SingleChannelDDR3_1600("1GiB")

There exists many memory components to choose from within gem5.components.memory. Here we are using a single-channel DDR3 1600, and setting its size to 1 GiB. It should be noted that setting the size here is technically optional. If not set, the SingleChannelDDR3_1600 will default to 8 GiB.

Then we consider the processor:

processor = SimpleProcessor(cpu_type=CPUTypes.ATOMIC, num_cores=1)

A processor in gem5.components is an object which contains a number of gem5 CPU cores, of a particular or varying type (ATOMIC, TIMING, KVM, O3, etc.). The SimpleProcessor used in this example is a processor where all the CPU Cores are of an identical type. It requires two arguments: the cpu_type, which we set to ATOMIC, and num_cores, the number of cores, which we set to one.

Finally we specify which board we are using:

board = SimpleBoard(
    clk_freq="3GHz",
    processor=processor,
    memory=memory,
    cache_hierarchy=cache_hierarchy,
)

While the constructor of each board may vary, they will typically require the user to specify the processor, memory system, and cache hierarchy, as well as the clock frequency to use. In this example we use the SimpleBoard. The SimpleBoard is a very basic system with no I/O which only supports SE-mode and can only work with “classic” cache hierarchies.

At this point in the script we have specified everything we require to simulate our system. Of course, in order to run a meaningful simulation, we must specify a workload for this system to run. To do so we add the following lines:

binary = Resource("x86-hello64-static")
board.set_se_binary_workload(binary)

The Resource class takes a string which specifies which resource, from gem5-resources, is to be obtained for the simulation. All the gem5 resources can be found on the gem5 Resources website.

If the resource is not present on the host system it’ll be automatically downloaded. In this example we are going to use the x86-hello-64-static resource; an x86, 64-bit, statically compiled binary which will print “Hello World!” to stdout. After specifying the resource we set the workload via the board’s set_se_binary_workload function. As the name suggests set_se_binary_workload is a function used to set a binary to be executed in Syscall Execution mode.

This is all that is required to setup your simulation. From this you simply need to construct and run the Simulator:

simulator = Simulator(board=board)
simulator.run()

It should also be noted that the Simulator module is still in a beta state, so its APIs may change upon the next release.

As a recap, your script should look like the following:

from gem5.components.boards.simple_board import SimpleBoard
from gem5.components.cachehierarchies.classic.no_cache import NoCache
from gem5.components.memory.single_channel import SingleChannelDDR3_1600
from gem5.components.processors.simple_processor import SimpleProcessor
from gem5.components.processors.cpu_types import CPUTypes
from gem5.resources.resource import Resource
from gem5.simulate.simulator import Simulator


# Obtain the components.
cache_hierarchy = NoCache()
memory = SingleChannelDDR3_1600("1GiB")
processor = SimpleProcessor(cpu_type=CPUTypes.ATOMIC, num_cores=1)

#Add them to the board.
board = SimpleBoard(
    clk_freq="3GHz",
    processor=processor,
    memory=memory,
    cache_hierarchy=cache_hierarchy,
)

# Set the workload.
binary = Resource("x86-hello64-static")
board.set_se_binary_workload(binary)

# Setup the Simulator and run the simulation.
simulator = Simulator(board=board)
simulator.run()

It can then be executed with:

./build/X86/gem5.opt hello-world.py

If setup correctly, the output will look something like:

...

warn: The simulate package is still in a beta state. The gem5 project does not guarantee the APIs within this package will remain consistent across upcoming releases.
Global frequency set at 1000000000000 ticks per second
build/X86/mem/mem_interface.cc:791: warn: DRAM device capacity (8192 Mbytes) does not match the address range assigned (1024 Mbytes)
0: board.remote_gdb: listening for remote gdb on port 7000
build/X86/sim/simulate.cc:194: info: Entering event queue @ 0.  Starting simulation...
build/X86/sim/syscall_emul.hh:1014: warn: readlink() called on '/proc/self/exe' may yield unexpected results in various settings.
      Returning '/scr/bbruce/.cache/gem5/x86-hello64-static'
build/X86/sim/mem_state.cc:443: info: Increasing stack size by one page.
Hello world!

It should be obvious from this point that a board’s parameters may be altered to test other designs. For example, if we want to test a TIMING CPU setup we’d change our processor to:

processor = SimpleProcessor(cpu_type=CPUTypes.TIMING, num_cores=1)

This is all that is required. The gem5 standard library will reconfigure the design as is necessary.

As another example, consider swapping out a component for another. In this design we decided on NoCache but we could use another classic cache hierarchy, such as PrivateL1CacheHierarchy. To do so we’d change our cache_hierarchy parameter:

# We import the cache hierarchy we want.
from gem5.components.cachehierarchies.classic.private_l1_cache_hierarchy import PrivateL1CacheHierarchy

...

# Then set it.
cache_hierarchy = PrivateL1CacheHierarchy(l1d_size="32kB", l1i_size="32kB")

Note here that PrivateL1CacheHierarchy requires the user to specify the L1 data and instruction cache sizes to be constructed. No other part of the design need change. The gem5 standard library will incorporate the cache hierarchy as required.

To recap on what was learned in this tutorial: