## **GigaDevice Semiconductor Inc.**

# Introduction to GD32G5x3 Flash read-whilewrite (RWW) feature

# Application Notes AN249

Revision 1.0

(Jan. 2025)



# **Table of Contents**

| Tab  | le | of Contents                                    | 2 |
|------|----|------------------------------------------------|---|
| List | of | f Figure                                       | 3 |
| List | of | f Table                                        | 4 |
| 1.   | I  | Foreword                                       | 5 |
| 2.   | (  | GD32G5x3 flash configuration                   | 6 |
| 2.1  | 1. | Single and double bank switching configuration | 6 |
| 2.2  | 2. | Dual-bank mapping switch                       | 6 |
| 2.3  | 3. | Bank startup configuration                     | 7 |
| 3.   | I  | AP upgrade                                     | 8 |
| 3.1  | 1. | IAP introduction                               | 8 |
| 3.2  | 2. | IAP upgrade steps                              | 8 |
| 3.3  | 3. | IAP upgrade code 1                             | 0 |
| 4.   | ł  | Revision history1                              | 4 |
|      |    |                                                |   |



# **List of Figure**

| Figure 2-1. Single and double bank switching register bit | 6 |
|-----------------------------------------------------------|---|
| Figure 2-2. Bank switch register bit                      | 7 |
| Figure 2-3. Bank start selection register bit             | 7 |
| Figure 3-1. Instruction for generating bin file           | 8 |
| Figure 3-2. generate a txt file                           | 9 |
| Figure 3-3. Store data                                    | 9 |
| Figure 3-4. Simple IAP upgrade plan for dual-bank1        | 0 |



# **List of Table**

| Table 3-1. IAP Upgrade Code | 10 |
|-----------------------------|----|
| Table 4-1. Revision history | 14 |



## 1. Foreword

The dual-bank flash of GD32G5x3 features RWW capabilities, allowing the CPU to execute code in flash while erasing and programming another area of flash. This means that while one partition is being erased or written, the other can still be read. This capability is crucial for implementing imperceptible software updates, such as OTA updates. Thanks to the RWW feature, IAP and applications can be combined for operation without the need for separate handling. This dual-bank mode greatly simplifies the implementation of IAP, enhances the speed of IAP updates, and allows developers more flexibility in program upgrades and management.



## 2. GD32G5x3 flash configuration

The main storage flash memory of GD32G5x3 can reach up to 512KB, composed of 2x256 pages in a dual-bank structure, with each page being 1KB, and also includes 2x13KB of information blocks for boot loader. Each page of the main storage flash memory can be erased individually; in a single-bank structure, it is composed of 256 pages, with each page being 2KB. In the following context, the flash memory size is taken as 512KB.

## 2.1. Single and double bank switching configuration

The method to switch from single-bank to dual-bank is to set the DBS bit in the Option Byte Control Register (FMC\_OBCTL) to 1, at which point the read width is 64 bits. As shown in *Figure 2-1. Single and double bank switching register bit*.

| •   | 2.4.8.                                                | . →      | Option·byte·control·register·(FMC_OBCTL) |            |         |         |              |         |                   |        |          |      |          |         |         |         |   |
|-----|-------------------------------------------------------|----------|------------------------------------------|------------|---------|---------|--------------|---------|-------------------|--------|----------|------|----------|---------|---------|---------|---|
|     | Address offset: 0x20 //<br>Reset value: 0xXXXX XXX // |          |                                          |            |         |         |              |         |                   |        |          |      |          |         |         |         |   |
| 4   | *                                                     |          | This∙re                                  | egister    | has to  | be acc  | essed        | by-wore | d (32-bi          | it). ₊ |          |      |          |         |         |         |   |
|     | 31 🕫                                                  | 30 +2    | 29.0                                     | 28.0       | 27 @    | 26 @    | 25₽          | 24+2    | 23 0              | 22 🕫   | 21.∉     | 20+2 | 19₽      | 18.0    | 17₽     | 16₽     | ÷ |
|     |                                                       |          |                                          |            |         |         | TCMSRA       | SRAM_EC |                   |        | _        |      | _        | FWDGSP  | FWDGSP  | nFWDG_H | + |
|     | Rese                                                  | erved @  | NRSI_M                                   | DSEL[1:0]# | nBOOI04 | nSWB104 | M_ERS+       | CEN 🕫   | nBOOI1+           | DBS₽   | Reserved | BB∻  | Reserved | D_STDBY | D_DPSLP | W⇔      |   |
|     |                                                       | e        | r                                        | ₩÷         | rw.⇔    | rw ↔    | rw⇔          | rw.⇔    | rw ¢              | rw ₽   | ę        | rw⇔  | ę        | rw 0    | rw ₽    | rw ₽    | ÷ |
|     | 15₽                                                   | 14 🕫     | 13₽                                      | 12.0       | 11₽     | 10₽     | 9€           | 8.0     | 7.0               | 6.4    | 5₽       | 40   | 3₽       | 2.0     | 1₽      | 0.0     | ÷ |
|     | FMC_SW                                                |          | nRST_ST                                  | nRST_DP    | _       |         |              |         |                   |        |          |      |          |         |         |         |   |
| P.∉ |                                                       | Reserved | DBY SLP -                                |            |         | erved e | BOR_TH[1:0]₽ |         | SPC[7:0]+         |        |          |      |          |         |         |         |   |
|     | rw e                                                  |          | rw.≎                                     | rw ∂       |         | e       | r            | ₩÷      | T₩ + <sup>2</sup> |        |          |      |          |         |         |         |   |

Figure 2-1. Single and double bank switching register bit

### 2.2. Dual-bank mapping switch

The bank0 and bank1 of main memory can be mapped to address 0x08000000 and 0x08040000 respectively, as shown in *Figure 2-2. Bank switch register bit*.

#### Figure 2-2. Bank switch register bit



**Note:** After modifying the option bytes through option byte programming operations, the changes will only take effect after a system reset or by setting the OBRLD bit in the FMC\_CTL register to 1. If you need the changes to take effect immediately without saving to the option byte register, you can switch the memory mapping in real-time by operating the FMC\_SWP bit in the SYSCFG\_CFG0 register.

#### 2.3. Bank startup configuration

The GD32G5x3 series of MCUs supports dual-bank mode for the main memory, and when configured in dual-bank mode, users can configure registers to determine which bank the application starts from.

When the BB bit is 0, if currently booting from the main flash area, boot from bank0; when the BB bit is 1, if currently booting from the main flash area, boot from bank1, and if bank1 is empty, boot from bank0. As shown in *Figure 2-3. Bank start selection register bit*.



Figure 2-3. Bank start selection register bit



## 3. IAP upgrade

#### 3.1. IAP introduction

IAP (In-Application Programming) upgrade is a technology that allows firmware to be upgraded during the application's operation. Its advantage is that it allows the firmware to be updated through the application code without stopping the application. For systems that need to run continuously, IAP can reduce downtime caused by firmware updates. Of course, it also has certain defects. One is that implementing IAP may require a complex software architecture and additional storage space. The other is that if the firmware verification is not sufficient, it may introduce security risks, so if the new firmware has defects, there must be a reliable recovery mechanism to switch back to the old firmware.

The RWW feature of GD32G5x3 allows for updating the code in bank1 while the code in bank0 is running, followed by a bank switch to implement firmware updates. When the FMC\_SWP bit is set, the code in bank0 can be updated while the code in bank1 is running, followed by a bank switch to implement firmware updates.

#### 3.2. IAP upgrade steps

Based on the above characteristics, assuming the current BB position is 0, the simplified IAP upgrade process for GD32G5x3 dual-bank is as follows.

Add the instruction to generate bin file during compilation in the Keil for the upgrade project to be written.

| 🗑 Options for Target 'GD32G5x3'                                                            |                                            |      |                  |        |         |  |  |
|--------------------------------------------------------------------------------------------|--------------------------------------------|------|------------------|--------|---------|--|--|
| Device   Target   Output   Listing   User   C/C++ (AC6)   Asm   Linker   Debug   Utilities |                                            |      |                  |        |         |  |  |
|                                                                                            |                                            |      |                  |        | _       |  |  |
| Command Items                                                                              | User Command                               |      | Stop on Exi      | S      |         |  |  |
| Before Compile C/C++ File                                                                  |                                            |      |                  |        |         |  |  |
| Run #1                                                                                     |                                            | 2    | Not Specified    | $\Box$ |         |  |  |
| Run #2                                                                                     |                                            | 2    | Not Specified    | $\Box$ |         |  |  |
| Before Build/Rebuild                                                                       |                                            |      |                  |        |         |  |  |
| Run #1                                                                                     |                                            | 2    | Not Specified    | $\Box$ |         |  |  |
| Run #2                                                                                     |                                            | 2    | Not Specified    | $\Box$ |         |  |  |
| After Build/Rebuild                                                                        |                                            |      |                  |        |         |  |  |
| Run #1                                                                                     | C:\Keil_v5\ARM\ARMCC\bin\fromelf.exebin -o | Proj | ect.bin ./output | /Pro   | ject.ax |  |  |
| Run #2                                                                                     |                                            | 2    | Not Specified    | Γ      |         |  |  |
|                                                                                            |                                            |      |                  |        |         |  |  |

Figure 3-1. Instruction for generating bin file

After compilation, use the bin2hex software to convert the generated bin file into a txt file.



Figure 3-2. generate a txt file

| 📧 bin2hex.exe          |  |
|------------------------|--|
| EventRecorderStub.scvd |  |
| JLinkLog.txt           |  |
| 📓 JLinkSettings.ini    |  |
| 🎱 Project.bin          |  |
| Project.txt            |  |

Open the txt file and write the data from the txt into a const array.

#### Figure 3-3. Store data



Note: These steps are simplified, and in engineering practice, one can write code to transmit the code to be upgraded via methods such as UART.





#### Figure 3-4. Simple IAP upgrade plan for dual-bank

#### 3.3. IAP upgrade code

The IAP upgrade simple code is shown as in <u>Table 3-1. IAP Upgrade Code</u>. This code is only used to demonstrate the simple IAP upgrade using the RWW feature, which is quite basic and can be used as an idea for code upgrading.

Table 3-1. IAP Upgrade Code

| #include <stdint.h></stdint.h>   |  |  |  |  |
|----------------------------------|--|--|--|--|
| #include <stdbool.h></stdbool.h> |  |  |  |  |
| #include "gd32g5x3.h"            |  |  |  |  |
| /* Added by the user */          |  |  |  |  |
| const uint8_t app2[] = {}        |  |  |  |  |

GigaDevice

```
#define destinaion_add0 0x08000000
#define destinaion add1 0x08040000
uint32_t destinaion_add = 0;
uint32_t NUM = sizeof(app2);
void switch_to_new_firmware(uint64_t add)
{
    for(uint16_t i = 0; i < NUM; i += 8) {
        if(*(uint64_t *)(add + i) != (uint64_t)((uint64_t)app2[i] | ((uint64_t)app2[i + 1] << 8) |
((uint64_t)app2[i + 2] << 16) | ((
                 uint64_t)app2[i + 3] << 24) | ((uint64_t)app2[i + 4] << 32) | ((uint64_t)app2[i + 5] <<
40) | ((uint64_t)app2[i + 6] << 48) | ((
                         uint64_t)app2[i + 7] << 56))) {
            return;
        }
    }
    /* The implementation needs to be modified according to the actual situation */
    __disable_irq();
    /* disable pre-fetch */
    FMC_WS &= ~FMC_WS_PFEN;
    /* disable IBUS cache */
    FMC_WS &= ~FMC_WS_ICEN;
    /* disable DBUS cache */
    FMC_WS &= ~FMC_WS_DCEN;
    /* reset IBUS cache */
    FMC_WS |= FMC_WS_ICRST;
    /* reset DBUS cache */
    FMC_WS |= FMC_WS_DCRST;
    /* bank mapping switch */
    ob_bank_memory_swap_config(OB_BANK_MAPPING_SWAP);
    /* enable pre-fetch */
    FMC_WS |= FMC_WS_PFEN;
    /* enable IBUS cache */
    FMC_WS |= FMC_WS_ICEN;
    /* enable DBUS cache */
    FMC_WS |= FMC_WS_DCEN;
    /* Recovery interrupt */
    __enable_irq();
    /* Pay attention to ensure the safe state during the restart or switchover proces */
}
```

GigaDevice

```
void confirm_to_switch(void)
{
    if(RESET == gpio_input_bit_get(KEY_D_GPIO_PORT, KEY_D_PIN)) {
        delay_1ms(1000);
        /* check whether the key is pressed */
        if(RESET == gpio_input_bit_get(KEY_D_GPIO_PORT, KEY_D_PIN)) {
            fmc_unlock();
            ob_unlock();
            switch_to_new_firmware(destinaion_add);
            ob_reload();
        }
    }
}
/*!
    \brief
               main function
    \param[in] none
    \param[out] none
    \retval
              none
*/
int main(void)
{
    /* configure systick */
    systick_config();
    if(FMC_OBCTL & FMC_OBCTL_FMC_SWP) {
        destinaion_add = destinaion_add1;
    } else {
        destinaion_add = destinaion_add0;
    }
    /* enable the LED2 GPIO clock */
    rcu_periph_clock_enable(RCU_GPIOE);
    /* configure LED2 GPIO pin */
    gpio_mode_set(GPIOE, GPIO_MODE_OUTPUT, GPIO_PUPD_NONE, GPIO_PIN_4);
    gpio_output_options_set(GPIOE, GPIO_OTYPE_PP, GPIO_OSPEED_60MHZ, GPIO_PIN_4);
    /* reset LED2 GPIO pin */
    gpio_bit_reset(GPIOE, GPIO_PIN_4);
    /* enable the key GPIO clock */
    rcu_periph_clock_enable(RCU_GPIOA);
    rcu_periph_clock_enable(RCU_GPIOC);
    rcu_periph_clock_enable(RCU_GPIOF);
```

# GigaDevice

#### AN249 Introduction to GD32G5x3 Flash read-while-write(RWW) feature

```
/* configure key pin as input */
    gpio_mode_set(KEY_B_GPIO_PORT,
                                             GPIO MODE INPUT,
                                                                        GPIO PUPD NONE,
KEY_B_PIN);
    gpio_mode_set(KEY_C_GPIO_PORT,
                                             GPIO_MODE_INPUT,
                                                                       GPIO_PUPD_NONE,
KEY_C_PIN);
    gpio_mode_set(KEY_D_GPIO_PORT,
                                             GPIO_MODE_INPUT,
                                                                        GPIO_PUPD_NONE,
KEY_D_PIN);
   while(1) {
        /* user code */
        /* check whether the key is pressed */
        if(RESET == gpio_input_bit_get(KEY_B_GPIO_PORT, KEY_B_PIN)) {
            delay_1ms(100);
            /* check whether the key is pressed */
            if(RESET == gpio_input_bit_get(KEY_B_GPIO_PORT, KEY_B_PIN)) {
                /* toggle LED2 GPIO pin */
                gpio_bit_toggle(GPIOE, GPIO_PIN_4);
            }
        }
        /* upgrade code */
        if(RESET == gpio_input_bit_get(KEY_C_GPIO_PORT, KEY_C_PIN)) {
            delay_1ms(1000);
            /* check whether the key is pressed */
            if(RESET == gpio_input_bit_get(KEY_C_GPIO_PORT, KEY_C_PIN)) {
                fmc_unlock();
                fmc_bank1_erase();
                /* Write app2 code to another bank, where the upgrade code can be imported by
other means such as UART */
                for(uint16_t i = 0; i < NUM; i += 8) {
                     fmc_doubleword_program(destinaion_add + i, (uint64_t)((uint64_t)app2[i] |
((uint64_t)app2[i + 1] << 8) | ((uint64_t)app2[i + 2] << 16) | ((
                                                uint64_t)app2[i + 3] << 24) | ((uint64_t)app2[i +
4] << 32) | ((uint64_t)app2[i + 5] << 40) | ((uint64_t)app2[i + 6] << 48) | ((
                                                             uint64_t)app2[i + 7] << 56)));
                }
            }
        }
        confirm_to_switch();
   }
```



## 4. Revision history

#### Table 4-1. Revision history

| Revision No. | Description     | Date         |  |  |  |
|--------------|-----------------|--------------|--|--|--|
| 1.0          | Initial release | Jan.14, 2025 |  |  |  |

#### **Important Notice**

This document is the property of GigaDevice Semiconductor Inc. and its subsidiaries (the "Company"). This document, including any product of the Company described in this document (the "Product"), is owned by the Company under the intellectual property laws and treaties of the People's Republic of China and other jurisdictions worldwide. The Company reserves all rights under such laws and treaties and does not grant any license under its patents, copyrights, trademarks, or other intellectual property rights. The names and brands of third party referred thereto (if any) are the property of their respective owner and referred to for identification purposes only.

The Company makes no warranty of any kind, express or implied, with regard to this document or any Product, including, but not limited to, the implied warranties of merchantability and fitness for a particular purpose. The Company does not assume any liability arising out of the application or use of any Product described in this document. Any information provided in this document is provided only for reference purposes. It is the responsibility of the user of this document to properly design, program, and test the functionality and safety of any application made of this information and any resulting product. Except for customized products which have been expressly identified in the applicable agreement, the Products are designed, developed, and/or manufactured for ordinary business, industrial, personal, and/or household applications only. The Products are not designed, intended, or authorized for use as components in systems designed or intended for the operation of weapons, weapons systems, nuclear installations, atomic energy control instruments, combustion control instruments, airplane or spaceship instruments, transportation instruments, traffic signal instruments, life-support devices or systems, other medical devices or systems (including resuscitation equipment and surgical implants), pollution control or hazardous substances management, or other uses where the failure of the device or Product could cause personal injury, death, property or environmental damage ("Unintended Uses"). Customers shall take any and all actions to ensure using and selling the Products in accordance with the applicable laws and regulations. The Company is not liable, in whole or in part, and customers shall and hereby do release the Company as well as its suppliers and/or distributors from any claim, damage, or other liability arising from or related to all Unintended Uses of the Products. Customers shall indemnify and hold the Company as well as its suppliers and/or distributors harmless from and against all claims, costs, damages, and other liabilities, including claims for personal injury or death, arising from or related to any Unintended Uses of the Products.

Information in this document is provided solely in connection with the Products. The Company reserves the right to make changes, corrections, modifications or improvements to this document and Products and services described herein at any time, without notice.

© 2025 GigaDevice - All rights reserved