mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
Koen Zandberg
3d470b0c12
This adds support for the gd32v class devices from Gigadevice. The gd32vf103 contains an 108 MHz RISC-V core with similar peripherals as the stm32f1 devices Co-authored-by: Benjamin Valentin <benjamin.valentin@ml-pa.com>
313 lines
18 KiB
C
313 lines
18 KiB
C
/*!
|
|
\file gd32vf103_fmc.h
|
|
\brief definitions for the FMC
|
|
|
|
\version 2019-06-05, V1.0.0, firmware for GD32VF103
|
|
\version 2019-09-18, V1.0.1, firmware for GD32VF103
|
|
*/
|
|
|
|
/*
|
|
Copyright (c) 2019, GigaDevice Semiconductor Inc.
|
|
|
|
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.
|
|
|
|
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.
|
|
*/
|
|
|
|
#ifndef GD32VF103_FMC_H
|
|
#define GD32VF103_FMC_H
|
|
|
|
#include "gd32vf103.h"
|
|
|
|
/* FMC and option byte definition */
|
|
#define FMC FMC_BASE /*!< FMC register base address */
|
|
#define OB OB_BASE /*!< option bytes base address */
|
|
|
|
/* registers definitions */
|
|
#define FMC_WS REG32((FMC) + 0x00U) /*!< FMC wait state register */
|
|
#define FMC_KEY REG32((FMC) + 0x04U) /*!< FMC unlock key register */
|
|
#define FMC_OBKEY REG32((FMC) + 0x08U) /*!< FMC option bytes unlock key register */
|
|
#define FMC_STAT REG32((FMC) + 0x0CU) /*!< FMC status register */
|
|
#define FMC_CTL REG32((FMC) + 0x10U) /*!< FMC control register */
|
|
#define FMC_ADDR REG32((FMC) + 0x14U) /*!< FMC address register */
|
|
#define FMC_OBSTAT REG32((FMC) + 0x1CU) /*!< FMC option bytes status register */
|
|
#define FMC_WP REG32((FMC) + 0x20U) /*!< FMC erase/program protection register */
|
|
#define FMC_PID REG32((FMC) + 0x100U) /*!< FMC product ID register */
|
|
|
|
#define OB_SPC REG16((OB) + 0x00U) /*!< option byte security protection value */
|
|
#define OB_USER REG16((OB) + 0x02U) /*!< option byte user value*/
|
|
#define OB_WP0 REG16((OB) + 0x08U) /*!< option byte write protection 0 */
|
|
#define OB_WP1 REG16((OB) + 0x0AU) /*!< option byte write protection 1 */
|
|
#define OB_WP2 REG16((OB) + 0x0CU) /*!< option byte write protection 2 */
|
|
#define OB_WP3 REG16((OB) + 0x0EU) /*!< option byte write protection 3 */
|
|
|
|
/* bits definitions */
|
|
/* FMC_WS */
|
|
#define FMC_WS_WSCNT BITS(0,2) /*!< wait state counter */
|
|
|
|
/* FMC_KEY */
|
|
#define FMC_KEY_KEY BITS(0,31) /*!< FMC_CTL unlock key bits */
|
|
|
|
/* FMC_OBKEY */
|
|
#define FMC_OBKEY_OBKEY BITS(0,31) /*!< option bytes unlock key bits */
|
|
|
|
/* FMC_STAT */
|
|
#define FMC_STAT_BUSY BIT(0) /*!< flash busy flag bit */
|
|
#define FMC_STAT_PGERR BIT(2) /*!< flash program error flag bit */
|
|
#define FMC_STAT_WPERR BIT(4) /*!< erase/program protection error flag bit */
|
|
#define FMC_STAT_ENDF BIT(5) /*!< end of operation flag bit */
|
|
|
|
/* FMC_CTL */
|
|
#define FMC_CTL_PG BIT(0) /*!< main flash program command bit */
|
|
#define FMC_CTL_PER BIT(1) /*!< main flash page erase command bit */
|
|
#define FMC_CTL_MER BIT(2) /*!< main flash mass erase command bit */
|
|
#define FMC_CTL_OBPG BIT(4) /*!< option bytes program command bit */
|
|
#define FMC_CTL_OBER BIT(5) /*!< option bytes erase command bit */
|
|
#define FMC_CTL_START BIT(6) /*!< send erase command to FMC bit */
|
|
#define FMC_CTL_LK BIT(7) /*!< FMC_CTL lock bit */
|
|
#define FMC_CTL_OBWEN BIT(9) /*!< option bytes erase/program enable bit */
|
|
#define FMC_CTL_ERRIE BIT(10) /*!< error interrupt enable bit */
|
|
#define FMC_CTL_ENDIE BIT(12) /*!< end of operation interrupt enable bit */
|
|
|
|
/* FMC_ADDR */
|
|
#define FMC_ADDR0_ADDR BITS(0,31) /*!< Flash erase/program command address bits */
|
|
|
|
/* FMC_OBSTAT */
|
|
#define FMC_OBSTAT_OBERR BIT(0) /*!< option bytes read error bit. */
|
|
#define FMC_OBSTAT_SPC BIT(1) /*!< option bytes security protection code */
|
|
#define FMC_OBSTAT_USER BITS(2,9) /*!< store USER of option bytes block after system reset */
|
|
#define FMC_OBSTAT_DATA BITS(10,25) /*!< store DATA of option bytes block after system reset. */
|
|
|
|
/* FMC_WP */
|
|
#define FMC_WP_WP BITS(0,31) /*!< store WP of option bytes block after system reset */
|
|
|
|
/* FMC_WSEN */
|
|
#define FMC_WSEN_WSEN BIT(0) /*!< FMC wait state enable bit */
|
|
|
|
/* FMC_PID */
|
|
#define FMC_PID_PID BITS(0,31) /*!< product ID bits */
|
|
|
|
/* constants definitions */
|
|
/* define the FMC bit position and its register index offset */
|
|
#define FMC_REGIDX_BIT(regidx, bitpos) (((uint32_t)(regidx) << 6) | (uint32_t)(bitpos))
|
|
#define FMC_REG_VAL(offset) (REG32(FMC + ((uint32_t)(offset) >> 6)))
|
|
#define FMC_BIT_POS(val) ((uint32_t)(val) & 0x1FU)
|
|
#define FMC_REGIDX_BITS(regidx, bitpos0, bitpos1) (((uint32_t)(regidx) << 12) | ((uint32_t)(bitpos0) << 6) | (uint32_t)(bitpos1))
|
|
#define FMC_REG_VALS(offset) (REG32(FMC + ((uint32_t)(offset) >> 12)))
|
|
#define FMC_BIT_POS0(val) (((uint32_t)(val) >> 6) & 0x1FU)
|
|
#define FMC_BIT_POS1(val) ((uint32_t)(val) & 0x1FU)
|
|
#define FMC_REG_OFFSET_GET(flag) ((uint32_t)(flag) >> 12)
|
|
|
|
/* configuration register */
|
|
#define FMC_STAT_REG_OFFSET 0x0CU /*!< status register offset */
|
|
#define FMC_CTL_REG_OFFSET 0x10U /*!< control register offset */
|
|
#define FMC_OBSTAT_REG_OFFSET 0x1CU /*!< option byte status register offset */
|
|
|
|
/* fmc state */
|
|
typedef enum
|
|
{
|
|
FMC_READY, /*!< the operation has been completed */
|
|
FMC_BUSY, /*!< the operation is in progress */
|
|
FMC_PGERR, /*!< program error */
|
|
FMC_WPERR, /*!< erase/program protection error */
|
|
FMC_TOERR, /*!< timeout error */
|
|
}fmc_state_enum;
|
|
|
|
/* FMC interrupt enable */
|
|
typedef enum
|
|
{
|
|
FMC_INT_END = FMC_REGIDX_BIT(FMC_CTL_REG_OFFSET, 12U), /*!< enable FMC end of program interrupt */
|
|
FMC_INT_ERR = FMC_REGIDX_BIT(FMC_CTL_REG_OFFSET, 10U), /*!< enable FMC error interrupt */
|
|
}fmc_int_enum;
|
|
|
|
/* FMC flags */
|
|
typedef enum
|
|
{
|
|
FMC_FLAG_BUSY = FMC_REGIDX_BIT(FMC_STAT_REG_OFFSET, 0U), /*!< FMC busy flag */
|
|
FMC_FLAG_PGERR = FMC_REGIDX_BIT(FMC_STAT_REG_OFFSET, 2U), /*!< FMC operation error flag bit */
|
|
FMC_FLAG_WPERR = FMC_REGIDX_BIT(FMC_STAT_REG_OFFSET, 4U), /*!< FMC erase/program protection error flag bit */
|
|
FMC_FLAG_END = FMC_REGIDX_BIT(FMC_STAT_REG_OFFSET, 5U), /*!< FMC end of operation flag bit */
|
|
FMC_FLAG_OBERR = FMC_REGIDX_BIT(FMC_OBSTAT_REG_OFFSET, 0U), /*!< FMC option bytes read error flag */
|
|
}fmc_flag_enum;
|
|
|
|
/* FMC interrupt flags */
|
|
typedef enum
|
|
{
|
|
FMC_INT_FLAG_PGERR = FMC_REGIDX_BITS(FMC_STAT_REG_OFFSET, 2U, 10U), /*!< FMC operation error interrupt flag bit */
|
|
FMC_INT_FLAG_WPERR = FMC_REGIDX_BITS(FMC_STAT_REG_OFFSET, 4U, 10U), /*!< FMC erase/program protection error interrupt flag bit */
|
|
FMC_INT_FLAG_END = FMC_REGIDX_BITS(FMC_STAT_REG_OFFSET, 5U, 12U), /*!< FMC end of operation interrupt flag bit */
|
|
}fmc_interrupt_flag_enum;
|
|
|
|
/* unlock key */
|
|
#define UNLOCK_KEY0 ((uint32_t)0x45670123U) /*!< unlock key 0 */
|
|
#define UNLOCK_KEY1 ((uint32_t)0xCDEF89ABU) /*!< unlock key 1 */
|
|
|
|
/* FMC wait state counter */
|
|
#define WS_WSCNT(regval) (BITS(0,2) & ((uint32_t)(regval)))
|
|
#define WS_WSCNT_0 WS_WSCNT(0) /*!< FMC 0 wait */
|
|
#define WS_WSCNT_1 WS_WSCNT(1) /*!< FMC 1 wait */
|
|
#define WS_WSCNT_2 WS_WSCNT(2) /*!< FMC 2 wait */
|
|
|
|
/* option bytes software/hardware free watch dog timer */
|
|
#define OB_FWDGT_SW ((uint8_t)0x01U) /*!< software free watchdog */
|
|
#define OB_FWDGT_HW ((uint8_t)0x00U) /*!< hardware free watchdog */
|
|
|
|
/* option bytes reset or not entering deep sleep mode */
|
|
#define OB_DEEPSLEEP_NRST ((uint8_t)0x02U) /*!< no reset when entering deepsleep mode */
|
|
#define OB_DEEPSLEEP_RST ((uint8_t)0x00U) /*!< generate a reset instead of entering deepsleep mode */
|
|
|
|
/* option bytes reset or not entering standby mode */
|
|
#define OB_STDBY_NRST ((uint8_t)0x04U) /*!< no reset when entering deepsleep mode */
|
|
#define OB_STDBY_RST ((uint8_t)0x00U) /*!< generate a reset instead of entering standby mode */
|
|
|
|
/* option bytes boot bank value */
|
|
#define OB_BOOT_B0 ((uint8_t)0x08U) /*!< boot from bank0 */
|
|
|
|
#define OB_USER_MASK ((uint8_t)0xF0U) /*!< MASK value */
|
|
|
|
/* read protect configure */
|
|
#define FMC_NSPC ((uint8_t)0xA5U) /*!< no security protection */
|
|
#define FMC_USPC ((uint8_t)0xBBU) /*!< under security protection */
|
|
|
|
/* OB_SPC */
|
|
#define OB_SPC_SPC ((uint32_t)0x000000FFU) /*!< option byte security protection value */
|
|
#define OB_SPC_SPC_N ((uint32_t)0x0000FF00U) /*!< option byte security protection complement value */
|
|
|
|
/* OB_USER */
|
|
#define OB_USER_USER ((uint32_t)0x00FF0000U) /*!< user option value */
|
|
#define OB_USER_USER_N ((uint32_t)0xFF000000U) /*!< user option complement value */
|
|
|
|
/* OB_WP0 */
|
|
#define OB_WP0_WP0 ((uint32_t)0x000000FFU) /*!< FMC write protection option value */
|
|
|
|
/* OB_WP1 */
|
|
#define OB_WP1_WP1 ((uint32_t)0x0000FF00U) /*!< FMC write protection option complement value */
|
|
|
|
/* OB_WP2 */
|
|
#define OB_WP2_WP2 ((uint32_t)0x00FF0000U) /*!< FMC write protection option value */
|
|
|
|
/* OB_WP3 */
|
|
#define OB_WP3_WP3 ((uint32_t)0xFF000000U) /*!< FMC write protection option complement value */
|
|
|
|
/* option bytes write protection */
|
|
#define OB_WP_0 ((uint32_t)0x00000001U) /*!< erase/program protection of sector 0 */
|
|
#define OB_WP_1 ((uint32_t)0x00000002U) /*!< erase/program protection of sector 1 */
|
|
#define OB_WP_2 ((uint32_t)0x00000004U) /*!< erase/program protection of sector 2 */
|
|
#define OB_WP_3 ((uint32_t)0x00000008U) /*!< erase/program protection of sector 3 */
|
|
#define OB_WP_4 ((uint32_t)0x00000010U) /*!< erase/program protection of sector 4 */
|
|
#define OB_WP_5 ((uint32_t)0x00000020U) /*!< erase/program protection of sector 5 */
|
|
#define OB_WP_6 ((uint32_t)0x00000040U) /*!< erase/program protection of sector 6 */
|
|
#define OB_WP_7 ((uint32_t)0x00000080U) /*!< erase/program protection of sector 7 */
|
|
#define OB_WP_8 ((uint32_t)0x00000100U) /*!< erase/program protection of sector 8 */
|
|
#define OB_WP_9 ((uint32_t)0x00000200U) /*!< erase/program protection of sector 9 */
|
|
#define OB_WP_10 ((uint32_t)0x00000400U) /*!< erase/program protection of sector 10 */
|
|
#define OB_WP_11 ((uint32_t)0x00000800U) /*!< erase/program protection of sector 11 */
|
|
#define OB_WP_12 ((uint32_t)0x00001000U) /*!< erase/program protection of sector 12 */
|
|
#define OB_WP_13 ((uint32_t)0x00002000U) /*!< erase/program protection of sector 13 */
|
|
#define OB_WP_14 ((uint32_t)0x00004000U) /*!< erase/program protection of sector 14 */
|
|
#define OB_WP_15 ((uint32_t)0x00008000U) /*!< erase/program protection of sector 15 */
|
|
#define OB_WP_16 ((uint32_t)0x00010000U) /*!< erase/program protection of sector 16 */
|
|
#define OB_WP_17 ((uint32_t)0x00020000U) /*!< erase/program protection of sector 17 */
|
|
#define OB_WP_18 ((uint32_t)0x00040000U) /*!< erase/program protection of sector 18 */
|
|
#define OB_WP_19 ((uint32_t)0x00080000U) /*!< erase/program protection of sector 19 */
|
|
#define OB_WP_20 ((uint32_t)0x00100000U) /*!< erase/program protection of sector 20 */
|
|
#define OB_WP_21 ((uint32_t)0x00200000U) /*!< erase/program protection of sector 21 */
|
|
#define OB_WP_22 ((uint32_t)0x00400000U) /*!< erase/program protection of sector 22 */
|
|
#define OB_WP_23 ((uint32_t)0x00800000U) /*!< erase/program protection of sector 23 */
|
|
#define OB_WP_24 ((uint32_t)0x01000000U) /*!< erase/program protection of sector 24 */
|
|
#define OB_WP_25 ((uint32_t)0x02000000U) /*!< erase/program protection of sector 25 */
|
|
#define OB_WP_26 ((uint32_t)0x04000000U) /*!< erase/program protection of sector 26 */
|
|
#define OB_WP_27 ((uint32_t)0x08000000U) /*!< erase/program protection of sector 27 */
|
|
#define OB_WP_28 ((uint32_t)0x10000000U) /*!< erase/program protection of sector 28 */
|
|
#define OB_WP_29 ((uint32_t)0x20000000U) /*!< erase/program protection of sector 29 */
|
|
#define OB_WP_30 ((uint32_t)0x40000000U) /*!< erase/program protection of sector 30 */
|
|
#define OB_WP_31 ((uint32_t)0x80000000U) /*!< erase/program protection of sector 31 */
|
|
#define OB_WP_ALL ((uint32_t)0xFFFFFFFFU) /*!< erase/program protection of all sectors */
|
|
|
|
/* FMC timeout */
|
|
#define FMC_TIMEOUT_COUNT ((uint32_t)0x000F0000U) /*!< FMC timeout count value */
|
|
|
|
/* FMC BANK address */
|
|
#define FMC_SIZE (*(uint16_t *)0x1FFFF7E0U) /*!< FMC size */
|
|
#define SRAM_SIZE (*(uint16_t *)0x1FFFF7E2U) /*!< SRAM size*/
|
|
|
|
/* function declarations */
|
|
/* FMC main memory programming functions */
|
|
/* set the FMC wait state counter */
|
|
void fmc_wscnt_set(uint32_t wscnt);
|
|
/* unlock the main FMC operation */
|
|
void fmc_unlock(void);
|
|
/* lock the main FMC operation */
|
|
void fmc_lock(void);
|
|
/* FMC erase page */
|
|
fmc_state_enum fmc_page_erase(uint32_t page_address);
|
|
/* FMC erase whole chip */
|
|
fmc_state_enum fmc_mass_erase(void);
|
|
/* FMC program a word at the corresponding address */
|
|
fmc_state_enum fmc_word_program(uint32_t address, uint32_t data);
|
|
/* FMC program a half word at the corresponding address */
|
|
fmc_state_enum fmc_halfword_program(uint32_t address, uint16_t data);
|
|
|
|
/* FMC option bytes programming functions */
|
|
/* unlock the option byte operation */
|
|
void ob_unlock(void);
|
|
/* lock the option byte operation */
|
|
void ob_lock(void);
|
|
/* erase the FMC option byte */
|
|
fmc_state_enum ob_erase(void);
|
|
/* enable write protection */
|
|
fmc_state_enum ob_write_protection_enable(uint32_t ob_wp);
|
|
/* configure security protection */
|
|
fmc_state_enum ob_security_protection_config(uint8_t ob_spc);
|
|
/* program the FMC user option byte */
|
|
fmc_state_enum ob_user_write(uint8_t ob_fwdgt, uint8_t ob_deepsleep, uint8_t ob_stdby, uint8_t ob_boot);
|
|
/* program the FMC data option byte */
|
|
fmc_state_enum ob_data_program(uint32_t address, uint8_t data);
|
|
/* get OB_USER in register FMC_OBSTAT */
|
|
uint8_t ob_user_get(void);
|
|
/* get OB_DATA in register FMC_OBSTAT */
|
|
uint16_t ob_data_get(void);
|
|
/* get the FMC option byte write protection */
|
|
uint32_t ob_write_protection_get(void);
|
|
/* get FMC option byte security protection state */
|
|
FlagStatus ob_spc_get(void);
|
|
|
|
/* FMC interrupts and flags management functions */
|
|
/* enable FMC interrupt */
|
|
void fmc_interrupt_enable(uint32_t interrupt);
|
|
/* disable FMC interrupt */
|
|
void fmc_interrupt_disable(uint32_t interrupt);
|
|
/* check flag is set or not */
|
|
FlagStatus fmc_flag_get(uint32_t flag);
|
|
/* clear the FMC flag */
|
|
void fmc_flag_clear(uint32_t flag);
|
|
/* get FMC interrupt flag state */
|
|
FlagStatus fmc_interrupt_flag_get(fmc_interrupt_flag_enum flag);
|
|
/* clear FMC interrupt flag state */
|
|
void fmc_interrupt_flag_clear(fmc_interrupt_flag_enum flag);
|
|
/* return the FMC state */
|
|
fmc_state_enum fmc_state_get(void);
|
|
/* check FMC ready or not */
|
|
fmc_state_enum fmc_ready_wait(uint32_t timeout);
|
|
|
|
#endif /* GD32VF103_FMC_H */
|