| // Copyright lowRISC contributors. |
| // Licensed under the Apache License, Version 2.0, see LICENSE for details. |
| // SPDX-License-Identifier: Apache-2.0 |
| |
| use anyhow::Result; |
| use std::time::Duration; |
| use structopt::StructOpt; |
| |
| use opentitanlib::app::TransportWrapper; |
| use opentitanlib::execute_test; |
| use opentitanlib::io::eeprom::AddressMode; |
| use opentitanlib::io::spi::{Target, Transfer}; |
| use opentitanlib::spiflash::{Sfdp, SpiFlash}; |
| use opentitanlib::test_utils::init::InitializeTest; |
| use opentitanlib::test_utils::spi_passthru::{ |
| ConfigJedecId, SfdpData, SpiFlashEraseSector, SpiFlashReadSfdp, SpiFlashWrite, StatusRegister, |
| UploadInfo, |
| }; |
| use opentitanlib::uart::console::UartConsole; |
| |
| const FLASH_STATUS_WIP: u32 = 0x01; |
| const FLASH_STATUS_WEL: u32 = 0x02; |
| const FLASH_STATUS_STD_BITS: u32 = FLASH_STATUS_WEL | FLASH_STATUS_WIP; |
| |
| #[derive(Debug, StructOpt)] |
| struct Opts { |
| #[structopt(flatten)] |
| init: InitializeTest, |
| |
| #[structopt( |
| long, parse(try_from_str=humantime::parse_duration), |
| default_value = "600s", |
| help = "Console receive timeout", |
| )] |
| timeout: Duration, |
| |
| #[structopt( |
| long, |
| default_value = "BOOTSTRAP", |
| help = "Name of the debugger's SPI interface" |
| )] |
| spi: String, |
| } |
| |
| fn test_jedec_id(opts: &Opts, transport: &TransportWrapper) -> Result<()> { |
| let uart = transport.uart("console")?; |
| let config = ConfigJedecId { |
| device_id: 0x1234, |
| manufacturer_id: 0x56, |
| continuation_code: 0x7f, |
| continuation_len: 3, |
| }; |
| config.execute(&*uart)?; |
| |
| let spi = transport.spi(&opts.spi)?; |
| let jedec_id = SpiFlash::read_jedec_id(&*spi, 16)?; |
| log::info!("jedec_id = {:x?}", jedec_id); |
| // Note: there is no specified bit pattern after the end of the JEDEC ID. |
| // OpenTitan returns zeros. Some devices return 0xFF or repeat the byte sequence. |
| assert_eq!( |
| jedec_id, |
| [ |
| 0x7f, 0x7f, 0x7f, // 3 continuation codes of 0x7F. |
| 0x56, // Manufacturer ID of 0x56. |
| 0x34, 0x12, // Device ID 0x1234 (in little endian order). |
| 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 // All zeros up to read length. |
| ] |
| ); |
| Ok(()) |
| } |
| |
| fn test_enter_exit_4b_mode(opts: &Opts, transport: &TransportWrapper) -> Result<()> { |
| let uart = transport.uart("console")?; |
| let spi = transport.spi(&opts.spi)?; |
| |
| log::info!("Entering 4B address mode"); |
| spi.run_transaction(&mut [Transfer::Write(&[SpiFlash::ENTER_4B])])?; |
| let sr = StatusRegister::read(&*uart)?; |
| assert!(sr.addr_4b, "expected to be in 4b mode"); |
| |
| log::info!("Exiting 4B address mode"); |
| spi.run_transaction(&mut [Transfer::Write(&[SpiFlash::EXIT_4B])])?; |
| let sr = StatusRegister::read(&*uart)?; |
| assert!(!sr.addr_4b, "expected to be in 3b mode"); |
| Ok(()) |
| } |
| |
| fn test_write_enable_disable(opts: &Opts, transport: &TransportWrapper) -> Result<()> { |
| let uart = transport.uart("console")?; |
| let spi = transport.spi(&opts.spi)?; |
| |
| log::info!("Sending WRITE_ENABLE"); |
| spi.run_transaction(&mut [Transfer::Write(&[SpiFlash::WRITE_ENABLE])])?; |
| let status = SpiFlash::read_status(&*spi)?; |
| let sr = StatusRegister::read(&*uart)?; |
| assert!( |
| status as u32 & FLASH_STATUS_WEL != 0, |
| "expected WEL set via read_status" |
| ); |
| assert!( |
| sr.status & FLASH_STATUS_WEL != 0, |
| "expected WEL set on the device" |
| ); |
| |
| log::info!("Sending WRITE_DISABLE"); |
| spi.run_transaction(&mut [Transfer::Write(&[SpiFlash::WRITE_DISABLE])])?; |
| let status = SpiFlash::read_status(&*spi)?; |
| let sr = StatusRegister::read(&*uart)?; |
| assert!( |
| status as u32 & FLASH_STATUS_WEL == 0, |
| "expected WEL clear via read_status" |
| ); |
| assert!( |
| sr.status & FLASH_STATUS_WEL == 0, |
| "expected WEL clear on the device" |
| ); |
| Ok(()) |
| } |
| |
| fn test_read_status_extended(opts: &Opts, transport: &TransportWrapper) -> Result<()> { |
| let uart = transport.uart("console")?; |
| let spi = transport.spi(&opts.spi)?; |
| |
| let sr = StatusRegister { |
| status: 0x5A55AA, |
| addr_4b: false, |
| }; |
| sr.write(&*uart)?; |
| // Note: because we're programming the flash_status register in firmware, |
| // we require one CS low-to-high transition to latch the values from the |
| // CSR into the spi device. We'd normally expect this type of register |
| // setup to be done at init time and that the first SPI transaction will |
| // be a READ_ID or READ_SFDP, thus latching the flash_status contents. |
| // |
| // In this test program, we simply issue a NOP transaction to the device. |
| spi.run_transaction(&mut [Transfer::Write(&[SpiFlash::NOP])])?; |
| let value = SpiFlash::read_status_ex(&*spi, None)?; |
| assert_eq!(value, sr.status); |
| Ok(()) |
| } |
| |
| fn read_sfdp(spi: &dyn Target, offset: u32) -> Result<Vec<u8>> { |
| let mut buf = vec![0u8; 256]; |
| spi.run_transaction(&mut [ |
| // READ_SFDP always takes a 3-byte address followed by a dummy |
| // byte regardless of address mode. |
| Transfer::Write(&[ |
| SpiFlash::READ_SFDP, |
| (offset >> 16) as u8, |
| (offset >> 8) as u8, |
| (offset >> 0) as u8, |
| 0, // Dummy byte. |
| ]), |
| Transfer::Read(&mut buf), |
| ])?; |
| Ok(buf) |
| } |
| |
| fn test_read_sfdp(opts: &Opts, transport: &TransportWrapper) -> Result<()> { |
| let uart = transport.uart("console")?; |
| let spi = transport.spi(&opts.spi)?; |
| |
| let sfdp = SfdpData { |
| data: (0..256).map(|x| x as u8).collect(), |
| }; |
| sfdp.write(&*uart)?; |
| |
| // Read and compare the whole SFDP buffer. |
| let buf = read_sfdp(&*spi, 0)?; |
| assert_eq!(buf, sfdp.data.as_slice()); |
| |
| // Test a read that would go beyond the length of the SFDP data. |
| // The observed behavior should be that the buffer recieved from |
| // the device should wrap around. |
| let buf = read_sfdp(&*spi, 0x30)?; |
| let data = sfdp.data.as_slice(); |
| assert_eq!(buf[0x00..0xd0], data[0x30..0x100]); |
| assert_eq!(buf[0xd0..0x100], data[0x00..0x30]); |
| |
| Ok(()) |
| } |
| |
| fn test_chip_erase(opts: &Opts, transport: &TransportWrapper) -> Result<()> { |
| let uart = transport.uart("console")?; |
| let spi = transport.spi(&opts.spi)?; |
| let flash = SpiFlash::default(); |
| let info = UploadInfo::execute(&*uart, || { |
| flash.chip_erase(&*spi)?; |
| Ok(()) |
| })?; |
| |
| assert_eq!(info.opcode, SpiFlash::CHIP_ERASE); |
| assert_eq!(info.has_address, false); |
| assert_eq!(info.data_len, 0); |
| assert_eq!( |
| info.flash_status & FLASH_STATUS_STD_BITS, |
| FLASH_STATUS_WEL | FLASH_STATUS_WIP |
| ); |
| Ok(()) |
| } |
| |
| fn test_sector_erase(opts: &Opts, transport: &TransportWrapper, address: u32) -> Result<()> { |
| let uart = transport.uart("console")?; |
| let spi = transport.spi(&opts.spi)?; |
| let mut flash = SpiFlash::default(); |
| // Double the flash size so we can test 3b and 4b addresses. |
| flash.size = 32 * 1024 * 1024; |
| |
| // Make sure we're in a mode appropriate for the address. |
| let mode = if address < 0x1000000 { |
| AddressMode::Mode3b |
| } else { |
| AddressMode::Mode4b |
| }; |
| flash.set_address_mode(&*spi, mode)?; |
| let info = UploadInfo::execute(&*uart, || { |
| flash.erase(&*spi, address, flash.erase_size)?; |
| Ok(()) |
| })?; |
| |
| assert_eq!(info.opcode, SpiFlash::SECTOR_ERASE); |
| assert_eq!(info.has_address, true); |
| assert_eq!(info.addr_4b, mode == AddressMode::Mode4b); |
| assert_eq!(info.address, address); |
| assert_eq!(info.data_len, 0); |
| assert_eq!( |
| info.flash_status & FLASH_STATUS_STD_BITS, |
| FLASH_STATUS_WEL | FLASH_STATUS_WIP |
| ); |
| Ok(()) |
| } |
| |
| fn test_page_program(opts: &Opts, transport: &TransportWrapper, address: u32) -> Result<()> { |
| let uart = transport.uart("console")?; |
| let spi = transport.spi(&opts.spi)?; |
| let data = (0..256).map(|x| x as u8).collect::<Vec<u8>>(); |
| let mut flash = SpiFlash::default(); |
| // Double the flash size so we can test 3b and 4b addresses. |
| flash.size = 32 * 1024 * 1024; |
| |
| // Make sure we're in a mode appropriate for the address. |
| let mode = if address < 0x1000000 { |
| AddressMode::Mode3b |
| } else { |
| AddressMode::Mode4b |
| }; |
| flash.set_address_mode(&*spi, mode)?; |
| let info = UploadInfo::execute(&*uart, || { |
| flash.program(&*spi, address, &data)?; |
| Ok(()) |
| })?; |
| |
| assert_eq!(info.opcode, SpiFlash::PAGE_PROGRAM); |
| assert_eq!(info.has_address, true); |
| assert_eq!(info.addr_4b, mode == AddressMode::Mode4b); |
| assert_eq!(info.address, address); |
| assert_eq!(info.data_len as usize, data.len()); |
| assert_eq!(info.data.as_slice(), data.as_slice()); |
| assert_eq!( |
| info.flash_status & FLASH_STATUS_STD_BITS, |
| FLASH_STATUS_WEL | FLASH_STATUS_WIP |
| ); |
| Ok(()) |
| } |
| |
| fn test_write_status(opts: &Opts, transport: &TransportWrapper, opcode: u8) -> Result<()> { |
| let uart = transport.uart("console")?; |
| let spi = transport.spi(&opts.spi)?; |
| let info = UploadInfo::execute(&*uart, || { |
| spi.run_transaction(&mut [Transfer::Write(&[opcode])])?; |
| Ok(()) |
| })?; |
| |
| assert_eq!(info.opcode, opcode); |
| assert_eq!(info.has_address, false); |
| assert_eq!(info.data_len, 0); |
| assert_eq!(info.flash_status & FLASH_STATUS_STD_BITS, FLASH_STATUS_WIP); |
| Ok(()) |
| } |
| |
| fn test_read_flash(opts: &Opts, transport: &TransportWrapper) -> Result<()> { |
| let uart = transport.uart("console")?; |
| let spi = transport.spi(&opts.spi)?; |
| |
| let sfdp_read = SpiFlashReadSfdp { |
| address: 0u32, |
| length: 256u16, |
| }; |
| let sfdp_data = sfdp_read.execute(&*uart)?; |
| let sfdp = Sfdp::try_from(sfdp_data.data.as_slice())?; |
| let spi_flash = SpiFlash::from_sfdp(sfdp); |
| |
| let address = 0x1000u32; |
| let erase_op = SpiFlashEraseSector { |
| address: address, |
| addr4b: false, |
| }; |
| erase_op.execute(&*uart)?; |
| |
| let write_op = SpiFlashWrite { |
| address: address, |
| addr4b: false, |
| data: (0..256).map(|x| x as u8).collect(), |
| length: 256, |
| }; |
| write_op.execute(&*uart)?; |
| |
| let mut read_data = vec![0; 256]; |
| spi_flash.read(&*spi, address, &mut read_data)?; |
| assert_eq!(read_data.as_slice(), write_op.data.as_slice()); |
| Ok(()) |
| } |
| |
| fn main() -> Result<()> { |
| let opts = Opts::from_args(); |
| opts.init.init_logging(); |
| let transport = opts.init.init_target()?; |
| |
| let uart = transport.uart("console")?; |
| uart.set_flow_control(true)?; |
| let _ = UartConsole::wait_for(&*uart, r"Running [^\r\n]*", opts.timeout)?; |
| uart.clear_rx_buffer()?; |
| |
| execute_test!(test_read_flash, &opts, &transport); |
| execute_test!(test_jedec_id, &opts, &transport); |
| execute_test!(test_enter_exit_4b_mode, &opts, &transport); |
| execute_test!(test_write_enable_disable, &opts, &transport); |
| execute_test!(test_read_status_extended, &opts, &transport); |
| execute_test!(test_read_sfdp, &opts, &transport); |
| execute_test!(test_chip_erase, &opts, &transport); |
| execute_test!(test_sector_erase, &opts, &transport, 0x0000_4000); |
| execute_test!(test_sector_erase, &opts, &transport, 0x0100_4000); |
| execute_test!(test_page_program, &opts, &transport, 0x0000_4000); |
| execute_test!(test_page_program, &opts, &transport, 0x0100_4000); |
| execute_test!(test_write_status, &opts, &transport, SpiFlash::WRITE_STATUS); |
| execute_test!( |
| test_write_status, |
| &opts, |
| &transport, |
| SpiFlash::WRITE_STATUS2 |
| ); |
| execute_test!( |
| test_write_status, |
| &opts, |
| &transport, |
| SpiFlash::WRITE_STATUS3 |
| ); |
| Ok(()) |
| } |