From b78da066c09255a3f5e6b3c6d655c71dcdd461e2 Mon Sep 17 00:00:00 2001 From: Eduardo Bart Date: Sun, 28 Apr 2024 09:59:47 -0300 Subject: [PATCH] feat: support snapshot/rollback for local machines --- src/cartesi-machine.lua | 5 +- src/machine-state.h | 16 ++-- src/machine.cpp | 144 ++++++++++++++++++++++++++---- src/machine.h | 7 +- src/os.cpp | 51 ++++++++--- src/os.h | 19 +++- src/pma.cpp | 66 +++----------- src/pma.h | 73 ++++++++------- src/uarch-machine.h | 4 +- src/uarch-state.h | 15 ++-- src/virtual-machine.cpp | 18 +++- src/virtual-machine.h | 1 + tests/misc/test-machine-c-api.cpp | 31 +++---- 13 files changed, 287 insertions(+), 163 deletions(-) diff --git a/src/cartesi-machine.lua b/src/cartesi-machine.lua index aee81f419..bad958eaa 100755 --- a/src/cartesi-machine.lua +++ b/src/cartesi-machine.lua @@ -2106,10 +2106,7 @@ if gdb_address then end if config.processor.iunrep ~= 0 then stderr("Running in unreproducible mode!\n") end if store_config == stderr then store_machine_config(config, stderr) end -if cmio_advance or cmio_inspect then - check_cmio_htif_config(config.htif) - assert(remote_address or not perform_rollbacks, "cmio requires --remote-address for snapshot/commit/rollback") -end +if cmio_advance or cmio_inspect then check_cmio_htif_config(config.htif) end if initial_hash then assert(config.processor.iunrep == 0, "hashes are meaningless in unreproducible mode") print_root_hash(machine, stderr_unsilenceable) diff --git a/src/machine-state.h b/src/machine-state.h index b5d03de3f..4fc1a2bbd 100644 --- a/src/machine-state.h +++ b/src/machine-state.h @@ -44,16 +44,16 @@ struct unpacked_iflags { /// \brief Machine state. /// \details The machine_state structure contains the entire /// state of a Cartesi machine. -struct machine_state { - - ~machine_state() { - ; - } // Due to bug in clang++ - - /// \brief No copy or move constructor or assignment - machine_state(const machine_state &other) = delete; +struct machine_state final { + /// \brief Destructor + ~machine_state() = default; + /// \brief Copy constructor + machine_state(const machine_state &other) = default; + /// \brief No move constructor machine_state(machine_state &&other) = delete; + /// \brief No copy assignment machine_state &operator=(const machine_state &other) = delete; + /// \brief No move assignment machine_state &operator=(machine_state &&other) = delete; // The following state fields are very hot, diff --git a/src/machine.cpp b/src/machine.cpp index c2c22a738..d200db7d1 100644 --- a/src/machine.cpp +++ b/src/machine.cpp @@ -191,6 +191,11 @@ static bool DID_is_protected(PMA_ISTART_DID DID) { } } +static uint64_t get_task_concurrency(uint64_t value) { + const uint64_t concurrency = value > 0 ? value : std::max(os_get_concurrency(), UINT64_C(1)); + return std::min(concurrency, static_cast(THREADS_MAX)); +} + void machine::replace_memory_range(const memory_range_config &range) { for (auto &pma : m_s.pmas) { if (pma.get_start() == range.start && pma.get_length() == range.length) { @@ -250,6 +255,49 @@ static void init_tlb_entry(machine &m, uint64_t eidx) { tlbce.pma_index = TLB_INVALID_PMA; } +template +static void adjust_tlb_entry(machine &m, uint64_t eidx) { + tlb_hot_entry &tlbhe = m.get_state().tlb.hot[ETYPE][eidx]; + const tlb_cold_entry &tlbce = m.get_state().tlb.cold[ETYPE][eidx]; + auto vaddr_page = tlbhe.vaddr_page; + auto paddr_page = tlbce.paddr_page; + auto pma_index = tlbce.pma_index; + if (tlbhe.vaddr_page != TLB_INVALID_PAGE) { + if ((vaddr_page & ~PAGE_OFFSET_MASK) != vaddr_page) { + throw std::invalid_argument{"misaligned virtual page address in TLB entry"}; + } + if ((paddr_page & ~PAGE_OFFSET_MASK) != paddr_page) { + throw std::invalid_argument{"misaligned physical page address in TLB entry"}; + } + const pma_entry &pma = m.find_pma_entry(paddr_page); + // Checks if the PMA still valid + if (pma.get_length() == 0 || !pma.get_istart_M() || pma_index >= m.get_state().pmas.size() || + &pma != &m.get_state().pmas[pma_index]) { + throw std::invalid_argument{"invalid PMA for TLB entry"}; + } + const unsigned char *hpage = pma.get_memory().get_host_memory() + (paddr_page - pma.get_start()); + // Valid TLB entry + tlbhe.vh_offset = cast_ptr_to_addr(hpage) - vaddr_page; + } else { + tlbhe.vh_offset = 0; + } +} + +void machine::build_pma_list() { + // Initialize the vector of the pmas used by the merkle tree to compute hashes. + // First, add the pmas visible to the big machine, except the sentinel + for (auto &pma : m_s.pmas | sliced(0, m_s.pmas.size() - 1)) { + m_pmas.push_back(&pma); + } + + // Second, push uarch pmas that are visible only to the microarchitecture interpreter + m_pmas.push_back(&m_uarch.get_state().shadow_state); + m_pmas.push_back(&m_uarch.get_state().ram); + + // Last, add sentinel + m_pmas.push_back(&m_s.empty_pma); +} + machine::machine(const machine_config &c, const machine_runtime_config &r) : m_s{}, m_t{}, @@ -462,18 +510,8 @@ machine::machine(const machine_config &c, const machine_runtime_config &r) : // Add sentinel to PMA vector register_pma_entry(make_empty_pma_entry("sentinel"s, 0, 0)); - // Initialize the vector of the pmas used by the merkle tree to compute hashes. - // First, add the pmas visible to the big machine, except the sentinel - for (auto &pma : m_s.pmas | sliced(0, m_s.pmas.size() - 1)) { - m_pmas.push_back(&pma); - } - - // Second, push uarch pmas that are visible only to the microarchitecture interpreter - m_pmas.push_back(&m_uarch.get_state().shadow_state); - m_pmas.push_back(&m_uarch.get_state().ram); - - // Last, add sentinel - m_pmas.push_back(&m_s.empty_pma); + // Populate m_pmas + build_pma_list(); // Initialize TLB device // this must be done after all PMA entries are already registered, so we can lookup page addresses @@ -543,6 +581,83 @@ machine::machine(const std::string &dir, const machine_runtime_config &r) : mach } } +machine::machine(const machine &other) : + m_s(other.m_s), + m_t(), + m_pmas(), + m_c(other.m_c), + m_uarch(other.m_uarch), + m_r(other.m_r), + m_mrds(other.m_mrds) { + + // Cannot copy machine with VirtIO devices + if (!other.m_vdevs.empty()) { + throw std::runtime_error{"cannot copy machine with VirtIO devices"}; + } + + // Populate m_pmas + build_pma_list(); + + // ??(edubart) Mark all pages as dirty for now, we need to implement a deep copy for merkle tree later. + for (auto *pma : m_pmas) { + pma->mark_pages_dirty(); + } + + // Clone memory mapped PMAs + for (auto *pma : m_pmas) { + // Need to remap only memory + if (pma->get_istart_M()) { + auto &mem = pma->get_memory(); + + // Map a new PMA memory with the same size + pma_memory cloned_mem(pma->get_description(), mem.get_length(), pma_memory::callocd{}); + const unsigned char *data = mem.get_host_memory(); + unsigned char *cloned_data = cloned_mem.get_host_memory(); + + // Copy memory from the old PMA to the new PMA memory + const uint64_t num_threads = get_task_concurrency(m_r.concurrency.update_merkle_tree); + if (num_threads == 1) { + memcpy(cloned_data, data, mem.get_length()); + } else { // Use multiple threads to copy the memory + const uint64_t pages_in_range = (pma->get_length() + PMA_PAGE_SIZE - 1) / PMA_PAGE_SIZE; + const uint64_t pages_per_thread = pages_in_range / num_threads; + if (pages_per_thread == 0) { + memcpy(cloned_mem.get_host_memory(), mem.get_host_memory(), mem.get_length()); + } else { + const uint64_t rest_pages = pages_per_thread % num_threads; + const uint64_t chunk_len = pages_per_thread * PMA_PAGE_SIZE; + // Let each thread copy a chunk of memory + const bool succeeded = + os_parallel_for(num_threads, [&](int i, const parallel_for_mutex &mutex) -> bool { + (void) mutex; + const uint64_t page_start = i * chunk_len; + memcpy(cloned_data + page_start, data + page_start, chunk_len); + return true; + }); + if (!succeeded) { + throw std::runtime_error{"copy memory parallel for failed"}; + } + // Copy remaining pages + if (rest_pages > 0) { + const uint64_t page_start = num_threads * chunk_len; + memcpy(cloned_data, data, mem.get_length() - page_start); + } + } + } + + // Replace the PMA with the new mapped PMA + mem.replace(std::move(cloned_mem)); + } + } + + // Adjust TLB offsets to host memory pointers + for (uint64_t i = 0; i < PMA_TLB_SIZE; ++i) { + adjust_tlb_entry(*this, i); + adjust_tlb_entry(*this, i); + adjust_tlb_entry(*this, i); + } +} + void machine::prepare_virtio_devices_select(select_fd_sets *fds, uint64_t *timeout_us) { for (auto &vdev : m_vdevs) { vdev->prepare_select(fds, timeout_us); @@ -1550,11 +1665,6 @@ bool machine::verify_dirty_page_maps(void) const { return !broken; } -static uint64_t get_task_concurrency(uint64_t value) { - const uint64_t concurrency = value > 0 ? value : std::max(os_get_concurrency(), UINT64_C(1)); - return std::min(concurrency, static_cast(THREADS_MAX)); -} - bool machine::update_merkle_tree(void) const { machine_merkle_tree::hasher_type gh; static_assert(PMA_PAGE_SIZE == machine_merkle_tree::get_page_size(), diff --git a/src/machine.h b/src/machine.h index 17e28be42..717189ee0 100644 --- a/src/machine.h +++ b/src/machine.h @@ -72,6 +72,9 @@ class machine final { static const pma_entry::flags m_cmio_rx_buffer_flags; ///< PMA flags used for cmio rx buffer static const pma_entry::flags m_cmio_tx_buffer_flags; ///< PMA flags used for cmio tx buffer + /// \brief Build PMA list from machine state and uarch machine state. + void build_pma_list(); + /// \brief Allocates a new PMA entry. /// \param pma PMA entry to add to machine. /// \returns Reference to corresponding entry in machine state. @@ -186,8 +189,8 @@ class machine final { /// \brief No default constructor machine(void) = delete; - /// \brief No copy constructor - machine(const machine &other) = delete; + /// \brief Copy constructor + machine(const machine &other); /// \brief No move constructor machine(machine &&other) = delete; /// \brief No copy assignment diff --git a/src/os.cpp b/src/os.cpp index cb1a48750..4e41a3cd7 100644 --- a/src/os.cpp +++ b/src/os.cpp @@ -524,7 +524,29 @@ int os_mkdir(const char *path, int mode) { #endif // HAVE_MKDIR } -unsigned char *os_map_file(const char *path, uint64_t length, bool shared) { +mmapd os_mmap_mem(uint64_t length) { + if (length == 0) { + return {nullptr, 0, -1, false}; + } + +#ifdef HAVE_MMAP + auto *host_memory = + static_cast(mmap(nullptr, length, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0)); + if (host_memory == MAP_FAILED) { // NOLINT(cppcoreguidelines-pro-type-cstyle-cast,performance-no-int-to-ptr) + throw std::system_error{errno, std::generic_category(), "could not map memory file"s}; + } + return {host_memory, length, -1, false}; + +#else + // Use calloc to improve performance and to initialize it to zeros. + // NOLINTNEXTLINE(cppcoreguidelines-no-malloc, cppcoreguidelines-prefer-member-initializer) + auto *host_memory = static_cast(std::calloc(1, length)); + return {host_memory, length, -1, false}; + +#endif // HAVE_MMAP +} + +mmapd os_mmap_file(const char *path, uint64_t length, bool shared) { if (!path || *path == '\0') { throw std::runtime_error{"image file path must be specified"s}; } @@ -563,9 +585,7 @@ unsigned char *os_map_file(const char *path, uint64_t length, bool shared) { throw std::system_error{errno, std::generic_category(), "could not map image file '"s + path + "' to memory"s}; } - // We can close the file after mapping it, because the OS will retain a reference of the file on its own - close(backing_file); - return host_memory; + return {host_memory, length, backing_file, shared}; #elif defined(_WIN32) const int oflag = (shared ? _O_RDWR : _O_RDONLY) | _O_BINARY; @@ -609,9 +629,7 @@ unsigned char *os_map_file(const char *path, uint64_t length, bool shared) { throw std::system_error{errno, std::generic_category(), "could not map image file '"s + path + "' to memory"s}; } - // We can close the file after mapping it, because the OS will retain a reference of the file on its own - _close(backing_file); - return host_memory; + return {host_memory, length, backing_file, shared}; #else if (shared) { @@ -637,7 +655,7 @@ unsigned char *os_map_file(const char *path, uint64_t length, bool shared) { throw std::runtime_error{"image file '"s + path + "' of "s + " is too large for range"s}; } - // use calloc to improve performance + // Use calloc to improve performance and to initialize it to zeros. // NOLINTNEXTLINE(cppcoreguidelines-no-malloc, cppcoreguidelines-prefer-member-initializer) auto host_memory = static_cast(std::calloc(1, length)); if (!host_memory) { @@ -650,22 +668,27 @@ unsigned char *os_map_file(const char *path, uint64_t length, bool shared) { if (ferror(fp.get())) { throw std::system_error{errno, std::generic_category(), "error reading from image file '"s + path + "'"s}; } - return host_memory; + return {host_memory, length, -1, false}; #endif // HAVE_MMAP } -void os_unmap_file(unsigned char *host_memory, uint64_t length) { +void os_munmap(const mmapd &mem) { #ifdef HAVE_MMAP - munmap(host_memory, length); + munmap(mem.host_memory, mem.length); + if (mem.backing_file != -1) { + close(mem.backing_file); + } #elif defined(_WIN32) (void) length; - UnmapViewOfFile(host_memory); + UnmapViewOfFile(mem.host_memory); + if (mem.backing_file != -1) { + _close(mem.backing_file); + } #else - (void) length; - std::free(host_memory); + std::free(mem.host_memory); #endif // HAVE_MMAP } diff --git a/src/os.h b/src/os.h index c8178e525..d8653a092 100644 --- a/src/os.h +++ b/src/os.h @@ -88,11 +88,22 @@ void os_putchars(const uint8_t *data, size_t len); /// \brief Creates a new directory int os_mkdir(const char *path, int mode); -/// \brief Maps a file to memory -unsigned char *os_map_file(const char *path, uint64_t length, bool shared); +/// \brief Mapped memory entry +struct mmapd { + unsigned char *host_memory = nullptr; ///< Start of associated memory region in host. + uint64_t length = 0; ///< Length of memory range (copy of PMA length field). + int backing_file = -1; ///< File descriptor backing the memory + bool shared = false; ///< True when the memory is shared with the file descriptor +}; + +/// \brief Maps memory +mmapd os_mmap_mem(uint64_t length); + +/// \brief Maps memory from a file +mmapd os_mmap_file(const char *path, uint64_t length, bool shared); -/// \brief Unmaps a file from memory -void os_unmap_file(unsigned char *host_memory, uint64_t length); +/// \brief Unmaps memory +void os_munmap(const mmapd &m); /// \brief Get time elapsed since its first call with microsecond precision int64_t os_now_us(); diff --git a/src/pma.cpp b/src/pma.cpp index 2a0481f19..3d2217efd 100644 --- a/src/pma.cpp +++ b/src/pma.cpp @@ -29,47 +29,27 @@ namespace cartesi { using namespace std::string_literals; void pma_memory::release(void) { - if (m_mmapped) { - os_unmap_file(m_host_memory, m_length); - m_mmapped = false; - } else { - std::free(m_host_memory); // NOLINT(cppcoreguidelines-no-malloc) + if (m_mem.length > 0) { + os_munmap(m_mem); } - m_host_memory = nullptr; - m_length = 0; + m_mem = mmapd{}; } pma_memory::~pma_memory() { release(); } -pma_memory::pma_memory(pma_memory &&other) noexcept : - m_length{std::move(other.m_length)}, - m_host_memory{std::move(other.m_host_memory)}, - m_mmapped{std::move(other.m_mmapped)} { - // set other to safe state - other.m_host_memory = nullptr; - other.m_mmapped = false; - other.m_length = 0; -} - -pma_memory::pma_memory(const std::string &description, uint64_t length, const callocd &c) : - m_length{length}, - m_host_memory{nullptr}, - m_mmapped{false} { +pma_memory::pma_memory(const std::string &description, uint64_t length, const callocd &c) { (void) c; - // use calloc to improve performance - // NOLINTNEXTLINE(cppcoreguidelines-no-malloc, cppcoreguidelines-prefer-member-initializer) - m_host_memory = static_cast(std::calloc(1, length)); - if (!m_host_memory) { - throw std::runtime_error{"error allocating memory for "s + description}; + try { + m_mem = os_mmap_mem(length); + } catch (std::exception &e) { + throw std::runtime_error{e.what() + " when initializing "s + description}; } } -pma_memory::pma_memory(const std::string &description, uint64_t length, const mockd &m) : - m_length{length}, - m_host_memory{nullptr}, - m_mmapped{false} { +pma_memory::pma_memory(const std::string &description, uint64_t length, const mockd &m) { + m_mem.length = length; (void) m; (void) description; } @@ -98,7 +78,7 @@ pma_memory::pma_memory(const std::string &description, uint64_t length, const st throw std::runtime_error{"image file '"s + path + "' of "s + description + " is too large for range"s}; } // Read to host memory - auto read = fread(m_host_memory, 1, length, fp.get()); + auto read = fread(m_mem.host_memory, 1, length, fp.get()); (void) read; if (ferror(fp.get())) { throw std::system_error{errno, std::generic_category(), @@ -107,31 +87,14 @@ pma_memory::pma_memory(const std::string &description, uint64_t length, const st } } -pma_memory::pma_memory(const std::string &description, uint64_t length, const std::string &path, const mmapd &m) : - m_length{length}, - m_host_memory{nullptr}, - m_mmapped{false} { +pma_memory::pma_memory(const std::string &description, uint64_t length, const std::string &path, bool shared) { try { - m_host_memory = os_map_file(path.c_str(), length, m.shared); - m_mmapped = true; + m_mem = os_mmap_file(path.c_str(), length, shared); } catch (std::exception &e) { throw std::runtime_error{e.what() + " when initializing "s + description}; } } -pma_memory &pma_memory::operator=(pma_memory &&other) noexcept { - release(); - // copy from other - m_host_memory = std::move(other.m_host_memory); - m_mmapped = std::move(other.m_mmapped); - m_length = std::move(other.m_length); - // set other to safe state - other.m_host_memory = nullptr; - other.m_mmapped = false; - other.m_length = 0; - return *this; -} - uint64_t pma_entry::get_istart(void) const { uint64_t istart = m_start; istart |= (static_cast(get_istart_M()) << PMA_ISTART_M_SHIFT); @@ -206,8 +169,7 @@ pma_entry make_mmapd_memory_pma_entry(const std::string &description, uint64_t s if (length == 0) { throw std::invalid_argument{description + " length cannot be zero"s}; } - return pma_entry{description, start, length, pma_memory{description, length, path, pma_memory::mmapd{shared}}, - memory_peek}; + return pma_entry{description, start, length, pma_memory{description, length, path, shared}, memory_peek}; } pma_entry make_callocd_memory_pma_entry(const std::string &description, uint64_t start, uint64_t length) { diff --git a/src/pma.h b/src/pma.h index e11e0ec93..73b9f6474 100644 --- a/src/pma.h +++ b/src/pma.h @@ -24,6 +24,7 @@ #include #include +#include "os.h" #include "pma-constants.h" #include "pma-driver.h" @@ -91,26 +92,21 @@ class pma_device final { /// \brief Data for memory ranges. class pma_memory final { - - uint64_t m_length; ///< Length of memory range (copy of PMA length field). - unsigned char *m_host_memory; ///< Start of associated memory region in host. - bool m_mmapped; ///< True if memory was mapped from a file. + mmapd m_mem; ///< Memory map entry /// \brief Close file and/or release memory. void release(void); public: - /// \brief Mmap'd range data (shared or not). - struct mmapd { - bool shared; - }; + /// \brief Destructor + ~pma_memory(void); /// \brief Constructor for mmap'd ranges. /// \param description Informative description of PMA entry for use in error messages /// \param length Length of range. /// \param path Path for backing file. /// \param m Mmap'd range data (shared or not). - pma_memory(const std::string &description, uint64_t length, const std::string &path, const mmapd &m); + pma_memory(const std::string &description, uint64_t length, const std::string &path, bool shared); /// \brief Calloc'd range data (just a tag). struct callocd {}; @@ -137,34 +133,42 @@ class pma_memory final { /// \param m Mock'd range data (just a tag). pma_memory(const std::string &description, uint64_t length, const mockd &m); - /// \brief No copy constructor - pma_memory(const pma_memory &) = delete; + /// \brief Copy constructor. + /// \details Performs a SHALLOW copy. + pma_memory(const pma_memory &other) = default; - /// \brief No copy assignment - pma_memory &operator=(const pma_memory &) = delete; + /// \brief Copy assignment. + /// \details Performs a SHALLOW copy. + pma_memory &operator=(const pma_memory &other) = default; - /// \brief Move constructor - pma_memory(pma_memory &&) noexcept; - - /// \brief Move assignment - pma_memory &operator=(pma_memory &&) noexcept; + /// \brief Move constructor. + pma_memory(pma_memory &&other) noexcept { + std::swap(m_mem, other.m_mem); + } - /// \brief Destructor - ~pma_memory(void); + /// \brief Move assignment. + pma_memory &operator=(pma_memory &&other) noexcept { + release(); + std::swap(m_mem, other.m_mem); + return *this; + } - /// \brief Returns start of associated memory region in host - unsigned char *get_host_memory(void) { - return m_host_memory; + /// \brief Replace current mapped memory with the mapped memory from another PMA. + /// \details It takes ownership of the other PMA memory, + /// however it DOES NOT release currently mapped memory. + void replace(pma_memory &&other) noexcept { + m_mem = std::move(other.m_mem); + other.m_mem = mmapd{}; } /// \brief Returns start of associated memory region in host - const unsigned char *get_host_memory(void) const { - return m_host_memory; + unsigned char *get_host_memory(void) const { + return m_mem.host_memory; } /// \brief Returns copy of PMA length field (needed for munmap). uint64_t get_length(void) const { - return m_length; + return m_mem.length; } }; @@ -207,14 +211,14 @@ class pma_entry final { m_data; public: - /// \brief No copy constructor - pma_entry(const pma_entry &) = delete; - /// \brief No copy assignment - pma_entry &operator=(const pma_entry &) = delete; + /// \brief Default copy constructor + pma_entry(const pma_entry &) = default; + /// \brief Default copy assignment + pma_entry &operator=(const pma_entry &) = default; /// \brief Default move constructor - pma_entry(pma_entry &&) = default; // NOLINT(bugprone-exception-escape) + pma_entry(pma_entry &&) noexcept = default; /// \brief Default move assignment - pma_entry &operator=(pma_entry &&) = default; // NOLINT(bugprone-exception-escape) + pma_entry &operator=(pma_entry &&) noexcept = default; /// \bried Default destructor ~pma_entry() = default; @@ -481,6 +485,11 @@ class pma_entry final { return std::fill(m_dirty_page_map.begin(), m_dirty_page_map.end(), 0); } + /// \brief Marks all pages in range as dirty + void mark_pages_dirty(void) { + return m_dirty_page_map.clear(); + } + /// \brief Returns PMA description as a string /// \returns Description const std::string &get_description(void) const { diff --git a/src/uarch-machine.h b/src/uarch-machine.h index 42b72c32b..6ad89bda4 100644 --- a/src/uarch-machine.h +++ b/src/uarch-machine.h @@ -40,8 +40,8 @@ class uarch_machine final { /// \brief No default constructor uarch_machine(void) = delete; - /// \brief No copy constructor - uarch_machine(const uarch_machine &other) = delete; + /// \brief Copy constructor + uarch_machine(const uarch_machine &other) = default; /// \brief No move constructor uarch_machine(uarch_machine &&other) = delete; /// \brief No copy assignment diff --git a/src/uarch-state.h b/src/uarch-state.h index 462d1a17b..b054048ae 100644 --- a/src/uarch-state.h +++ b/src/uarch-state.h @@ -28,15 +28,16 @@ namespace cartesi { -struct uarch_state { - ~uarch_state() { - ; - } - - /// \brief No copy or move constructor or assignment - uarch_state(const uarch_state &other) = delete; +struct uarch_state final { + /// \brief Destructor + ~uarch_state() = default; + /// \brief Copy constructor + uarch_state(const uarch_state &other) = default; + /// \brief No move constructor uarch_state(uarch_state &&other) = delete; + /// \brief No copy assignment uarch_state &operator=(const uarch_state &other) = delete; + /// \brief No move assignment uarch_state &operator=(uarch_state &&other) = delete; uint64_t pc; ///< Program counter. diff --git a/src/virtual-machine.cpp b/src/virtual-machine.cpp index cfcebfd7b..d312322a8 100644 --- a/src/virtual-machine.cpp +++ b/src/virtual-machine.cpp @@ -26,6 +26,7 @@ virtual_machine::virtual_machine(const std::string &dir, const machine_runtime_c virtual_machine::~virtual_machine(void) { delete m_machine; + delete m_forked_machine; } void virtual_machine::do_store(const std::string &dir) { @@ -465,15 +466,26 @@ void virtual_machine::do_destroy() { } void virtual_machine::do_snapshot(void) { - throw std::runtime_error("snapshot is not supported"); + // Copy current machine + m_forked_machine = new machine(*m_machine); } void virtual_machine::do_commit(void) { - throw std::runtime_error("commit is not supported"); + // Discard forked machine + if (m_forked_machine) { + delete m_forked_machine; + m_forked_machine = nullptr; + } } void virtual_machine::do_rollback(void) { - throw std::runtime_error("rollback is not supported"); + if (!m_forked_machine) { + throw std::runtime_error("no machine snapshot to rollback to"); + } + // Replace current machine with the forked machine + delete m_machine; + m_machine = m_forked_machine; + m_forked_machine = nullptr; } uint64_t virtual_machine::do_read_uarch_x(int i) const { diff --git a/src/virtual-machine.h b/src/virtual-machine.h index f6fc0c943..c3df886a1 100644 --- a/src/virtual-machine.h +++ b/src/virtual-machine.h @@ -28,6 +28,7 @@ namespace cartesi { class virtual_machine : public i_virtual_machine { using machine = cartesi::machine; machine *m_machine; + machine *m_forked_machine = nullptr; public: virtual_machine(const machine_config &c, const machine_runtime_config &r = {}); diff --git a/tests/misc/test-machine-c-api.cpp b/tests/misc/test-machine-c-api.cpp index 69400e102..7259bbf5e 100644 --- a/tests/misc/test-machine-c-api.cpp +++ b/tests/misc/test-machine-c-api.cpp @@ -1592,29 +1592,24 @@ BOOST_AUTO_TEST_CASE_NOLINT(snapshot_null_machine_test) { BOOST_CHECK_EQUAL(error_code, CM_ERROR_INVALID_ARGUMENT); } -BOOST_FIXTURE_TEST_CASE_NOLINT(snapshot_basic_test, ordinary_machine_fixture) { +BOOST_FIXTURE_TEST_CASE_NOLINT(snapshot_rollback_basic_test, ordinary_machine_fixture) { char *err_msg = nullptr; int error_code = cm_snapshot(_machine, &err_msg); - std::string result = err_msg; - std::string origin("snapshot is not supported"); - BOOST_CHECK_EQUAL(error_code, CM_ERROR_RUNTIME_ERROR); - BOOST_CHECK_EQUAL(origin, result); - cm_delete_cstring(err_msg); -} - -BOOST_AUTO_TEST_CASE_NOLINT(rollback_null_machine_test) { - int error_code = cm_rollback(nullptr, nullptr); - BOOST_CHECK_EQUAL(error_code, CM_ERROR_INVALID_ARGUMENT); + BOOST_CHECK_EQUAL(error_code, CM_ERROR_OK); + BOOST_CHECK_EQUAL(err_msg, nullptr); + error_code = cm_rollback(_machine, &err_msg); + BOOST_CHECK_EQUAL(error_code, CM_ERROR_OK); + BOOST_CHECK_EQUAL(err_msg, nullptr); } -BOOST_FIXTURE_TEST_CASE_NOLINT(rollback_basic_test, ordinary_machine_fixture) { +BOOST_FIXTURE_TEST_CASE_NOLINT(snapshot_commit_basic_test, ordinary_machine_fixture) { char *err_msg = nullptr; - int error_code = cm_rollback(_machine, &err_msg); - std::string result = err_msg; - std::string origin("rollback is not supported"); - BOOST_CHECK_EQUAL(error_code, CM_ERROR_RUNTIME_ERROR); - BOOST_CHECK_EQUAL(origin, result); - cm_delete_cstring(err_msg); + int error_code = cm_snapshot(_machine, &err_msg); + BOOST_CHECK_EQUAL(error_code, CM_ERROR_OK); + BOOST_CHECK_EQUAL(err_msg, nullptr); + error_code = cm_commit(_machine, &err_msg); + BOOST_CHECK_EQUAL(error_code, CM_ERROR_OK); + BOOST_CHECK_EQUAL(err_msg, nullptr); } BOOST_AUTO_TEST_CASE_NOLINT(read_x_null_machine_test) {