Skip to content

Commit

Permalink
feat: add tooling for creating snapshots programmatically
Browse files Browse the repository at this point in the history
  • Loading branch information
renan061 committed Jul 1, 2024
1 parent 46438d4 commit a97dec4
Show file tree
Hide file tree
Showing 2 changed files with 116 additions and 53 deletions.
54 changes: 1 addition & 53 deletions build/Dockerfile
Original file line number Diff line number Diff line change
Expand Up @@ -103,58 +103,6 @@ RUN <<EOF
rm -rf /var/lib/apt/lists/*
EOF

# =============================================================================
# STAGE: emulator-devel
#
# - Install libarchive13 (setup -- required by xgenext2fs).
# - Install libcmt.
# - Install xgenext2fs.
# =============================================================================

FROM emulator as emulator-devel

ARG MACHINE_TOOLS_VERSION
ARG MACHINE_XGENEXT2FS_VERSION
ARG DEBIAN_FRONTEND=noninteractive

# Install libarchive13 (setup -- required by xgenext2fs).
RUN <<EOF
set -e
apt-get update
apt-get install -y --no-install-recommends libarchive13
EOF

# Install libcmt (cartesi/machine-emulator-tools).
RUN <<EOF
set -e
URL=https://github.com/cartesi/machine-emulator-tools/releases/download
VERSION=v${MACHINE_TOOLS_VERSION}
ARTIFACT=libcmt-${VERSION}-dev.deb
curl -fsSL ${URL}/${VERSION}/${ARTIFACT} -o ./libcmt.deb
apt-get install -y ./libcmt.deb
rm ./libcmt.deb
mv /usr/riscv64-linux-gnu/include/libcmt /usr/include/
mv /usr/riscv64-linux-gnu/lib/libcmt.a /usr/lib/
rm -rf /usr/riscv64-linux-gnu
EOF

# Install xgenext2fs (cartesi/genext2fs).
RUN <<EOF
set -e
URL=https://github.com/cartesi/genext2fs/releases/download
VERSION=v${MACHINE_XGENEXT2FS_VERSION}
ARCH=$(dpkg --print-architecture)
ARTIFACT=xgenext2fs_${ARCH}.deb
curl -fsSL ${URL}/${VERSION}/${ARTIFACT} -o ./xgenext2fs.deb
apt-get install -y ./xgenext2fs.deb
rm ./xgenext2fs.deb
EOF

# Clean up.
RUN <<EOF
rm -rf /var/lib/apt/lists/*
EOF

# =============================================================================
# STAGE: rollups-node-ci
#
Expand All @@ -165,7 +113,7 @@ EOF
# - Install docker.
# =============================================================================

FROM emulator-devel as rollups-node-ci
FROM emulator as rollups-node-ci

# Install git and build-essential (setup).
RUN <<EOF
Expand Down
115 changes: 115 additions & 0 deletions test/snapshot/snapshot.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,115 @@
// (c) Cartesi and individual authors (see AUTHORS)
// SPDX-License-Identifier: Apache-2.0 (see LICENSE)

package snapshot

import (
"errors"
"fmt"
"log/slog"

Check failure on line 9 in test/snapshot/snapshot.go

View workflow job for this annotation

GitHub Actions / test-go

"log/slog" imported and not used
"math"
"math/rand"
"os"
"os/exec"

Check failure on line 13 in test/snapshot/snapshot.go

View workflow job for this annotation

GitHub Actions / test-go

"os/exec" imported and not used (typecheck)
"strings"

"github.com/cartesi/rollups-node/pkg/emulator"
)

const (
ramLength = 64 << 20
)

var ImagesPath = "/usr/share/cartesi-machine/images/"

type Snapshot struct {
pkg string
temp string

// Directory where the snapshot was stored.
Dir string

// Reason why the snapshot stopped running before being stored.
BreakReason emulator.BreakReason
}

// FromScript creates a snapshot given an one-line command.
func FromScript(command string, cycles uint64) (*Snapshot, error) {
snapshot := &Snapshot{pkg: fmt.Sprintf("fromscript%d", rand.Int())}

err := snapshot.createTempDir()
if err != nil {
return nil, err
}

config := defaultMachineConfig()
config.Dtb.Entrypoint = command

err = snapshot.createRunAndStore(config, cycles)
return snapshot, err
}

// Close deletes the temporary directory created to hold the snapshot files.
func (snapshot *Snapshot) Close() error {
return os.RemoveAll(snapshot.temp)
}

// ------------------------------------------------------------------------------------------------

func defaultMachineConfig() *emulator.MachineConfig {
config := emulator.NewDefaultMachineConfig()
config.Ram.Length = ramLength
config.Ram.ImageFilename = ImagesPath + "linux.bin"
config.Dtb.Bootargs = strings.Join([]string{"quiet",
"no4lvl",
"quiet",
"earlycon=sbi",
"console=hvc0",
"rootfstype=ext2",
"root=/dev/pmem0",
"rw",
"init=/usr/sbin/cartesi-init"}, " ")
config.FlashDrive = []emulator.MemoryRangeConfig{{
Start: 0x80000000000000, //nolint:mnd
Length: math.MaxUint64,
ImageFilename: ImagesPath + "rootfs.ext2",
}}
return config
}

func (snapshot *Snapshot) createTempDir() error {
temp, err := os.MkdirTemp("", snapshot.pkg+"-*")
if err != nil {
return err
}
snapshot.temp = temp
snapshot.Dir = snapshot.temp + "/snapshot"
return nil
}

func (snapshot *Snapshot) createRunAndStore(config *emulator.MachineConfig, cycles uint64) error {
// Creates the (local) machine.
machine, err := emulator.NewMachine(config, &emulator.MachineRuntimeConfig{})
if err != nil {
return snapshot.errAndClose(err)
}
defer machine.Delete()

// Runs the machine.
snapshot.BreakReason, err = machine.Run(cycles)
if err != nil {
return snapshot.errAndClose(err)
}

// Stores the machine.
err = machine.Store(snapshot.Dir)
if err != nil {
return snapshot.errAndClose(err)
}

return nil
}

func (snapshot Snapshot) errAndClose(err error) error {
return errors.Join(err, snapshot.Close())
}

0 comments on commit a97dec4

Please sign in to comment.