nander-rs 0.5.5

Modern SPI NAND/NOR Flash programmer for CH341A with cross-platform GUI and comprehensive diagnostics
Documentation
//! SPI NOR Flash Protocol Implementation
//!
//! This module implements the SPI NOR protocol according to infrastructure standards.

#[cfg(test)]
mod tests;

use std::time::{Duration, Instant};

use crate::domain::chip::ChipSpec;
use crate::domain::{EraseRequest, FlashOperation, Progress, ReadRequest, WriteRequest};
use crate::error::{Error, Result};
use crate::infrastructure::flash_protocol::commands::*;
use crate::infrastructure::programmer::Programmer;

pub struct SpiNor<P: Programmer> {
    programmer: P,
    spec: ChipSpec,
}

impl<P: Programmer> SpiNor<P> {
    pub fn new(programmer: P, spec: ChipSpec) -> Self {
        Self { programmer, spec }
    }

    /// Get chip specification
    pub fn spec(&self) -> &ChipSpec {
        &self.spec
    }

    /// Get mutable access to the programmer (for testing)
    #[cfg(test)]
    pub fn programmer_mut(&mut self) -> &mut P {
        &mut self.programmer
    }
    // =========================================================================
    // Internal Helper Methods
    // =========================================================================

    fn wait_ready(&mut self) -> Result<()> {
        let start = Instant::now();
        let timeout = Duration::from_secs(30); // NOR chip erase can take longer
        loop {
            let status = self.read_status()?;
            if status & STATUS_NOR_WIP == 0 {
                return Ok(());
            }
            if start.elapsed() > timeout {
                return Err(Error::Timeout);
            }
            std::thread::sleep(Duration::from_micros(100));
        }
    }

    pub(crate) fn read_status(&mut self) -> Result<u8> {
        self.programmer.set_cs(true)?;
        self.programmer.spi_write(&[CMD_READ_STATUS_ALT])?;
        let data = self.programmer.spi_read(1)?;
        self.programmer.set_cs(false)?;
        Ok(data[0])
    }

    fn write_enable(&mut self) -> Result<()> {
        self.programmer.set_cs(true)?;
        self.programmer.spi_write(&[CMD_WRITE_ENABLE])?;
        self.programmer.set_cs(false)?;
        Ok(())
    }

    fn addr_to_bytes(&self, addr: u32) -> Vec<u8> {
        if self.spec.layout.is_dataflash {
            let page_size = self.spec.layout.page_size;
            let page_offset_bits = match page_size {
                256 => 8,
                264 => 9,
                512 => 9,
                528 => 10,
                1024 => 10,
                1056 => 11,
                2048 => 11,
                2112 => 12,
                _ => 8, // Default to 256
            };
            let page = addr / page_size;
            let offset = addr % page_size;
            let df_addr = (page << page_offset_bits) | offset;

            vec![(df_addr >> 16) as u8, (df_addr >> 8) as u8, df_addr as u8]
        } else if self.spec.capabilities.supports_4byte_addr {
            vec![
                (addr >> 24) as u8,
                (addr >> 16) as u8,
                (addr >> 8) as u8,
                addr as u8,
            ]
        } else {
            vec![(addr >> 16) as u8, (addr >> 8) as u8, addr as u8]
        }
    }

    #[allow(dead_code)]
    fn read_internal(&mut self, address: u32, len: usize) -> Result<Vec<u8>> {
        let addr_bytes = self.addr_to_bytes(address);
        let cmd = if self.spec.capabilities.supports_4byte_addr {
            CMD_NOR_READ_4B
        } else {
            CMD_NOR_READ
        };

        let mut tx = vec![cmd];
        tx.extend_from_slice(&addr_bytes);

        self.programmer.set_cs(true)?;
        self.programmer.spi_write(&tx)?;
        let data = self.programmer.spi_read(len)?;
        self.programmer.set_cs(false)?;
        Ok(data)
    }
}

impl<P: Programmer> FlashOperation for SpiNor<P> {
    fn read(&mut self, request: ReadRequest, on_progress: &dyn Fn(Progress)) -> Result<Vec<u8>> {
        // NOR flash supports arbitrary address reads with continuous read mode
        let address = request.address.as_u32();
        let length = request.length as usize;

        // Use larger chunks for better throughput
        // The programmer's max_bulk_transfer_size tells us the optimal chunk size
        let chunk_size = self.programmer.max_bulk_transfer_size().min(32 * 1024);

        let mut result = Vec::with_capacity(length);
        let mut remaining = length;
        let mut current_addr = address;

        while remaining > 0 {
            let read_size = remaining.min(chunk_size);

            // Use Fast Read command (0x0B) with dummy byte for higher speed
            // Format: CMD + 3-byte addr + 1 dummy byte, then read data
            let addr_bytes = self.addr_to_bytes(current_addr);
            let cmd_byte = if self.spec.capabilities.supports_4byte_addr {
                CMD_NOR_FAST_READ_4B
            } else {
                CMD_NOR_FAST_READ
            };

            let mut cmd = vec![cmd_byte];
            cmd.extend_from_slice(&addr_bytes);
            cmd.push(0x00); // Dummy byte

            let mut attempts = 0;
            let chunk = loop {
                match self.programmer.spi_transaction(&cmd, read_size) {
                    Ok(data) => break data,
                    Err(e) => {
                        if attempts < request.retry_count {
                            attempts += 1;
                            log::warn!(
                                "Read error at 0x{:08X}, retrying (attempt {}): {}",
                                current_addr,
                                attempts,
                                e
                            );
                            continue;
                        } else {
                            return Err(e);
                        }
                    }
                }
            };
            result.extend_from_slice(&chunk);

            remaining -= read_size;
            current_addr += read_size as u32;

            on_progress(Progress::new(result.len() as u64, length as u64));
        }

        Ok(result)
    }

    fn write(&mut self, request: WriteRequest, on_progress: &dyn Fn(Progress)) -> Result<()> {
        // NOR flash must be written in 256-byte pages
        const PAGE_SIZE: usize = 256;

        let data = request.data;
        let mut offset = 0usize;
        let mut current_addr = request.address.as_u32();

        while offset < data.len() {
            // Calculate bytes remaining in current page
            let page_offset = (current_addr as usize) % PAGE_SIZE;
            let bytes_in_page = PAGE_SIZE - page_offset;
            let bytes_to_write = bytes_in_page.min(data.len() - offset);

            self.write_enable()?;

            let addr_bytes = self.addr_to_bytes(current_addr);
            let cmd_byte = if self.spec.capabilities.supports_4byte_addr {
                CMD_NOR_PAGE_PROGRAM_4B
            } else {
                CMD_NOR_PAGE_PROGRAM
            };

            let mut cmd = vec![cmd_byte];
            cmd.extend_from_slice(&addr_bytes);

            self.programmer.set_cs(true)?;
            self.programmer.spi_write(&cmd)?;
            self.programmer
                .spi_write(&data[offset..offset + bytes_to_write])?;
            self.programmer.set_cs(false)?;

            self.wait_ready()?;

            offset += bytes_to_write;
            current_addr += bytes_to_write as u32;

            on_progress(Progress::new(offset as u64, data.len() as u64));
        }

        if request.verify {
            let verify_req = ReadRequest {
                address: request.address,
                length: request.data.len() as u32,
                use_ecc: request.use_ecc,
                ignore_ecc_errors: request.ignore_ecc_errors,
                oob_mode: request.oob_mode,
                bad_block_strategy: request.bad_block_strategy,
                bbt: None,
                retry_count: request.retry_count,
            };
            let read_back = self.read(verify_req, &|_| {})?;
            if read_back != request.data {
                for (i, (&actual, &expected)) in
                    read_back.iter().zip(request.data.iter()).enumerate()
                {
                    if actual != expected {
                        return Err(Error::VerificationFailed {
                            address: request.address.as_u32() + i as u32,
                            expected,
                            actual,
                        });
                    }
                }
            }
        }

        Ok(())
    }

    fn erase(&mut self, request: EraseRequest, on_progress: &dyn Fn(Progress)) -> Result<()> {
        let block_size = self.spec.layout.block_size;
        let address = request.address.as_u32();

        if !address.is_multiple_of(block_size) {
            return Err(Error::InvalidParameter(
                "NOR erase address must be block-aligned".to_string(),
            ));
        }

        let total_blocks = request.length.div_ceil(block_size);

        for i in 0..total_blocks {
            let block_addr = address + (i * block_size);

            self.write_enable()?;

            let addr_bytes = self.addr_to_bytes(block_addr);
            let cmd_byte = if self.spec.capabilities.supports_4byte_addr {
                CMD_NOR_BLOCK_ERASE_64K_4B
            } else {
                CMD_NOR_BLOCK_ERASE_64K
            };

            let mut cmd = vec![cmd_byte];
            cmd.extend_from_slice(&addr_bytes);

            self.programmer.set_cs(true)?;
            self.programmer.spi_write(&cmd)?;
            self.programmer.set_cs(false)?;

            self.wait_ready()?;

            on_progress(Progress::new((i + 1) as u64, total_blocks as u64));
        }

        Ok(())
    }

    fn get_status(&mut self) -> Result<Vec<u8>> {
        Ok(vec![self.read_status()?])
    }

    fn set_status(&mut self, status: &[u8]) -> Result<()> {
        if status.is_empty() {
            return Ok(());
        }
        self.write_enable()?;
        let cmd = vec![NOR_CMD_WRSR, status[0]];
        self.programmer.spi_transfer(&cmd, &mut [])?;
        self.wait_ready()
    }
}