947 lines
30 KiB
C
Raw Normal View History

2017-08-30 12:18:28 +08:00
/**
* \file
*
* \brief SAM Non-Volatile Memory driver
*
* Copyright (C) 2012-2016 Atmel Corporation. All rights reserved.
*
* \asf_license_start
*
* \page License
*
* 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. The name of Atmel may not be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* 4. This software may only be redistributed and used in connection with an
* Atmel microcontroller product.
*
* THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
* EXPRESSLY AND SPECIFICALLY DISCLAIMED. IN NO EVENT SHALL ATMEL 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.
*
* \asf_license_stop
*
*/
/*
* Support and FAQ: visit <a href="http://www.atmel.com/design-support/">Atmel Support</a>
*/
#ifndef NVM_H_INCLUDED
#define NVM_H_INCLUDED
/**
* \defgroup asfdoc_sam0_nvm_group SAM Non-Volatile Memory (NVM) Driver
*
* This driver for Atmel&reg; | SMART ARM&reg;-based microcontrollers provides
* an interface for the configuration and management of non-volatile memories
* within the device, for partitioning, erasing, reading, and writing of data.
*
* The following peripheral is used by this module:
* - NVM (Non-Volatile Memory)
*
* The following devices can use this module:
* - Atmel | SMART SAM D20/D21
* - Atmel | SMART SAM R21
* - Atmel | SMART SAM D09/D10/D11
* - Atmel | SMART SAM L21/L22
* - Atmel | SMART SAM DA1
* - Atmel | SMART SAM C20/C21
* - Atmel | SMART SAM HA1
* - Atmel | SMART SAM R30
*
* The outline of this documentation is as follows:
* - \ref asfdoc_sam0_nvm_prerequisites
* - \ref asfdoc_sam0_nvm_module_overview
* - \ref asfdoc_sam0_nvm_special_considerations
* - \ref asfdoc_sam0_nvm_extra_info
* - \ref asfdoc_sam0_nvm_examples
* - \ref asfdoc_sam0_nvm_api_overview
*
*
* \section asfdoc_sam0_nvm_prerequisites Prerequisites
*
* There are no prerequisites for this module.
*
*
* \section asfdoc_sam0_nvm_module_overview Module Overview
*
* The Non-Volatile Memory (NVM) module provides an interface to the device's
* Non-Volatile Memory controller, so that memory pages can be written, read,
* erased, and reconfigured in a standardized manner.
*
* \subsection asfdoc_sam0_nvm_features Driver Feature Macro Definition
* <table>
* <tr>
* <th>Driver feature macro</th>
* <th>Supported devices</th>
* </tr>
* <tr>
* <td>FEATURE_NVM_RWWEE</td>
* <td>SAM L21/L22, SAM D21-64K, SAM DA1, SAM C20/C21, SAM R30</td>
* </tr>
* <tr>
* <td>FEATURE_BOD12</td>
* <td>SAM L21, SAMR30</td>
* </tr>
* </table>
* \note The specific features are only available in the driver when the
* selected device supports those features.
*
* \subsection asfdoc_sam0_nvm_module_overview_regions Memory Regions
* The NVM memory space of the SAM devices is divided into two sections:
* a Main Array section, and an Auxiliary space section. The Main Array space
* can be configured to have an (emulated) EEPROM and/or boot loader section.
* The memory layout with the EEPROM and bootloader partitions is shown in
* \ref asfdoc_sam0_nvm_module_mem_layout "the figure below".
*
* \anchor asfdoc_sam0_nvm_module_mem_layout
* \dot
* digraph memory_layout {
* size="5,5"
* node [shape=plaintext, fontname=arial]
* memory [label=<
* <table border="0" cellborder="1" cellspacing="0" >
* <tr>
* <td align="right" border="0"> End of NVM Memory </td>
* <td rowspan="3" align="center"> Reserved EEPROM Section </td>
* </tr>
* <tr>
* <td align="right" border="0"> </td>
* </tr>
* <tr>
* <td align="right" border="0"> Start of EEPROM Memory </td>
* </tr>
* <tr>
* <td align="right" border="0"> End of Application Memory </td>
* <td rowspan="3" align="center"> Application Section </td>
* </tr>
* <tr>
* <td height="300" align="right" border="0"> </td>
* </tr>
* <tr>
* <td align="right" border="0"> Start of Application Memory </td>
* </tr>
* <tr>
* <td align="right" border="0"> End of Bootloader Memory </td>
* <td rowspan="3" align="center"> BOOT Section </td>
* </tr>
* <tr>
* <td align="right" border="0"> </td>
* </tr>
* <tr>
* <td align="right" border="0"> Start of NVM Memory</td>
* </tr>
* </table>
* >]
* }
* \enddot
*
* The Main Array is divided into rows and pages, where each row contains four
* pages. The size of each page may vary from 8-1024 bytes dependent of the
* device. Device specific parameters such as the page size and total number of
* pages in the NVM memory space are available via the \ref nvm_get_parameters()
* function.
*
* An NVM page number and address can be computed via the following equations:
*
* \f[ PageNum = (RowNum \times 4) + PagePosInRow \f]
* \f[ PageAddr = PageNum \times PageSize \f]
*
* \ref asfdoc_sam0_nvm_module_row_layout "The figure below" shows an example
* of the memory page and address values associated with logical row 7 of the
* NVM memory space.
*
* \anchor asfdoc_sam0_nvm_module_row_layout
* \dot
* digraph row_layout {
* size="4,4"
* node [shape=plaintext, fontname=arial]
* row [label=<
* <table border="0" cellborder="1" cellspacing="0">
* <tr>
* <td align="right" border ="0"> Row 0x07 </td>
* <td > Page 0x1F </td>
* <td > Page 0x1E </td>
* <td > Page 0x1D </td>
* <td > Page 0x1C </td>
* </tr>
* <tr>
* <td align="right" border ="0"> Address </td>
* <td border="0"> 0x7C0 </td>
* <td border="0"> 0x780 </td>
* <td border="0"> 0x740 </td>
* <td border="0"> 0x700 </td>
* </tr>
* </table>
* >]
* }
* \enddot
*
* \subsection asfdoc_sam0_nvm_module_overview_locking_regions Region Lock Bits
* As mentioned in \ref asfdoc_sam0_nvm_module_overview_regions, the main
* block of the NVM memory is divided into a number of individually addressable
* pages. These pages are grouped into 16 equal sized regions, where each region
* can be locked separately issuing an \ref NVM_COMMAND_LOCK_REGION command or
* by writing the LOCK bits in the User Row. Rows reserved for the EEPROM
* section are not affected by the lock bits or commands.
*
* \note By using the \ref NVM_COMMAND_LOCK_REGION or
* \ref NVM_COMMAND_UNLOCK_REGION commands the settings will remain in
* effect until the next device reset. By changing the default lock
* setting for the regions, the auxiliary space must to be written,
* however the adjusted configuration will not take effect until the next
* device reset.
*
* \note If the \ref asfdoc_sam0_nvm_special_consideration_security_bit is
* set, the auxiliary space cannot be written to. Clearing of the security
* bit can only be performed by a full chip erase.
*
* \subsection asfdoc_sam0_nvm_module_overview_sub_rw Read/Write
* Reading from the NVM memory can be performed using direct addressing into the
* NVM memory space, or by calling the \ref nvm_read_buffer() function.
*
* Writing to the NVM memory must be performed by the \ref nvm_write_buffer()
* function - additionally, a manual page program command must be issued if
* the NVM controller is configured in manual page writing mode.
*
* Before a page can be updated, the associated NVM memory row must be erased
* first via the \ref nvm_erase_row() function. Writing to a non-erased page
* will result in corrupt data being stored in the NVM memory space.
*
* \section asfdoc_sam0_nvm_special_considerations Special Considerations
*
* \subsection asfdoc_sam0_nvm_special_consideration_pageerase Page Erasure
* The granularity of an erase is per row, while the granularity of a write is
* per page. Thus, if the user application is modifying only one page of a row,
* the remaining pages in the row must be buffered and the row erased, as an
* erase is mandatory before writing to a page.
*
* \subsection asfdoc_sam0_nvm_special_consideration_clocks Clocks
* The user must ensure that the driver is configured with a proper number of
* wait states when the CPU is running at high frequencies.
*
* \subsection asfdoc_sam0_nvm_special_consideration_security_bit Security Bit
* The User Row in the Auxiliary Space cannot be read or written when
* the Security Bit is set. The Security Bit can be set by using passing
* \ref NVM_COMMAND_SET_SECURITY_BIT to the \ref nvm_execute_command() function,
* or it will be set if one tries to access a locked region. See
* \ref asfdoc_sam0_nvm_module_overview_locking_regions.
*
* The Security Bit can only be cleared by performing a chip erase.
*
*
* \section asfdoc_sam0_nvm_extra_info Extra Information
*
* For extra information, see \ref asfdoc_sam0_nvm_extra. This includes:
* - \ref asfdoc_sam0_nvm_extra_acronyms
* - \ref asfdoc_sam0_nvm_extra_dependencies
* - \ref asfdoc_sam0_nvm_extra_errata
* - \ref asfdoc_sam0_nvm_extra_history
*
*
* \section asfdoc_sam0_nvm_examples Examples
*
* For a list of examples related to this driver, see
* \ref asfdoc_sam0_nvm_exqsg.
*
*
* \section asfdoc_sam0_nvm_api_overview API Overview
* @{
*/
#include <compiler.h>
#include <status_codes.h>
#include <stdbool.h>
#ifdef __cplusplus
extern "C" {
#endif
/* Define SAMD21-64K devices */
#if defined(SAMD21E15L) || defined(SAMD21E16L) || defined(__SAMD21E15L__) || defined(__SAMD21E16L__) \
|| defined(SAMD21E15B) || defined(SAMD21E16B) || defined(__SAMD21E15B__) || defined(__SAMD21E16B__) \
|| defined(SAMD21E15BU) || defined(SAMD21E16BU) || defined(__SAMD21E15BU__) || defined(__SAMD21E16BU__) \
|| defined(SAMD21G15L) || defined(SAMD21G16L) || defined(__SAMD21G15L__) || defined(__SAMD21G16L__) \
|| defined(SAMD21G15B) || defined(SAMD21G16B) || defined(__SAMD21G15B__) || defined(__SAMD21G16B__) \
|| defined(SAMD21J15B) || defined(SAMD21J16B) || defined(__SAMD21J15B__) || defined(__SAMD21J16B__)
# define SAMD21_64K
#endif
/**
* \name Driver Feature Definition
*
* Define NVM features set according to the different device families.
* @{
*/
#if (SAML21) || (SAML22) || (SAMDA1) || (SAMC20) || (SAMC21) || (SAMR30) || defined(SAMD21_64K) || (SAMHA1) \
|| defined(__DOXYGEN__)
/** Read while write EEPROM emulation feature. */
# define FEATURE_NVM_RWWEE
#endif
#if (SAML21) || (SAMR30) || defined(__DOXYGEN__)
/** Brown-out detector internal to the voltage regulator for VDDCORE. */
#define FEATURE_BOD12
#endif
/*@}*/
#if !defined(__DOXYGEN__)
/**
* \brief Mask for the error flags in the status register.
*/
# define NVM_ERRORS_MASK (NVMCTRL_STATUS_PROGE | \
NVMCTRL_STATUS_LOCKE | \
NVMCTRL_STATUS_NVME)
#endif
/**
* \brief NVM error flags.
*
* Possible NVM controller error codes, which can be returned by the NVM
* controller after a command is issued.
*/
enum nvm_error {
/** No errors */
NVM_ERROR_NONE = 0,
/** Lock error, a locked region was attempted accessed */
NVM_ERROR_LOCK = NVMCTRL_STATUS_NVME | NVMCTRL_STATUS_LOCKE,
/** Program error, invalid command was executed */
NVM_ERROR_PROG = NVMCTRL_STATUS_NVME | NVMCTRL_STATUS_PROGE,
};
/**
* \brief NVM controller commands.
*/
enum nvm_command {
/** Erases the addressed memory row */
NVM_COMMAND_ERASE_ROW = NVMCTRL_CTRLA_CMD_ER,
/** Write the contents of the page buffer to the addressed memory page */
NVM_COMMAND_WRITE_PAGE = NVMCTRL_CTRLA_CMD_WP,
/** Erases the addressed auxiliary memory row.
*
* \note This command can only be given when the security bit is not set.
*/
NVM_COMMAND_ERASE_AUX_ROW = NVMCTRL_CTRLA_CMD_EAR,
/** Write the contents of the page buffer to the addressed auxiliary memory
* row.
*
* \note This command can only be given when the security bit is not set.
*/
NVM_COMMAND_WRITE_AUX_ROW = NVMCTRL_CTRLA_CMD_WAP,
/** Locks the addressed memory region, preventing further modifications
* until the region is unlocked or the device is erased
*/
NVM_COMMAND_LOCK_REGION = NVMCTRL_CTRLA_CMD_LR,
/** Unlocks the addressed memory region, allowing the region contents to be
* modified
*/
NVM_COMMAND_UNLOCK_REGION = NVMCTRL_CTRLA_CMD_UR,
/** Clears the page buffer of the NVM controller, resetting the contents to
* all zero values
*/
NVM_COMMAND_PAGE_BUFFER_CLEAR = NVMCTRL_CTRLA_CMD_PBC,
/** Sets the device security bit, disallowing the changing of lock bits and
* auxiliary row data until a chip erase has been performed
*/
NVM_COMMAND_SET_SECURITY_BIT = NVMCTRL_CTRLA_CMD_SSB,
/** Enter power reduction mode in the NVM controller to reduce the power
* consumption of the system
*/
NVM_COMMAND_ENTER_LOW_POWER_MODE = NVMCTRL_CTRLA_CMD_SPRM,
/** Exit power reduction mode in the NVM controller to allow other NVM
* commands to be issued
*/
NVM_COMMAND_EXIT_LOW_POWER_MODE = NVMCTRL_CTRLA_CMD_CPRM,
#ifdef FEATURE_NVM_RWWEE
/** Read while write (RWW) EEPROM area erase row */
NVM_COMMAND_RWWEE_ERASE_ROW = NVMCTRL_CTRLA_CMD_RWWEEER,
/** RWW EEPROM write page */
NVM_COMMAND_RWWEE_WRITE_PAGE = NVMCTRL_CTRLA_CMD_RWWEEWP,
#endif
};
/**
* \brief NVM controller power reduction mode configurations.
*
* Power reduction modes of the NVM controller, to conserve power while the
* device is in sleep.
*/
enum nvm_sleep_power_mode {
/** NVM controller exits low-power mode on first access after sleep */
NVM_SLEEP_POWER_MODE_WAKEONACCESS = NVMCTRL_CTRLB_SLEEPPRM_WAKEONACCESS_Val,
/** NVM controller exits low-power mode when the device exits sleep mode */
NVM_SLEEP_POWER_MODE_WAKEUPINSTANT = NVMCTRL_CTRLB_SLEEPPRM_WAKEUPINSTANT_Val,
/** Power reduction mode in the NVM controller disabled */
NVM_SLEEP_POWER_MODE_ALWAYS_AWAKE = NVMCTRL_CTRLB_SLEEPPRM_DISABLED_Val,
};
/**
* \brief NVM controller cache readmode configuration.
*
* Control how the NVM cache prefetch data from flash.
*
*/
enum nvm_cache_readmode {
/** The NVM Controller (cache system) does not insert wait states on
* a cache miss. Gives the best system performance.
*/
NVM_CACHE_READMODE_NO_MISS_PENALTY,
/** Reduces power consumption of the cache system, but inserts a
* wait state each time there is a cache miss
*/
NVM_CACHE_READMODE_LOW_POWER,
/** The cache system ensures that a cache hit or miss takes the same
* amount of time, determined by the number of programmed flash
* wait states
*/
NVM_CACHE_READMODE_DETERMINISTIC,
};
/**
* \brief NVM controller configuration structure.
*
* Configuration structure for the NVM controller within the device.
*/
struct nvm_config {
/** Power reduction mode during device sleep */
enum nvm_sleep_power_mode sleep_power_mode;
/** Manual write mode; if enabled, pages loaded into the NVM buffer will
* not be written until a separate write command is issued. If disabled,
* writing to the last byte in the NVM page buffer will trigger an automatic
* write.
*
* \note If a partial page is to be written, a manual write command must be
* executed in either mode.
*/
bool manual_page_write;
/** Number of wait states to insert when reading from flash, to prevent
* invalid data from being read at high clock frequencies
*/
uint8_t wait_states;
/**
* Setting this to true will disable the pre-fetch cache in front of the
* NVM controller
*/
bool disable_cache;
#if (SAMC20) || (SAMC21)
/**
* Setting this to true will disable the pre-fetch RWW cache in front of the
* NVM controller.
* If RWW cache is enabled, NVM cache will also be enabled.
*/
bool disable_rww_cache;
#endif
/**
* Select the mode for how the cache will pre-fetch data from the flash
*/
enum nvm_cache_readmode cache_readmode;
};
/**
* \brief NVM memory parameter structure.
*
* Structure containing the memory layout parameters of the NVM module.
*/
struct nvm_parameters {
/** Number of bytes per page */
uint8_t page_size;
/** Number of pages in the main array */
uint16_t nvm_number_of_pages;
/** Size of the emulated EEPROM memory section configured in the NVM
* auxiliary memory space */
uint32_t eeprom_number_of_pages;
/** Size of the Bootloader memory section configured in the NVM auxiliary
* memory space */
uint32_t bootloader_number_of_pages;
#ifdef FEATURE_NVM_RWWEE
/** Number of pages in read while write EEPROM (RWWEE) emulation area */
uint16_t rww_eeprom_number_of_pages;
#endif
};
/**
* \brief Bootloader size.
*
* Available bootloader protection sizes in kilobytes.
*
*/
enum nvm_bootloader_size {
/** Boot Loader Size is 32768 bytes */
NVM_BOOTLOADER_SIZE_128,
/** Boot Loader Size is 16384 bytes */
NVM_BOOTLOADER_SIZE_64,
/** Boot Loader Size is 8192 bytes */
NVM_BOOTLOADER_SIZE_32,
/** Boot Loader Size is 4096 bytes */
NVM_BOOTLOADER_SIZE_16,
/** Boot Loader Size is 2048 bytes */
NVM_BOOTLOADER_SIZE_8,
/** Boot Loader Size is 1024 bytes */
NVM_BOOTLOADER_SIZE_4,
/** Boot Loader Size is 512 bytes */
NVM_BOOTLOADER_SIZE_2,
/** Boot Loader Size is 0 bytes */
NVM_BOOTLOADER_SIZE_0,
};
/**
* \brief EEPROM emulator size.
*
* Available space in flash dedicated for EEPROM emulator in bytes.
*
*/
enum nvm_eeprom_emulator_size {
/** EEPROM Size for EEPROM emulation is 16384 bytes */
NVM_EEPROM_EMULATOR_SIZE_16384,
/** EEPROM Size for EEPROM emulation is 8192 bytes */
NVM_EEPROM_EMULATOR_SIZE_8192,
/** EEPROM Size for EEPROM emulation is 4096 bytes */
NVM_EEPROM_EMULATOR_SIZE_4096,
/** EEPROM Size for EEPROM emulation is 2048 bytes */
NVM_EEPROM_EMULATOR_SIZE_2048,
/** EEPROM Size for EEPROM emulation is 1024 bytes */
NVM_EEPROM_EMULATOR_SIZE_1024,
/** EEPROM Size for EEPROM emulation is 512 bytes */
NVM_EEPROM_EMULATOR_SIZE_512,
/** EEPROM Size for EEPROM emulation is 256 bytes */
NVM_EEPROM_EMULATOR_SIZE_256,
/** EEPROM Size for EEPROM emulation is 0 bytes */
NVM_EEPROM_EMULATOR_SIZE_0,
};
/**
* \brief BOD33 Action.
*
* What action should be triggered when BOD33 is detected.
*
*/
enum nvm_bod33_action {
/** No action */
NVM_BOD33_ACTION_NONE,
/** The BOD33 generates a reset */
NVM_BOD33_ACTION_RESET,
/** The BOD33 generates an interrupt */
NVM_BOD33_ACTION_INTERRUPT,
};
#ifdef FEATURE_BOD12
/**
* \brief BOD12 Action.
*
* What action should be triggered when BOD12 is detected.
*
*/
enum nvm_bod12_action {
/** No action */
NVM_BOD12_ACTION_NONE,
/** The BOD12 generates a reset */
NVM_BOD12_ACTION_RESET,
/** The BOD12 generates an interrupt */
NVM_BOD12_ACTION_INTERRUPT,
};
#endif
/**
* \brief WDT Window time-out period.
*
* Window mode time-out period in clock cycles.
*
*/
enum nvm_wdt_window_timeout {
/** 8 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_8,
/** 16 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_16,
/** 32 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_32,
/** 64 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_64,
/** 128 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_128,
/** 256 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_256,
/** 512 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_512,
/** 1024 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_1024,
/** 2048 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_2048,
/** 4096 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_4096,
/** 8192 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_8192,
/** 16384 clock cycles */
NVM_WDT_WINDOW_TIMEOUT_PERIOD_16384,
};
/**
* \brief WDT Early warning offset.
*
* This setting determine how many GCLK_WDT cycles before a watchdog time-out period
* an early warning interrupt should be triggered.
*
*/
enum nvm_wdt_early_warning_offset {
/** 8 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_8,
/** 16 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_16,
/** 32 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_32,
/** 64 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_64,
/** 128 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_128,
/** 256 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_256,
/** 512 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_512,
/** 1024 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_1024,
/** 2048 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_2048,
/** 4096 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_4096,
/** 8192 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_8192,
/** 16384 clock cycles */
NVM_WDT_EARLY_WARNING_OFFSET_16384,
};
/**
* \brief NVM user row fuse setting structure.
*
* This structure contain the layout of the first 64 bits of the user row
* which contain the fuse settings.
*/
struct nvm_fusebits {
/** Bootloader size */
enum nvm_bootloader_size bootloader_size;
/** EEPROM emulation area size */
enum nvm_eeprom_emulator_size eeprom_size;
#if (SAMC20) || (SAMC21)
/** BODVDD Threshold level at power on */
uint8_t bodvdd_level;
/** BODVDD Enable at power on */
bool bodvdd_enable;
/** BODVDD Action at power on */
enum nvm_bod33_action bodvdd_action;
/* BODVDD Hysteresis at power on */
bool bodvdd_hysteresis;
#else
/** BOD33 Threshold level at power on */
uint8_t bod33_level;
/** BOD33 Enable at power on */
bool bod33_enable;
/** BOD33 Action at power on */
enum nvm_bod33_action bod33_action;
/* BOD33 Hysteresis at power on */
bool bod33_hysteresis;
#endif
/** WDT Enable at power on */
bool wdt_enable;
/** WDT Always-on at power on */
bool wdt_always_on;
/** WDT Period at power on */
uint8_t wdt_timeout_period;
/** WDT Window mode time-out at power on */
enum nvm_wdt_window_timeout wdt_window_timeout;
/** WDT Early warning interrupt time offset at power on */
enum nvm_wdt_early_warning_offset wdt_early_warning_offset;
/** WDT Window mode enabled at power on */
bool wdt_window_mode_enable_at_poweron;
/** NVM Lock bits */
uint16_t lockbits;
#ifdef FEATURE_BOD12
/** BOD12 Threshold level at power on */
uint8_t bod12_level;
/** BOD12 Enable at power on */
bool bod12_enable;
/** BOD12 Action at power on */
enum nvm_bod12_action bod12_action;
/* BOD12 Hysteresis at power on */
bool bod12_hysteresis;
#endif
};
/**
* \name Configuration and Initialization
* @{
*/
/**
* \brief Initializes an NVM controller configuration structure to defaults.
*
* Initializes a given NVM controller configuration structure to a set of
* known default values. This function should be called on all new
* instances of these configuration structures before being modified by the
* user application.
*
* The default configuration is as follows:
* \li Power reduction mode enabled after sleep mode until first NVM access
* \li Automatic page write mode disabled
* \li Number of FLASH wait states left unchanged
*
* \param[out] config Configuration structure to initialize to default values
*
*/
static inline void nvm_get_config_defaults(
struct nvm_config *const config)
{
/* Sanity check the parameters */
Assert(config);
/* Write the default configuration for the NVM configuration */
config->sleep_power_mode = NVM_SLEEP_POWER_MODE_WAKEONACCESS;
config->manual_page_write = true;
config->wait_states = NVMCTRL->CTRLB.bit.RWS;
config->disable_cache = false;
#if (SAMC20) || (SAMC21)
config->disable_rww_cache = false;
#endif
config->cache_readmode = NVM_CACHE_READMODE_NO_MISS_PENALTY;
}
enum status_code nvm_set_config(
const struct nvm_config *const config);
/**
* \brief Checks if the NVM controller is ready to accept a new command.
*
* Checks the NVM controller to determine if it is currently busy execution an
* operation, or ready for a new command.
*
* \return Busy state of the NVM controller.
*
* \retval true If the hardware module is ready for a new command
* \retval false If the hardware module is busy executing a command
*
*/
static inline bool nvm_is_ready(void)
{
/* Get a pointer to the module hardware instance */
Nvmctrl *const nvm_module = NVMCTRL;
return nvm_module->INTFLAG.reg & NVMCTRL_INTFLAG_READY;
}
/** @} */
/**
* \name NVM Access Management
* @{
*/
void nvm_get_parameters(
struct nvm_parameters *const parameters);
enum status_code nvm_write_buffer(
const uint32_t destination_address,
const uint8_t *buffer,
uint16_t length);
enum status_code nvm_read_buffer(
const uint32_t source_address,
uint8_t *const buffer,
uint16_t length);
enum status_code nvm_update_buffer(
const uint32_t destination_address,
uint8_t *const buffer,
uint16_t offset,
uint16_t length);
enum status_code nvm_erase_row(
const uint32_t row_address);
enum status_code nvm_execute_command(
const enum nvm_command command,
const uint32_t address,
const uint32_t parameter);
enum status_code nvm_get_fuses(struct nvm_fusebits *fusebits);
enum status_code nvm_set_fuses(struct nvm_fusebits *fb);
bool nvm_is_page_locked(uint16_t page_number);
/**
* \brief Retrieves the error code of the last issued NVM operation.
*
* Retrieves the error code from the last executed NVM operation. Once
* retrieved, any error state flags in the controller are cleared.
*
* \note The \ref nvm_is_ready() function is an exception. Thus, errors
* retrieved after running this function should be valid for the function
* executed before \ref nvm_is_ready().
*
* \return Error caused by the last NVM operation.
*
* \retval NVM_ERROR_NONE No error occurred in the last NVM operation
*
* \retval NVM_ERROR_LOCK The last NVM operation attempted to access a locked
* region
* \retval NVM_ERROR_PROG An invalid NVM command was issued
*/
static inline enum nvm_error nvm_get_error(void)
{
enum nvm_error ret_val;
/* Get a pointer to the module hardware instance */
Nvmctrl *const nvm_module = NVMCTRL;
/* Mask out non-error bits */
ret_val = (enum nvm_error)(nvm_module->STATUS.reg & NVM_ERRORS_MASK);
/* Clear error flags */
nvm_module->STATUS.reg = NVM_ERRORS_MASK;
/* Return error code from the NVM controller */
return ret_val;
}
/** @} */
#ifdef __cplusplus
}
#endif
/** @} */
/**
* \page asfdoc_sam0_nvm_extra Extra Information for NVM Driver
*
* \section asfdoc_sam0_nvm_extra_acronyms Acronyms
* The table below presents the acronyms used in this module:
*
* <table>
* <tr>
* <th>Acronym</th>
* <th>Description</th>
* </tr>
* <tr>
* <td>NVM</td>
* <td>Non-Volatile Memory</td>
* </tr>
* <tr>
* <td>EEPROM</td>
* <td>Electrically Erasable Programmable Read-Only Memory</td>
* </tr>
* </table>
*
*
* \section asfdoc_sam0_nvm_extra_dependencies Dependencies
* This driver has the following dependencies:
*
* - None
*
*
* \section asfdoc_sam0_nvm_extra_errata Errata
* There are no errata related to this driver.
*
*
* \section asfdoc_sam0_nvm_extra_history Module History
* An overview of the module history is presented in the table below, with
* details on the enhancements and fixes made to the module since its first
* release. The current version of this corresponds to the newest version in
* the table.
*
* <table>
* <tr>
* <th>Changelog</th>
* </tr>
* <tr>
* <td>Removed BOD12 reference, removed nvm_set_fuses() API</td>
* </tr>
* <tr>
* <td>Added functions to read/write fuse settings</td>
* </tr>
* <tr>
* <td>Added support for NVM cache configuration</td>
* </tr>
* <tr>
* <td>Updated initialization function to also enable the digital interface
* clock to the module if it is disabled</td>
* </tr>
* <tr>
* <td>Initial Release</td>
* </tr>
* </table>
*/
/**
* \page asfdoc_sam0_nvm_exqsg Examples for NVM Driver
*
* This is a list of the available Quick Start guides (QSGs) and example
* applications for \ref asfdoc_sam0_nvm_group. QSGs are simple examples with
* step-by-step instructions to configure and use this driver in a selection of
* use cases. Note that a QSG can be compiled as a standalone application or be
* added to the user application.
*
* - \subpage asfdoc_sam0_nvm_basic_use_case
*
* \page asfdoc_sam0_nvm_document_revision_history Document Revision History
*
* <table>
* <tr>
* <th>Doc. Rev.</th>
* <th>Date</th>
* <th>Comments</th>
* </tr>
* <tr>
* <td>42114E</td>
* <td>12/2015</td>
* <td>Added support for SAM L21/L22, SAM C21, SAM D09, SAMR30 and SAM DA1</td>
* </tr>
* <tr>
* <td>42114D</td>
* <td>12/2014</td>
* <td>Added support for SAM R21 and SAM D10/D11</td>
* </tr>
* <tr>
* <td>42114C</td>
* <td>01/2014</td>
* <td>Added support for SAM D21</td>
* </tr>
* <tr>
* <td>42114B</td>
* <td>06/2013</td>
* <td>Corrected documentation typos</td>
* </tr>
* <tr>
* <td>42114A</td>
* <td>06/2013</td>
* <td>Initial document release</td>
* </tr>
* </table>
*/
#endif /* NVM_H_INCLUDED */