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

Gvsoc ci #142

Draft
wants to merge 5 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all 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
26 changes: 26 additions & 0 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -128,6 +128,32 @@ jobs:
run: |
make SIM_DIR=./runs/simple annotate -j

##############################################
# Simulate SW on Snitch Cluster w/ GVSOC #
##############################################

sw-snitch-cluster-gvsoc:
name: Simulate SW on Snitch Cluster w/ GVSOC
runs-on: ubuntu-22.04
needs: build-docker
if: >
github.event_name != 'pull_request' ||
github.event.pull_request.head.repo.full_name != github.repository
container:
image: ghcr.io/pulp-platform/snitch_cluster:${{ github.ref_name }}
steps:
- uses: actions/checkout@v2
with:
submodules: 'recursive'
- name: Build Software
working-directory: target/snitch_cluster
run: |
make CFG_OVERRIDE=cfg/github-ci.hjson OPENOCD_SEMIHOSTING=ON sw
- name: Run Tests
working-directory: target/snitch_cluster
run: |
GVSOC_TARGET=pulp.snitch.snitch_cluster_single && ./util/run.py sw/run.yaml --simulator gvsoc -j

# #########################################
# # Build SW on Snitch Cluster w/ Banshee #
# #########################################
Expand Down
50 changes: 30 additions & 20 deletions sw/blas/axpy/src/axpy.h
Original file line number Diff line number Diff line change
Expand Up @@ -43,24 +43,27 @@ static inline void axpy_opt(uint32_t n, double a, double *x, double *y,
int frac = n / snrt_cluster_compute_core_num();
int offset = core_idx;

snrt_ssr_loop_1d(SNRT_SSR_DM_ALL, frac,
snrt_cluster_compute_core_num() * sizeof(double));

snrt_ssr_read(SNRT_SSR_DM0, SNRT_SSR_1D, x + offset);
snrt_ssr_read(SNRT_SSR_DM1, SNRT_SSR_1D, y + offset);
snrt_ssr_write(SNRT_SSR_DM2, SNRT_SSR_1D, z + offset);

snrt_ssr_enable();

asm volatile(
"frep.o %[n_frep], 1, 0, 0 \n"
"fmadd.d ft2, %[a], ft0, ft1\n"
:
: [ n_frep ] "r"(frac - 1), [ a ] "f"(a)
: "ft0", "ft1", "ft2", "memory");

snrt_fpu_fence();
snrt_ssr_disable();
for (int i=0; i<1; i++)
{
snrt_ssr_loop_1d(SNRT_SSR_DM_ALL, frac,
snrt_cluster_compute_core_num() * sizeof(double));

snrt_ssr_read(SNRT_SSR_DM0, SNRT_SSR_1D, x + offset);
snrt_ssr_read(SNRT_SSR_DM1, SNRT_SSR_1D, y + offset);
snrt_ssr_write(SNRT_SSR_DM2, SNRT_SSR_1D, z + offset);

snrt_ssr_enable();

asm volatile(
"frep.o %[n_frep], 1, 0, 0 \n"
"fmadd.d ft2, %[a], ft0, ft1\n"
:
: [ n_frep ] "r"(frac - 1), [ a ] "f"(a)
: "ft0", "ft1", "ft2", "memory");

snrt_fpu_fence();
snrt_ssr_disable();
}
}

static inline void axpy_job(axpy_args_t *args) {
Expand Down Expand Up @@ -113,6 +116,8 @@ static inline void axpy_job(axpy_args_t *args) {
iterations = args->n_tiles;
if (DOUBLE_BUFFER) iterations += 2;

uint32_t cycles;

// Iterate over all tiles
for (i = 0; i < iterations; i++) {
if (snrt_is_dm_core()) {
Expand Down Expand Up @@ -167,7 +172,7 @@ static inline void axpy_job(axpy_args_t *args) {
if (!DOUBLE_BUFFER) snrt_cluster_hw_barrier();

if (!DOUBLE_BUFFER || (i > 0 && i < (args->n_tiles + 1))) {
snrt_mcycle();
uint32_t start_cycle = snrt_mcycle();

// Compute tile and buffer indices
i_compute = DOUBLE_BUFFER ? i - 1 : i;
Expand All @@ -178,7 +183,9 @@ static inline void axpy_job(axpy_args_t *args) {
fp(frac, args->a, local_x[buff_idx], local_y[buff_idx],
local_z[buff_idx]);

snrt_mcycle();
uint32_t end_cycle = snrt_mcycle();

cycles = end_cycle - start_cycle;
}

// Additional barrier required when not double buffering
Expand All @@ -188,4 +195,7 @@ static inline void axpy_job(axpy_args_t *args) {
// Synchronize cores after every iteration
snrt_cluster_hw_barrier();
}


printf("Cycles %d\n", cycles);
}
17 changes: 16 additions & 1 deletion sw/tests/simple.c
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,19 @@
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0

int main() { return 0; }
#include "snrt.h"

uint32_t buffer[32];

int main() {
if (snrt_global_core_idx() != 8) return 0; // only DMA core
uint32_t buffer_src[32], buffer_dst[32];
for (int i = 0; i < 100; i++)
{
snrt_mcycle();
snrt_dma_start_1d_wideptr(buffer_src, buffer_dst, 1);
snrt_dma_wait_all();
snrt_mcycle();
}
return 0;
}
2 changes: 2 additions & 0 deletions target/common/common.mk
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,8 @@ SED_SRCS := sed -e ${MATCH_END} -e ${MATCH_BGN} -e ${MATCH_DEF}

COMMON_BENDER_FLAGS += -t rtl

GVSOC_BUILDDIR ?= work-gvsoc

VSIM_BENDER += $(COMMON_BENDER_FLAGS) -t test -t simulation -t vsim
VSIM_SOURCES = $(shell ${BENDER} script flist-plus ${VSIM_BENDER} | ${SED_SRCS})
VSIM_BUILDDIR ?= work-vsim
Expand Down
21 changes: 21 additions & 0 deletions target/common/gvsoc.mk
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
# Copyright 2024 ETH Zurich and University of Bologna.
# Licensed under the Apache License, Version 2.0, see LICENSE for details.
# SPDX-License-Identifier: Apache-2.0

$(BIN_DIR)/$(TARGET).gvsoc:
@echo "#!/bin/bash" > $@
@echo 'binary=$$(realpath $$1)' >> $@
@echo 'echo $$binary > .rtlbinary' >> $@
@echo 'path="$$(dirname "$$(dirname "$$(readlink -f "$${BASH_SOURCE[0]}")")")"' >> $@
@echo 'if [ -z "$$GVSOC_TARGET" ]; then' >> $@
@echo ' GVSOC_TARGET=snitch' >> $@
@echo 'fi' >> $@
@echo 'gvsoc --target=$${GVSOC_TARGET} --binary $$binary \
--control-script=$${path}/../../util/sim/gvsoc_control.py $$2 run' >> $@
@chmod +x $@

.PHONY: clean-gvsoc
clean-gvsoc:
rm -rf $(BIN_DIR)/$(TARGET).gvsoc $(GVSOC_BUILDDIR)

clean: clean-gvsoc
6 changes: 6 additions & 0 deletions target/snitch_cluster/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -215,6 +215,12 @@ $(BIN_DIR)/$(TARGET).vcs: ${VCS_SOURCES} ${TB_SRCS} $(TB_CC_SOURCES) $(RTL_CC_SO
-assert disable_cover -override_timescale=1ns/1ps -full64 tb_bin $(TB_CC_SOURCES) $(RTL_CC_SOURCES) \
-CFLAGS "$(TB_CC_FLAGS)" -LDFLAGS "-L${FESVR}/lib" -lfesvr

#########
# GVSOC #
#########

include $(ROOT)/target/common/gvsoc.mk

########
# Util #
########
Expand Down
41 changes: 21 additions & 20 deletions target/snitch_cluster/sw/run.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -6,53 +6,53 @@ runs:
- elf: tests/build/alias.elf
simulators: [vsim, vcs, verilator] # banshee does not model alias regions
- elf: tests/build/atomics.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x4
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x4
- elf: tests/build/barrier.elf
- elf: tests/build/data_mover.elf
- elf: tests/build/dma_empty_transfer.elf
- elf: tests/build/dma_simple.elf
- elf: tests/build/event_unit.elf
- elf: tests/build/fence_i.elf
- elf: tests/build/fp8_comparison_scalar.elf
simulators: [vsim, vcs, verilator] # banshee fails with segfault
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with segfault
- elf: tests/build/fp8_comparison_vector.elf
simulators: [vsim, vcs, verilator] # banshee fails with segfault
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with segfault
- elf: tests/build/fp8_computation_scalar.elf
simulators: [vsim, vcs, verilator] # banshee fails with JIT issue
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with JIT issue
- elf: tests/build/fp8_computation_vector.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x6
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x6
- elf: tests/build/fp8alt_comparison_scalar.elf
simulators: [vsim, vcs, verilator] # banshee fails with segfault
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with segfault
- elf: tests/build/fp8alt_comparison_vector.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x10
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x10
- elf: tests/build/fp8alt_computation_scalar.elf
simulators: [vsim, vcs, verilator] # banshee fails with JIT issue
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with JIT issue
- elf: tests/build/fp8alt_computation_vector.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x12
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x12
- elf: tests/build/fp16_comparison_scalar.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x10
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x10
- elf: tests/build/fp16_comparison_vector.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x10
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x10
- elf: tests/build/fp16_computation_scalar.elf
simulators: [vsim, vcs, verilator] # banshee fails with JIT issue
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with JIT issue
- elf: tests/build/fp16_computation_vector.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x6
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x6
- elf: tests/build/fp16alt_comparison_scalar.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x10
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x10
- elf: tests/build/fp16alt_comparison_vector.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x10
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x10
- elf: tests/build/fp16alt_computation_scalar.elf
simulators: [vsim, vcs, verilator] # banshee fails with JIT issue
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with JIT issue
- elf: tests/build/fp16alt_computation_vector.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x16
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x16
- elf: tests/build/fp32_comparison_scalar.elf
- elf: tests/build/fp32_comparison_vector.elf
- elf: tests/build/fp32_computation_scalar.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x2
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x2
- elf: tests/build/fp32_computation_vector.elf
simulators: [vsim, vcs, verilator] # banshee fails with exit code 0x2
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with exit code 0x2
- elf: tests/build/fp32_conversions_scalar.elf
simulators: [vsim, vcs, verilator] # banshee fails with illegal instruction
simulators: [vsim, vcs, verilator, gvsoc] # banshee fails with illegal instruction
# - elf: tests/build/fp64_conversions_scalar.elf
# simulators: [vsim, vcs, verilator]
- elf: tests/build/interrupt_local.elf
Expand All @@ -72,6 +72,7 @@ runs:
- elf: tests/build/non_null_exitcode.elf
retcode: 56
- elf: tests/build/caq.elf
simulators: [vsim, vcs, verilator, banshee] # GVSOC does not model caq
- elf: tests/build/caq_frep.elf
simulators: [vsim, vcs, verilator] # banshee does not model FREP timing
- elf: apps/blas/axpy/build/axpy.elf
Expand Down
5 changes: 3 additions & 2 deletions target/snitch_cluster/util/run.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,15 +15,16 @@

sys.path.append(str(Path(__file__).parent / '../../../util/sim'))
import sim_utils # noqa: E402
from Simulator import QuestaSimulator, VCSSimulator, VerilatorSimulator, \
from Simulator import QuestaSimulator, VCSSimulator, VerilatorSimulator, GvsocSimulator, \
BansheeSimulator # noqa: E402


SIMULATORS = {
'vsim': QuestaSimulator(Path(__file__).parent.resolve() / '../bin/snitch_cluster.vsim'),
'vcs': VCSSimulator(Path(__file__).parent.resolve() / '../bin/snitch_cluster.vcs'),
'verilator': VerilatorSimulator(Path(__file__).parent.resolve() / '../bin/snitch_cluster.vlt'),
'banshee': BansheeSimulator(Path(__file__).parent.resolve() / '../src/banshee.yaml')
'banshee': BansheeSimulator(Path(__file__).parent.resolve() / '../src/banshee.yaml'),
'gvsoc': GvsocSimulator(Path(__file__).parent.resolve() / '../bin/snitch_cluster.gvsoc')
}


Expand Down
30 changes: 30 additions & 0 deletions util/sim/Simulation.py
Original file line number Diff line number Diff line change
Expand Up @@ -267,6 +267,36 @@ def get_cpu_time(self):
return hours*3600 + minutes*60 + seconds


class GvsocSimulation(Simulation):
"""A functional simulation running on GVSOC."""

def __init__(self, sim_bin=None, cmd=None, **kwargs):
super().__init__(**kwargs)

if cmd is None:
self.cmd = ['gvsoc', '--target', os.environ.get('GVSOC_TARGET'), '--binary',
str(self.elf), 'run']
else:
self.dynamic_args = {
'sim_bin': str(sim_bin),
'elf': str(self.elf),
'run_dir': str(self.run_dir)
}
self.cmd = [Template(arg).render(**self.dynamic_args) for arg in cmd]
self.cmd.append('--simulator=gvsoc')

def successful(self):
"""Return whether the simulation was successful."""
# On GVSOC, OpenOCD semi-hosting is used which can just report 0 or 1
actual_retcode = self.get_retcode()
if actual_retcode is not None:
if self.expected_retcode != 0:
return int(actual_retcode) != 0
else:
return int(actual_retcode) == 0
else:
return False

class VCSSimulation(QuestaVCSSimulation):
"""An RTL simulation running on VCS."""

Expand Down
19 changes: 18 additions & 1 deletion util/sim/Simulator.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
#
# Luca Colagrande <[email protected]>

from Simulation import QuestaSimulation, VCSSimulation, VerilatorSimulation, BansheeSimulation
from Simulation import QuestaSimulation, VCSSimulation, VerilatorSimulation, BansheeSimulation, GvsocSimulation


class Simulator(object):
Expand Down Expand Up @@ -96,6 +96,23 @@ def get_simulation(self, test):
cmd=cmd
)

class GvsocSimulator(RTLSimulator):
"""Gvsoc simulator

An [RTL simulator][Simulator.RTLSimulator], identified by the name
`vsim`, tailored to the creation of
[Gvsoc simulations][Simulation.GvsocSimulation].
"""

def __init__(self, binary):
"""Constructor for the GvsocSimulator class.

Arguments:
binary: The Gvsoc simulation binary.
"""
super().__init__(binary, name='gvsoc', simulation_cls=GvsocSimulation)
Comment on lines +99 to +113
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why does it inherit from the RTLSimulator class?




class VCSSimulator(RTLSimulator):
"""VCS simulator
Expand Down
10 changes: 8 additions & 2 deletions util/sim/SnitchSim.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,11 +25,12 @@

class SnitchSim:

def __init__(self, sim_bin: str, snitch_bin: str, log: str = None):
def __init__(self, sim_bin: str, snitch_bin: str, simulator: str = None, log: str = None):
self.sim_bin = sim_bin
self.snitch_bin = snitch_bin
self.sim = None
self.tmpdir = None
self.simulator = simulator
self.log = open(log, 'w+') if log else log

def start(self):
Expand All @@ -40,10 +41,15 @@ def start(self):
rx_fd = os.path.join(self.tmpdir.name, 'rx')
os.mkfifo(rx_fd)
# Start simulator process
ipc_arg = f'--ipc,{tx_fd},{rx_fd}'
if self.simulator == 'gvsoc':
ipc_arg = f'--ipc {tx_fd},{rx_fd}'
else:
ipc_arg = f'--ipc,{tx_fd},{rx_fd}'

self.sim = subprocess.Popen([self.sim_bin, self.snitch_bin, ipc_arg], stdout=self.log)
# Open FIFOs
self.tx = open(tx_fd, 'wb', buffering=0) # Unbuffered

self.rx = open(rx_fd, 'rb')
# Create thread to monitor simulation
self.stop_sim_monitor = threading.Event()
Expand Down
Loading
Loading