probe-rs 0.31.0

A collection of on chip debugging tools to communicate with microchips.
Documentation
use crate::config::{NvmRegion, RamRegion, TargetDescriptionSource};
use crate::error;
use std::ops::Range;

/// Describes any error that happened during the or in preparation for the flashing procedure.
#[derive(thiserror::Error, Debug)]
pub enum FlashError {
    /// No flash algorithm was found by the given name.
    #[error("The {name} target has no flash algorithm called {name}")]
    AlgorithmNotFound {
        /// The name of the target.
        name: String,
        /// The name of the algorithm that was not found.
        algo_name: String,
    },
    /// Detected multiple preferred algorithms for the same flash region.
    #[error("detected multiple preferred algorithms for a flash region {region:?}")]
    MultiplePreferredAlgos {
        /// The region which matched multiple preferred flash algorithms.
        region: NvmRegion,
    },
    /// No flash memory contains the entire requested memory range.
    #[error("No flash memory contains the entire requested memory range {range:#010X?}.")]
    NoSuitableNvm {
        /// The requested memory range.
        range: Range<u64>,
        /// The source of this target description (was it a built in target or one loaded externally and from what file path?).
        description_source: TargetDescriptionSource,
    },
    /// Erasing the full chip flash failed.
    #[error("Failed to erase the whole chip.")]
    ChipEraseFailed {
        /// The source error of this error.
        source: Box<dyn std::error::Error + 'static + Send + Sync>,
    },
    /// Failed to read data from flash.
    #[error("Failed to read data from flash.")]
    FlashReadFailed {
        /// The source error of this error.
        source: Box<dyn std::error::Error + 'static + Send + Sync>,
    },
    /// Erasing the given flash sector failed.
    #[error("Failed to erase flash sector at address {sector_address:#010x}.")]
    EraseFailed {
        /// The address of the sector that should have been erased.
        sector_address: u64,
        /// The source error of this error.
        #[source]
        source: Box<dyn std::error::Error + 'static + Send + Sync>,
    },
    /// Writing the given page failed.
    #[error("The page write of the page at address {page_address:#010x} failed.")]
    PageWrite {
        /// The address of the page that should have been written.
        page_address: u64,
        /// The source error of this error.
        #[source]
        source: Box<dyn std::error::Error + 'static + Send + Sync>,
    },
    /// Initializing the flash algorithm failed.
    #[error("The initialization of the flash algorithm failed.")]
    Init(#[source] Box<dyn std::error::Error + 'static + Send + Sync>),
    /// Uninitializing the flash algorithm failed.
    #[error("The uninitialization of the flash algorithm failed.")]
    Uninit(#[source] Box<dyn std::error::Error + 'static + Send + Sync>),
    /// This target does not support full chip flash erases.
    #[error("The chip erase routine is not supported with the given flash algorithm.")]
    ChipEraseNotSupported,
    /// Calling the given routine returned the given error code.
    #[error(
        "The execution of '{name}' failed with code {error_code}. This might indicate a problem with the flash algorithm."
    )]
    RoutineCallFailed {
        /// The name of the routine that was called.
        name: &'static str,
        /// The error code the called routine returned.
        error_code: u32,
    },
    /// Failed to read the core status.
    #[error("Failed to read the core status.")]
    UnableToReadCoreStatus(#[source] error::Error),
    /// The core entered an unexpected status while executing a flashing operation.
    #[error("The core entered an unexpected status: {status:?}.")]
    UnexpectedCoreStatus {
        /// The status that the core entered.
        status: crate::CoreStatus,
    },
    /// The given address was not contained in the given NVM region.
    #[error("{address:#010x} is not contained in {region:?}")]
    AddressNotInRegion {
        /// The address which was not contained in `region`.
        address: u32,
        /// The region which did not contain `address`.
        region: NvmRegion,
    },
    /// An error occurred during the interaction with the core.
    #[error("Something during the interaction with the core went wrong")]
    Core(#[source] error::Error),
    /// Failed to reset, and then halt the CPU.
    #[error("Failed to reset, and then halt the CPU.")]
    ResetAndHalt(#[source] error::Error),
    /// Failed to start running code on the CPU.
    #[error("Failed to start running code on the CPU")]
    Run(#[source] error::Error),
    /// The RAM contents did not match the flash algorithm.
    #[error(
        "The RAM contents did not match the expected contents after loading the flash algorithm."
    )]
    FlashAlgorithmNotLoaded,
    /// Failed to load the flash algorithm into RAM at given address. This can happen if there is not enough space.
    ///
    /// Check the algorithm code and settings before you try again.
    #[error(
        "Failed to load flash algorithm into RAM at address {address:#010x}. Is there space for the algorithm header?"
    )]
    InvalidFlashAlgorithmLoadAddress {
        /// The address where the algorithm was supposed to be loaded to.
        address: u64,
    },
    /// Failed to configure a valid stack size for the flash algorithm.
    #[error("Failed to configure a stack of size {size} for the flash algorithm.")]
    InvalidFlashAlgorithmStackSize {
        /// The size of the stack that was tried to be configured.
        size: u64,
    },
    /// Failed to configure the data region of a flash algorithm.
    #[error(
        "Failed to place data to address {data_load_addr:#010x} in RAM. The data must be placed in the range {data_ram:#x?}."
    )]
    InvalidDataAddress {
        /// The address where the data was supposed to be loaded to.
        data_load_addr: u64,
        /// The range of the data memory.
        data_ram: Range<u64>,
    },
    // TODO: Warn at YAML parsing stage.
    // TODO: 1 Add information about flash (name, address)
    // TODO: 2 Add source of target definition (built-in, yaml)
    /// No flash algorithm was linked to this target.
    #[error(
        "Trying to write to flash region {range:#010x?}, but no suitable (default) flash loader algorithm is linked to the given target: {name}."
    )]
    NoFlashLoaderAlgorithmAttached {
        /// The name of the chip.
        name: String,
        /// The memory region that was tried to be written.
        range: Range<u64>,
    },
    /// More than one matching flash algorithm was found for the given memory range and all of them is marked as default.
    #[error(
        "Trying to write flash, but found more than one suitable flash loader algorithim marked as default for {region:?}."
    )]
    MultipleDefaultFlashLoaderAlgorithms {
        /// The region which matched more than one flash algorithm.
        region: NvmRegion,
    },
    /// More than one matching flash algorithm was found for the given memory range and none of them is marked as default.
    #[error(
        "Trying to write flash, but found more than one suitable flash algorithims but none marked as default for {region:?}."
    )]
    MultipleFlashLoaderAlgorithmsNoDefault {
        /// The region which matched more than one flash algorithm.
        region: NvmRegion,
    },
    /// Flash content verification failed.
    #[error("Flash content verification failed.")]
    Verify,
    /// Failed to read flash size.
    #[error("Failed to read flash size.")]
    FlashSizeFailed {
        /// The source error of this error.
        #[source]
        source: Box<dyn std::error::Error + 'static + Send + Sync>,
    },
    // TODO: 1 Add source of target definition
    // TOOD: 2 Do this at target load time.
    /// The given chip has no RAM defined.
    #[error("No suitable RAM region is defined for target: {name}.")]
    NoRamDefined {
        /// The name of the chip.
        name: String,
    },
    /// The given flash algorithm did not have a length multiple of 4 bytes.
    ///
    /// This means that the flash algorithm that was loaded is broken.
    #[error("Flash algorithm {name} does not have a length that is 4 byte aligned.")]
    InvalidFlashAlgorithmLength {
        /// The name of the flash algorithm.
        name: String,
        /// The source of the flash algorithm (was it a built in target or one loaded externally and from what file path?).
        algorithm_source: Option<TargetDescriptionSource>,
    },
    /// Two blocks of data overlap each other which means the loaded binary is broken.
    ///
    /// Please check your data and try again.
    #[error(
        "Adding data for addresses {added_addresses:#010x?} overlaps previously added data for addresses {existing_addresses:#010x?}."
    )]
    DataOverlaps {
        /// The address range that was tried to be added.
        added_addresses: Range<u64>,
        /// The address range that was already present.
        existing_addresses: Range<u64>,
    },
    /// No core can access this NVM region.
    #[error("No core can access the NVM region {0:?}.")]
    NoNvmCoreAccess(NvmRegion),
    /// No core can access this RAM region.
    #[error("No core can access the RAM region {0:?}.")]
    NoRamCoreAccess(RamRegion),
    /// The register value supplied for this flash algorithm is out of the supported range.
    #[error("The register value {0:#010x} is out of the supported range.")]
    RegisterValueNotSupported(u64),
    /// Stack overflow while flashing.
    #[error("Stack overflow detected during {operation}.")]
    StackOverflowDetected {
        /// The operation that caused the stack overflow.
        operation: &'static str,
    },
}