vos/ambiq-hal-sys/ambiq-sparkfun-sdk/devices/am_devices_spiflash.c

438 lines
15 KiB
C
Raw Permalink Normal View History

2022-10-24 06:45:43 +00:00
//*****************************************************************************
//
//! @file am_devices_spiflash.c
//!
//! @brief Generic spiflash driver.
//
//*****************************************************************************
//*****************************************************************************
//
// Copyright (c) 2020, Ambiq Micro
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
// contributors may be used to endorse or promote products derived from this
// software without specific prior written permission.
//
// Third party software included in this distribution is subject to the
// additional license terms as defined in the /docs/licenses directory.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
//
// This is part of revision 2.4.2 of the AmbiqSuite Development Package.
//
//*****************************************************************************
#include "am_mcu_apollo.h"
#include "am_devices_spiflash.h"
//*****************************************************************************
//
// Global variables.
//
//*****************************************************************************
am_devices_spiflash_t *g_psIOMSettings;
//*****************************************************************************
//
//! @brief Initialize the spiflash driver.
//!
//! @param psIOMSettings - IOM device structure describing the target spiflash.
//! @param pfnWriteFunc - Function to use for spi writes.
//! @param pfnReadFunc - Function to use for spi reads.
//!
//! This function should be called before any other am_devices_spiflash
//! functions. It is used to set tell the other functions how to communicate
//! with the external spiflash hardware.
//!
//! The \e pfnWriteFunc and \e pfnReadFunc variables may be used to provide
//! alternate implementations of SPI write and read functions respectively. If
//! they are left set to 0, the default functions am_hal_iom_spi_write() and
//! am_hal_iom_spi_read() will be used.
//!
//! @return None.
//
//*****************************************************************************
void
am_devices_spiflash_init(am_devices_spiflash_t *psIOMSettings)
{
//
// Initialize the IOM settings for the spiflash.
//
g_psIOMSettings = psIOMSettings;
}
//*****************************************************************************
//
//! @brief Reads the current status of the external flash
//!
//! @param ui32DeviceNumber - Device number of the external flash
//!
//! This function reads the status register of the external flash, and returns
//! the result as an 8-bit unsigned integer value. The processor will block
//! during the data transfer process, but will return as soon as the status
//! register had been read.
//!
//! Macro definitions for interpreting the contents of the status register are
//! included in the header file.
//!
//! @return 8-bit status register contents
//
//*****************************************************************************
uint8_t
am_devices_spiflash_status(void)
{
am_hal_iom_buffer(1) psResponse;
//
// Send the command and read the response.
//
am_hal_iom_spi_read(g_psIOMSettings->ui32IOMModule,
g_psIOMSettings->ui32ChipSelect,
psResponse.words, 1,
AM_HAL_IOM_OFFSET(AM_DEVICES_SPIFLASH_RDRSR));
//
// Return the status read from the external flash.
//
return psResponse.bytes[0];
}
//*****************************************************************************
//
//! @brief Reads the ID register for the external flash
//!
//! @param ui32DeviceNumber - Device number of the external flash
//!
//! This function reads the ID register of the external flash, and returns the
//! result as a 32-bit unsigned integer value. The processor will block during
//! the data transfer process, but will return as soon as the ID register had
//! been read. The ID contents for this flash only contains 24 bits of data, so
//! the result will be stored in the lower 24 bits of the return value.
//!
//! @return 32-bit ID register contents
//
//*****************************************************************************
uint32_t
am_devices_spiflash_id(void)
{
am_hal_iom_buffer(3) psResponse;
//
// Send a command to read the ID register in the external flash.
//
am_hal_iom_spi_read(g_psIOMSettings->ui32IOMModule,
g_psIOMSettings->ui32ChipSelect,
psResponse.words, 3,
AM_HAL_IOM_OFFSET(AM_DEVICES_SPIFLASH_RDID));
//
// Return the ID
//
return psResponse.words[0] & 0x00FFFFFF;
}
//*****************************************************************************
//
//! @brief Reads the contents of the external flash into a buffer.
//!
//! @param ui32DeviceNumber - Device number of the external flash
//! @param pui8RxBuffer - Buffer to store the received data from the flash
//! @param ui32ReadAddress - Address of desired data in external flash
//! @param ui32NumBytes - Number of bytes to read from external flash
//!
//! This function reads the external flash at the provided address and stores
//! the received data into the provided buffer location. This function will
//! only store ui32NumBytes worth of data.
//
//*****************************************************************************
void
am_devices_spiflash_read(uint8_t *pui8RxBuffer, uint32_t ui32ReadAddress,
uint32_t ui32NumBytes)
{
uint32_t i, ui32BytesRemaining, ui32TransferSize, ui32CurrentReadAddress;
uint8_t *pui8Dest;
uint32_t pui32WriteBuffer[1];
uint8_t *pui8WritePtr;
uint32_t pui32ReadBuffer[16];
uint8_t *pui8ReadPtr;
pui8WritePtr = (uint8_t *)(&pui32WriteBuffer);
pui8ReadPtr = (uint8_t *)(&pui32ReadBuffer);
//
// Set the total number of bytes,and the starting transfer destination.
//
ui32BytesRemaining = ui32NumBytes;
pui8Dest = pui8RxBuffer;
ui32CurrentReadAddress = ui32ReadAddress;
while ( ui32BytesRemaining )
{
//
// Set the transfer size to either 64, or the number of remaining
// bytes, whichever is smaller.
//
ui32TransferSize = ui32BytesRemaining > 64 ? 64 : ui32BytesRemaining;
pui8WritePtr[0] = AM_DEVICES_SPIFLASH_READ;
pui8WritePtr[1] = (ui32CurrentReadAddress & 0x00FF0000) >> 16;
pui8WritePtr[2] = (ui32CurrentReadAddress & 0x0000FF00) >> 8;
pui8WritePtr[3] = ui32CurrentReadAddress & 0x000000FF;
//
// Send the read command.
//
am_hal_iom_spi_write(g_psIOMSettings->ui32IOMModule,
g_psIOMSettings->ui32ChipSelect,
pui32WriteBuffer, 4,
AM_HAL_IOM_CS_LOW | AM_HAL_IOM_RAW);
am_hal_iom_spi_read(g_psIOMSettings->ui32IOMModule,
g_psIOMSettings->ui32ChipSelect, pui32ReadBuffer,
ui32TransferSize, AM_HAL_IOM_RAW);
//
// Copy the received bytes over to the RxBuffer
//
for ( i = 0; i < ui32TransferSize; i++ )
{
pui8Dest[i] = pui8ReadPtr[i];
}
//
// Update the number of bytes remaining and the destination.
//
ui32BytesRemaining -= ui32TransferSize;
pui8Dest += ui32TransferSize;
ui32CurrentReadAddress += ui32TransferSize;
}
}
//*****************************************************************************
//
//! @brief Programs the given range of flash addresses.
//!
//! @param ui32DeviceNumber - Device number of the external flash
//! @param pui8TxBuffer - Buffer to write the external flash data from
//! @param ui32WriteAddress - Address to write to in the external flash
//! @param ui32NumBytes - Number of bytes to write to the external flash
//!
//! This function uses the data in the provided pui8TxBuffer and copies it to
//! the external flash at the address given by ui32WriteAddress. It will copy
//! exactly ui32NumBytes of data from the original pui8TxBuffer pointer. The
//! user is responsible for ensuring that they do not overflow the target flash
//! memory or underflow the pui8TxBuffer array
//
//*****************************************************************************
void
am_devices_spiflash_write(uint8_t *pui8TxBuffer, uint32_t ui32WriteAddress,
uint32_t ui32NumBytes)
{
uint32_t i;
uint32_t ui32DestAddress;
uint32_t ui32BytesRemaining;
uint32_t ui32TransferSize;
uint8_t *pui8Source;
am_hal_iom_buffer(1) psEnableCommand;
am_hal_iom_buffer(64) psWriteCommand;
//
// Prepare the command for write-enable.
//
psEnableCommand.bytes[0] = AM_DEVICES_SPIFLASH_WREN;
//
// Set the total number of bytes, and the starting transfer destination.
//
ui32BytesRemaining = ui32NumBytes;
pui8Source = pui8TxBuffer;
ui32DestAddress = ui32WriteAddress;
while ( ui32BytesRemaining )
{
//
// Set up a write command to hit the beginning of the next "chunk" of
// flash.
//
psWriteCommand.bytes[0] = AM_DEVICES_SPIFLASH_PP;
psWriteCommand.bytes[1] = (ui32DestAddress & 0x00FF0000) >> 16;
psWriteCommand.bytes[2] = (ui32DestAddress & 0x0000FF00) >> 8;
psWriteCommand.bytes[3] = ui32DestAddress & 0x000000FF;
//
// Set the transfer size to either 32, or the number of remaining
// bytes, whichever is smaller.
//
ui32TransferSize = ui32BytesRemaining > 32 ? 32 : ui32BytesRemaining;
//
// Fill the rest of the command buffer with the data that we actually
// want to write to flash.
//
for ( i = 0; i < ui32TransferSize; i++ )
{
psWriteCommand.bytes[4 + i] = pui8Source[i];
}
//
// Send the write-enable command to prepare the external flash for
// program operations, and wait for the write-enable latch to be set in
// the status register.
//
am_hal_iom_spi_write(g_psIOMSettings->ui32IOMModule,
g_psIOMSettings->ui32ChipSelect,
psEnableCommand.words,
1, AM_HAL_IOM_RAW);
while ( !(am_devices_spiflash_status() & AM_DEVICES_SPIFLASH_WEL) );
//
// Send the write command.
//
am_hal_iom_spi_write(g_psIOMSettings->ui32IOMModule,
g_psIOMSettings->ui32ChipSelect,
psWriteCommand.words,
(ui32TransferSize + 4), AM_HAL_IOM_RAW);
//
// Wait for status to indicate that the write is no longer in progress.
//
while ( am_devices_spiflash_status() & AM_DEVICES_SPIFLASH_WIP );
//
// Update the number of bytes remaining, as well as the source and
// destination pointers
//
ui32BytesRemaining -= ui32TransferSize;
pui8Source += ui32TransferSize;
ui32DestAddress += ui32TransferSize;
}
return;
}
//*****************************************************************************
//
//! @brief Erases the entire contents of the external flash
//!
//! @param ui32DeviceNumber - Device number of the external flash
//!
//! This function uses the "Bulk Erase" instruction to erase the entire
//! contents of the external flash.
//
//*****************************************************************************
void
am_devices_spiflash_mass_erase(void)
{
am_hal_iom_buffer(1) psCommand;
//
// Send the write-enable command to prepare the external flash for program
// operations.
//
psCommand.bytes[0] = AM_DEVICES_SPIFLASH_WREN;
am_hal_iom_spi_write(g_psIOMSettings->ui32IOMModule,
g_psIOMSettings->ui32ChipSelect,
psCommand.words, 1, AM_HAL_IOM_RAW);
//
// Wait for the write enable latch status bit.
//
while ( !(am_devices_spiflash_status() & AM_DEVICES_SPIFLASH_WEL) );
//
// Send the bulk erase command.
//
psCommand.bytes[0] = AM_DEVICES_SPIFLASH_BE;
am_hal_iom_spi_write(g_psIOMSettings->ui32IOMModule,
g_psIOMSettings->ui32ChipSelect,
psCommand.words, 1, AM_HAL_IOM_RAW);
//
// Wait for status to indicate that the write is no longer in progress.
//
while ( am_devices_spiflash_status() & AM_DEVICES_SPIFLASH_WIP );
return;
}
//*****************************************************************************
//
//! @brief Erases the contents of a single sector of flash
//!
//! @param ui32DeviceNumber - Device number of the external flash
//! @param ui32SectorAddress - Address to erase in the external flash
//!
//! This function erases a single sector of the external flash as specified by
//! ui32EraseAddress. The entire sector where ui32EraseAddress will
//! be erased.
//
//*****************************************************************************
void
am_devices_spiflash_sector_erase(uint32_t ui32SectorAddress)
{
am_hal_iom_buffer(4) psCommand;
//
// Send the write-enable command to prepare the external flash for program
// operations.
//
psCommand.bytes[0] = AM_DEVICES_SPIFLASH_WREN;
am_hal_iom_spi_write(g_psIOMSettings->ui32IOMModule,
g_psIOMSettings->ui32ChipSelect,
psCommand.words, 1, AM_HAL_IOM_RAW);
//
// Wait for the write enable latch status bit.
//
while ( !(am_devices_spiflash_status() & AM_DEVICES_SPIFLASH_WEL) );
//
// Prepare the sector erase command, followed by the three-byte external
// flash address.
//
psCommand.bytes[0] = AM_DEVICES_SPIFLASH_SE;
psCommand.bytes[1] = (ui32SectorAddress & 0x00FF0000) >> 16;
psCommand.bytes[2] = (ui32SectorAddress & 0x0000FF00) >> 8;
psCommand.bytes[3] = ui32SectorAddress & 0x000000FF;
//
// Send the command to erase the desired sector.
//
am_hal_iom_spi_write(g_psIOMSettings->ui32IOMModule,
g_psIOMSettings->ui32ChipSelect,
psCommand.words, 4, AM_HAL_IOM_RAW);
//
// Wait for status to indicate that the write is no longer in progress.
//
while ( am_devices_spiflash_status() & AM_DEVICES_SPIFLASH_WIP );
return;
}