

## Introduction

This migration guide is designed to help you analyze the steps required to migrate from an existing SXX32F103 device to an MH2103A device. This document gathers the most important information and lists the important things to be aware of.

To port an application from the SXX32F103 series to the MH2103A series, users need to analyze hardware migration, peripheral migration, and firmware migration.

Supported model list:

| Supported model | MH2103Axxxx |
|-----------------|-------------|
| Supported model | MHZ1U3Axxxx |



# Catal ogue

| 1.   | Quickly replace the SXX32F103 chip                                                                                      | 4   |
|------|-------------------------------------------------------------------------------------------------------------------------|-----|
| 2.   | Precautions for hardware migration                                                                                      | 5   |
| 2    | .1 TIM2 module reuse difference                                                                                         | 5   |
| 2    | .2 GPIO pin voltage withstand description                                                                               | 5   |
| 2.3  | Bootstrap mode select pin                                                                                               | 5   |
| 3.   | Precautions for software migration                                                                                      | 6   |
| 3. 1 | System module                                                                                                           |     |
|      | 3.1.1 Significant bits of interrupt priority                                                                            |     |
|      | 3. 1. 2 Turning the JTAG off or releasing the JTRST in emulation will cause a reset                                     | 6   |
|      | 3. 1. 3 Bootstrap mode configuration options are not reloaded during soft reset                                         | 7   |
|      | 3.1.4 Non-32bit alignment access to the APB bus failed                                                                  | 7   |
|      | 3. 1. 5 Method of distinguishing MH2103A chip and SXX32F103 chip                                                        | 8   |
|      | 3.1.6 Software delay variance                                                                                           | 9   |
| 3    | .1.7 TRACESWO as a Printf function                                                                                      | 9   |
| 3    | 3.2 ADC module                                                                                                          |     |
|      | conversion cannot be stopped by only disabling ADC                                                                      | 10  |
|      | 3. 2. 2 Software trigger Precautions for enabling ADON to perform external events twice in a row                        | 11  |
|      | 3.2.3 The ADC enables software triggers for external events twice in a row                                              | 12  |
|      | 3.2.4 Different modes of dual ADCs differ                                                                               | 12  |
|      | 3.2.5 Precautions for ADC and DMA configuration                                                                         | .13 |
| 3    | 3.2.6 ADC automatic injection mode requires scanning to be enabled 3 TIM module                                         |     |
|      | 3. 3. 1 Channel3 of the general timer TIM2-TIM5 does not support the comparison output function                         | 14  |
|      | $3.\ 3.\ 2$ The BIF cannot clear when the time interval between two consecutive brakes is less than one TIM clock cycle | 15  |
|      | 3. 3. 3 TIM did not reconfigure CNT for counting after the CNT value was modified by the software                       | .15 |
|      | 3. 3. 4 Precautions for using the TIM DMA Burst function                                                                | 16  |
| 3    | . 4 CAN module                                                                                                          | .17 |
|      | 3.4.1 Precautions for configuring a filter in hibernation mode                                                          | 17  |
|      | 3.4.2 The sending timestamp function is different                                                                       | 18  |
| 3    | 4.4.3 When TXFP is 1, the FIFO priority is different                                                                    | 19  |
| 3    | 5 FLASH module                                                                                                          | 21  |



| 3.5.1 There is a probability Fault when calling the FLASH_<br>EraseOptionBytes() interface to erase option bytes21                         |
|--------------------------------------------------------------------------------------------------------------------------------------------|
| 3. 6 USART module                                                                                                                          |
| 3.6.1 Smart card mode has no clock output24                                                                                                |
| 3.6.2 Using DMA to send data, check whether all data is sent through the USART TC flag. Precautions25                                      |
| 3.7 SPI module27 3.7.1 As the Master, this section describes the precautions for sending                                                   |
| and receiving data in full-duplex mode27                                                                                                   |
| 3.7.2 If the SPE is closed in BUSY state, data may not be sent or received.29 3.7.3 Using DMA to transfer data configuration differences29 |
| 3.8 USB module30                                                                                                                           |
| 3. 8. 1 USB use precautions                                                                                                                |
| 3.9 DMA module31                                                                                                                           |
| 3.9.1 DMA usage precautions31                                                                                                              |
| 3.10 IIC module                                                                                                                            |
| 3.10.1 Send start bit & Stop bit as Master Note                                                                                            |
| 4. ISP, emulator, offline burner use precautions32                                                                                         |
| 4.1 ISP tool32                                                                                                                             |
| 4.2 Emulator34                                                                                                                             |
| 4.2.1 Precautions for using Keil.SXX32F1xx_DFP.2.3.0 or later pack34                                                                       |
| 4.3 Downloader35                                                                                                                           |
| 4.3.1 The chip is burned abnormally in the read/write protection state3                                                                    |
| 4.3.2 Failed to burn using the WizPro200ST8 programmer36                                                                                   |
| Historical version3                                                                                                                        |



The MH2103A series microcontrollers are basically compatible with the SXX32F103 series, while enhancing many features, with some differences from the SXX32F103, detailed in this document.

### 1. Quickly replace the SXX32F103 chip

- •Step 1: Compare peripheral specifications, Flash capacity, SRAM capacity, etc., unsolder SXX32F103 and replace it with MH2103A corresponding model;
  - ●Step 2: Use ISP or KEIL, download HEX or BIN file;
  - •Step 3: Download data other than HEX or BIN files or perform system corrections if necessary;
  - •Step 4: Check whether the program can run normally and test whether the function of the equipment is normal;
  - •Step 5: For quick troubleshooting of other problems, see the migration Precautions in this document.
- •Step 6: If the program still does not work properly after the above steps, please contact the agent and MH support staff for assistance.



### 2. Precautions for hardware migration

### 2.1 TIM2 module reuse difference

The Channel 1-Channel 4 output reusable pin of TIM2 of MH2103A chip is different from that of SXX32F103. The detailed differences are as follows:

| Reuse        | TIM2_REMAP (No rem | [1:0] = 00<br>appi ng) | TIM2_REMAP             | [1:0] = 01<br>emapping) | TIM2_REMAP<br>(Partial r | [1:0] = 10<br>emapping) | TIM2_REMAP[1:0] = 11  (Full remapping) |            |  |
|--------------|--------------------|------------------------|------------------------|-------------------------|--------------------------|-------------------------|----------------------------------------|------------|--|
| function     | SXX32F103          | MH2103A                | SXX32F103              | MH2103A                 | SXX32F103                | MH2103A                 | SXX32F103                              | MH2103A    |  |
| TIM2_CH1_ETR | PAO                |                        | PA15 <b>nonsupport</b> |                         | PAO                      | nonsupport              | PA15                                   |            |  |
| TIM2_CH2     | PA                 | 1                      | PB3                    |                         | PA1                      | nonsupport              | PB3                                    | nonsupport |  |
| TIM2_CH3     | PA2                | nonsupport             | PA2                    | nonsupport              | PB10                     | nonsupport              | PB10 nonsuppor                         |            |  |
| TIM2_CH4     |                    |                        | PA3                    |                         |                          | РВ                      | 11                                     |            |  |

### 2.2 GPIO pin voltage withstand description

MH2103A PA11, PA12 pin withstand voltage upper limit is 3.6V, hardware circuit design pay attention to the voltage range.

### Note:

The optional multiplexing functions of PA11 pin are CAN\_RX, USART1\_CTS, USBDM, TIM1\_CH4;

The optional multiplexing functions of PA12 pin are CAN\_TX, USART1\_RTS, USBDP , TIM1\_ETR;

### 2.3 Bootstrap mode select pin

When the B00T0 pin is suspended, the chip has a probability of starting the execution program from the non-flash area after the reset startup. If you need to boot from user Flash, avoid dangling B00T0.

It is recommended that the BOOTO pin be connected to an external 10K pull-down resistor or direct ground, which can stably boot from the user Flash.

| Bootstrap mode<br>select pin |       | Bootstrap mode | aliasing                                        |  |  |  |
|------------------------------|-------|----------------|-------------------------------------------------|--|--|--|
| BOOT1                        | ВООТО | Bootstrap mode | arrasing                                        |  |  |  |
| X                            | 0     | User Flash     | Select user Flash as the bootstrap space        |  |  |  |
| 0                            | 1     | System memory  | Select system memory as the bootstrap space     |  |  |  |
| 1                            | 1     | Embedded SRAM  | Select the embedded SRAM as the bootstrap space |  |  |  |



### 3. Precautions for software migration

### 3.1 System module

### 3.1.1 Significant bits of interrupt priority

- Supports a maximum of 71 masked interrupt channels.
- > 8 programmable priority levels (interrupt priority is valid with 3 bits).

### 3.1.2 Turning the JTAG off or releasing the JTRST in emulation will cause a reset

In the simulation state, turning off the JTAG or releasing the JTRST (as shown below) will cause the chip to reset immediately, and subsequent code cannot be executed.

```
int main(void)
                       RCC_ClocksTypeDef clocks;
                       Delay_Init();
UART_Configuration(115200);
RCC_GetClocksFreq(&clocks);
       26
27 =
28
29 -
                      FRINTF_LOG("MIN," %3.1fMhz, HCLK: %3.1fMhz, PCLK1: %3.1fMhz, PCLK2: %3.1fMhz, ADCCLK: %3.1fMhz\n", \
(fioat)clocks.YSCLK Frequency/1000000, (fioat)clocks.HCLK Frequency/1000000, \
(fioat)clocks.PCLK2 Frequency/1000000, (float)clocks.PCLK2 Frequency / 1000000, (float)clocks.ADCCLK Frequency / 1000000, \
(FRINTF_LOG("MH2103 GPIO IO Input Tset.\n");
       30
31
32
33
GPIO_Configuration();
                       RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable,ENABLE);
                                                                                                                                              Disable Jtag
37
38 =
39
40 }
                        while(1)
               int main (void)
                       RCC ClocksTypeDef clocks;
                      Delay_Init();
UART_Configuration(115200);
RCC_GetClocksFreq(&clocks);
       26
27 🛱
                       PRINTF_LOG("\n");
PRINTF_LOG("SYSCLK: %3.1fMhz, HCLK: %3.1fMhz, PCLK1: %3.1fMhz, PCLK2: %3.1fMhz, ADCCLK: %3.1fMhz\n", \
                       (float)clocks.SYSCLK_Frequency/1000000, (float)clocks.HCLK_Frequency/1000000, \
(float)clocks.PCLK1_Frequency/1000000, (float)clocks.PCLK2_Frequency/1000000, (float)clocks.ADCCLK_Frequency/1000000);
PRINTF_LOG("MH2103 GPIO IO Input Tset.\n");
       31
32
33
34
35
36
                       GPIO Configuration();
                      RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
GPIO_PinRemapConfig(GPIO_Remap_SWJ_NoJTRST,ENABLE);
                                                                                                                                                 ► Release Jtrst
37
38 🖃
                        while (1)
        39
```

#### Solution:

In the simulation state, do not execute the JTAG off or JTRST release code (as shown in the red box above); In the non-simulation state, the JTAG off and JTRST release functions are normal and do not affect the program operation.



### 3.1.3 Bootstrap mode configuration options are not reloaded during soft reset

### Example of exceptions (Enabling read/write protection through an ISP) :

- $\blacktriangleright$  B00T0 connects to high, B00T1 connects to low, external Reset, successfully connects to ISP.
- > Set BOOTO to low and enable read/write protection through ISP.
- After read/write protection is enabled, programs in the Flash are not executed.

### Root cause:

After read/write protection is enabled through the ISP, the chip automatically initiates a system reset. During soft reset, the boot-up configuration pin is not reloaded and the BOOT pin configuration status is maintained. Therefore, the program is executed in the system memory again, resulting in no program execution in the Flash.

#### Solution:

After the bootstrap mode is changed, you can RESET the chip through external reset or power it on or off again.

### 3.1.4 Non-32bit alignment access to the APB bus failed

### Problem description:

When the ADC is configured in left-justified mode, the converted high 8-bit data is obtained from the address 0x4001244E, and the obtained value is fixed to 0.

The program is configured as follows:



```
void ADC_Configuration(void)
   uint32_t i;
   ADC_InitTypeDef ADC_InitStructure;
   GPIO_InitTypeDef GPIO_InitStructure;
   RCC APB2PeriphClockCmd(RCC APB2Periph GPIOA | RCC APB2Periph ADC1, ENABLE );
   GPIO InitStructure.GPIO Pin = ADC TEST CHANNEL PIN;
   GPIO InitStructure.GPIO Speed = GPIO Speed 50MHz;
   GPIO InitStructure.GPIO Mode = GPIO Mode AIN;
   GPIO Init (GPIOA, &GPIO InitStructure);
   RCC_ADCCLKConfig(RCC_PCLK2_Div8);
   ADC_DeInit(ADC1);
   ADC InitStructure.ADC Mode = ADC Mode Independent;
   ADC InitStructure.ADC ScanConvMode = ENABLE;
   ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
   ADC InitStructure.ADC ExternalTriqConv = ADC ExternalTriqConv None;
ADC InitStructure.ADC DataAlign = ADC DataAlign Left;
   ADC InitStructure.ADC NbrOfChannel = CONV_CHANNEL NUM;
                                                                    Justification left
   ADC_Init(ADC1, &ADC_InitStructure);
   for (i = 0; i < CONV_CHANNEL_NUM; i++)
        ADC_RegularChannelConfig(ADC1 , ADC_CovChannel[i] , i+1 , ADC_SampleTIME[i]);
   ADC SoftwareStartConvCmd(ADC1, ENABLE);
   ADC_Cmd (ADC1, ENABLE);
   ADC ResetCalibration (ADC1);
   while (ADC GetResetCalibrationStatus(ADC1));
   ADC_StartCalibration(ADC1);
   while (ADC GetCalibrationStatus (ADC1));
    while (1)
                                                        Non-32bit aligned access
         ADC SoftwareStartConvCmd(ADC1, ENABLE);
         if (ADC GetFlagStatus (ADC1, ADC FLAG EOC)
                                                             == SET)
          {
              adc value = *(uint32 t*)(&(ADC1->DR)+1);
              PRINTF LOG("adc value = 0x%x\n",adc value);
         }
                                                            The value is always 0
    }
```

#### Root cause:

The APB bus cannot be accessed properly unless it is 32-bit aligned.

#### Solution:

When accessing the registers of the APB bus, the alignment needs to be 32bit. If you need a field in the 32bit register, 32bit Alignment read before processing.

### 3.1.5 Method of distinguishing MH2103A chip and SXX32F103 chip

Read the base address 0x1FFFF7E8 and get a 32bit identifier to distinguish. As shown in the picture below:





The identification is described as follows:

| Model number | Type identification |
|--------------|---------------------|
| MH2103A CBT6 | 0x1A5A5BBX          |
| MH2103A CCT6 | 0x1A5A5CCX          |
| MH2103A RPT6 | 0x1A6A5CDX          |
| MH2103A VET6 | 0x1A8A6DDX          |
| MH2103A VGT6 | 0x1A8A6EDX          |
| MH2103A ZET6 | 0x1A9A6DDX          |

### 3.1.6 Software delay variance

Some software delays implemented using other platforms need to be adjusted, such as the following simple software delay function:

```
void delay(void)
{
    uint8_t i = 100;
    while(i--);
}
```

The delay time for SXX32F103 to execute this function is 10us.

The delay time for MH2103A to execute this function is 7.2us.

If the application has strict requirements on the software delay time, adjust the software delay parameters to some extent.

### 3.1.7 TRACESWO as a Printf function

### Problem description:

When TRACESWO pin (PB3) is used as the Log output, no Log is found.

### Root cause:



MH2103APB3 takes precedence over TRACESWO as a JTDO function.

#### Solution:

If TRACESWO pins (PB3) are used as Log output, JTAG multiplexing needs to be configured.

As shown below:

```
RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable,ENABLE);
```

### 3.2 ADC module

3.2.1 After ADC continuous sampling is configured and ADC is enabled, conversion cannot be stopped by only disabling ADC

Repeat the problem as follows:

- Set ADC to continuous sampling mode and enable DMA transport. After the software is triggered, each time the ADC conversion is completed, the DMA carries one time, and the DMA completes an interrupt.
- Configure to turn off ADC enablement in DMA completion interrupt.
- MH2103A will continue to convert ADC after disabling ADC, and will continue to advance DMA completion interrupts

```
void ADC_Configuration(void)
   uint32 t i;
   ADC_InitTypeDef ADC_InitStructure;
   GPIO_InitTypeDef GPIO_InitStructure;
   RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_ADC1, ENABLE );
   GPIO_InitStructure.GPIO_Pin = ADC_TEST_CHANNEL_PIN;
   GPIO InitStructure.GPIO Speed = GPIO Speed 50MHz;
GPIO InitStructure.GPIO Mode = GPIO Mode AIN;
   GPIO_Init(GPIOA, &GPIO_InitStructure);
   RCC ADCCLKConfig(RCC PCLK2 Div8);
   ADC DeInit (ADC1);
                                                                           The function is set to
   ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
   ADC InitStructure.ADC ScanConvMode = ENABLE;
                                                                           trigger continuously
   ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
   ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv None;
   ADC InitStructure.ADC DataAlign = ADC DataAlign Right;
ADC InitStructure.ADC_NbrOfChannel = CONV_CHANNEL_NUM;
   ADC Init(ADC1, &ADC InitStructure);
   for(i = 0; i < CONV_CHANNEL_NUM; i++)
       ADC_RegularChannelConfig(ADC1 , ADC_CovChannel[i] , i+1 , ADC_SampleTIME[i]);
                                             Configure DMA transport. Once the conversion is
                                            complete, DMA transport is performed once
   DMA Configuration();
   ADC Cmd(ADC1, ENABLE);
                                          Enabling ADC
   ADC_ResetCalibration(ADC1);
   while (ADC_GetResetCalibrationStatus(ADC1));
   ADC StartCalibration(ADC1):
   while (ADC GetCalibrationStatus(ADC1));
                                                       Software trigger
   ADC SoftwareStartConvCmd(ADC1, ENABLE);
```



```
void DMA Configuration(void)
        NVIC InitTypeDef NVIC InitStructure;
        DMA InitTypeDef DMA InitStructure;
       RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMAl,ENABLE);
DMA_DeInit(DMAl_Channell);
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_MA_DMA_DMA_DMA_Disable;
        RCC AHBPeriphClockCmd(RCC AHBPeriph DMA1, ENABLE);
        DMA InitStructure.DMA M2M = DMA M2M Disable;
DMA Init(DMAl Channell,&DMA InitStructure);
        DMA_Cmd(DMA1_Channell,ENABLE);
ADC_DMACmd(ADC1,ENABLE);
      NVIC_InitStructure.NVIC_IRQChannel = DMAl_Channell_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(aNVIC_InitStructure);
DMA_ITConfig(DMAl_Channell,DMAl_IT_TC1,ENABLE);
                                                                                                                                                    Enable DMA completion interrupt
void DMAl_Channell_IRQHandler(void)
        if(DMA_GetITStatus(DMA1_IT_TC1) != RESET)
                  DMA_ClearITPendingBit(DMAl_IT_TCl);
                 DMA_ClearFlag(DMA1_FLAG_TC1);
                 PRINTF_LOG("Code Value = %d , 电压值 = %2.4f\n" , DAM_ADC_Value[0] , (float)VREF * DAM_ADC_Value[0]/4095/1000);
                ADC Cmd(ADC1, DISABLE);
                                                                                                    Disable the ADC function
```

#### Solution:

Configure ADC for single conversion and wait until the last sampling period is ADC\_CLK before disabling ADC.

As shown in the picture below:

The ADC clock configured in the above example is 9M and the sampling period is 239.5.

Therefore, the waiting time is  $(239.5+12.5)*1/9000 \approx 27.922us$ , so the waiting time of 30us in the figure below is sufficient.

```
void DMAl_Channell_IRQHandler(void)
{
    if (DMA_GetITStatus (DMAl_IT_TC1) != RESET) {
        DMA_ClearITPendingBit (DMAl_IT_TC1);
        DMA_ClearIIQ (DMAl_IT_AG_TC1);
        PRINTF_LOG("Code Value = %d , 电压值 = %2.4f\n" , DAM_ADC_Value[0] , (float)VREF * DAM_ADC_Value[0]/4095/1000);

        ADC1->CR2 4= ~BIT(1);
        Delay Us(30);
        Wait 30us

        ADC_Cmd(ADC1, DISABLE);
        Disable the ADC function
}
```

# 3.2.2 Software trigger Precautions for enabling ADON to perform external events twice in a row

### Problem description:

```
133
134
135
136
ADC_Cmd(ADC1, ENABLE);
ADC_Cmd(ADC1, ENABLE);
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
137
138
```

After the above operation, MH2103A will fail to clear 0 after EOC setting, and the ADC conversion value will be abnormal.



#### Solution:

After enabling ADC, enable ADC again, wait for EOC to be set, and read the quantized value.

```
As shown in the picture below:

ADC_Cmd(ADC1, ENABLE);

ADC_Cmd(ADC1, ENABLE);

while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC) == RESET);

ADC_GetConversionValue(ADC1);

ADC_SoftwareStartConvCmd(ADC1, ENABLE);
```

#### Attention:

When the ADC is enabled and the ADC is enabled again, both SXX32F103 and MH2103A trigger an ADC conversion.

### 3.2.3 The ADC enables software triggers for external events twice in a row

### Problem description:

```
The software triggered two
consecutive external events

ADC_Cmd (ADC1, ENABLE);

ADC_SoftwareStartConvCmd (ADC1, ENABLE);

ADC_SoftwareStartConvCmd (ADC1, ENABLE);

ADC_SoftwareStartConvCmd (ADC1, ENABLE);
```

After the above operation, MH2103A will fail to clear 0 after EOC setting, and the ADC conversion value will be abnormal.

#### Solution:

The software triggers the external event, waits for the EOC to set, and reads the quantized value.

As shown in the picture below:

```
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC) == RESET);
ADC_GetConversionValue(ADC1);
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC) == RESET);
ADC_GetConversionValue(ADC1);
```

#### 3.2.4 Different modes of dual ADCs differ

### Synchronous injection, alternate trigger mode:

SXX32F103: After the injection group channel or rule group channel is triggered , the conversion is normal.

MH2103A: Normal transformation after the injection group channel is triggered; After the rule group channel is triggered, no conversion is performed.

#### Synchronous regular, fast crossover, slow crossover mode:

SXX32F103: After the injection group channel or rule group channel is triggered, the conversion is normal.



MH2103A: Normal transformation after the rule group channel is triggered; After the injection group channel is triggered, no conversion is performed.

# $3.\,2.\,5$ Precautions for ADC and DMA configuration Problem description:

When an ADC is configured to continuously sample multiple channels and DMA is used to move data, errors (mismatches) occur in ADC quantization values. For example, convert five channels, configure them as rule groups, and scan continuously for conversion.

Expectations for  $\{0V,1V,1.5V,2V,2.5V, 0V,1V,1.5V,2V,2.5V, 0V,1V,1.5V,2V,2.5V ...\}$ ; To obtain the value of the actual software  $\{2\ V,\ 1.5\ V,\ 2,\ 5\ V,\ 0\ V,\ 1\ V,\ 1.5\ V,\ 2\ V,\ 2,\ 5\ V,\ 0\ V,\ 1\ V...\ \}$  The following figure shows the configuration

```
void ADC_Configuration(void)
   uint32 t i;
   ADC_InitTypeDef ADC_InitStructure;
   GPIO_InitTypeDef GPIO_InitStructure;
   RCC APB2PeriphClockCmd(RCC APB2Periph GPIOA | RCC APB2Periph ADC1, ENABLE );
   GPIO InitStructure.GPIO Pin = ADC TEST CHANNEL PIN;
   GPIO InitStructure.GPIO Speed = GPIO Speed 50MHz;
   GPIO InitStructure.GPIO Mode = GPIO Mode AIN;
   GPIO Init (GPIOA, &GPIO InitStructure);
   RCC ADCCLKConfig(RCC PCLK2 Div8);
   ADC DeInit (ADC1);
   ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
   ADC_InitStructure.ADC_ScanConvMode = ENABLE;
   ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
       InitStructure.ADC ExternalTrigConv = ADC ExternalTrigConv None;
   ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
   ADC InitStructure.ADC NbrOfChannel = CONV CHANNEL NUM;
   ADC_Init(ADC1, &ADC_InitStructure);
   for (i = 0; i < CONV CHANNEL NUM: i++)
       ADC RegularChannelConfig(ADCl , ADC CovChannel[i] , i+1 , ADC SampleTIME[i]);
   ADC ResetCalibration(ADC1):
   while (ADC GetResetCalibrationStatus(ADCl));
   ADC StartCalibration(ADC1);
                                                       Enabling ADC and configuring
   while (ADC_GetCalibrationStatus(ADC1));
                                                      DMA directly have other
   ADC_Cmd(ADC1, ENABLE);
   instructions
    DMA_Configuration();
```

#### Solution:

If DMA is configured and then ADC is enabled, ADC quantization value error (misalignment) does not occur.

As shown in the following picture



```
void ADC Configuration(void)
   uint32 t i:
   ADC InitTypeDef ADC InitStructure;
   GPIO InitTypeDef GPIO InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_ADC1, ENABLE );
    GPIO_InitStructure.GPIO_Pin = ADC_TEST_CHANNEL_PIN;
    GPIO InitStructure.GPIO Speed = GPIO Speed 50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO Init(GPIOA, &GPIO InitStructure);
    RCC ADCCLKConfig(RCC_PCLK2_Div8);
   ADC_DeInit(ADC1);
   ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
    ADC_InitStructure.ADC_ScanConvMode = ENABLE;
    ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
       InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
    ADC_InitStructure.ADC_NbrOfChannel = CONV_CHANNEL_NUM;
   ADC Init(ADC1, &ADC InitStructure);
    for(i = 0; i < CONV_CHANNEL_NUM; i++)</pre>
       ADC_RegularChannelConfig(ADC1 , ADC_CovChannel[i] , i+1 , ADC_SampleTIME[i]);
   ADC_ResetCalibration(ADC1);
    while (ADC_GetResetCalibrationStatus(ADC1));
    ADC StartCalibration(ADC1);
    while(ADC_GetCalibrationStatus(ADC1));
   DMA Configuration();
   ADC Cmd(ADC1, ENABLE);
```

#### 3.2.6 ADC automatic injection mode requires scanning to be enabled

When using ADC auto-injection mode, scanning needs to be enabled, as follows:

```
ADC_DeInit(ADC1);
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrofChannel = 2;
ADC_Init(ADC1, &ADC_InitStructure);
```

### 3.3 TIM module

### 3.3.1 Channel 3 of the general timer TIM2-TIM5 does not support the comparison output function

Channel 3 of the MH2103A universal timer TIM2-TIM5 does not support the output comparison function, but only supports the input capture function.

Solution: Select another channel of TIM2-TIM5, or select another channel of TIM for output use.



# 3.3.2 The BIF cannot clear when the time interval between two consecutive brakes is less than one TIM clock cycle

### Problem description:

When the time interval between two consecutive brakes is less than one TIM clock cycle, there will be a BIF(brake interrupt mark) set 1, unable to clear 0 phenomenon.

### Solution:

Method 1: Software configuration brake signal is normal input IO interrupt, do not use BIF and brake interrupt.

Method 2: Set 1 in the BIF to soft-reset TIM.

3.3.3 TIM did not reconfigure CNT for counting after the CNT value was modified by

#### the software

### Problem description:

In some cases, after the software changes the CNT value, TIM does not count the CNT reconfigured.

#### Case 1:

- > TIM is configured in monopulse mode with update interrupt enabled
- Modify the CNT value in interrupt and enable TIM

```
TIM SelectOnePulseMode (TIMx, TIM OPMode Single);
TIM_ITConfig(TIMx, TIM_IT Update, ENABLE);
TIM_Cmd(TIMx, ENABLE); Set to monopulse
mode

if (TIM_GetITStatus(TIMx, TIM_IT_Update) != RESET)
{
    TIM_ClearITPendingBit(TIMx, TIM_IT_Update);
    TIM_SetCounter(TIMx, NewCnt);
    TIM_Cmd(TIMx, ENABLE);
}

Enable TIM
```

#### Attention:

In monopulse mode, the TIM counters of the SXX32F103 and MH2103A automatically stop when the next update event UEV is generated.

### Case 2:

- ➤ TIM is not configured with monopulse mode.
- In interrupt, the software disables TIM, changes the CNT value, and enables TIM again.



```
void TIM3_IRQHandler(void)
{
    if (TIM_GetITStatus(TIMx, TIM_IT_Update) != RESET)
    {
        TIM_Cmd(TIMx, DISABLE);
        TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
        TIM_SetCounter(TIMx, 1999);
        TIM_Cmd(TIMx, ENABLE);
}
```

#### Root cause:

The software modification CNT takes effect only when CEN is enabled.

#### Solution:

If you need to manually modify the CNT value after an update event occurs when the CNT is configured in monopulse mode, enable TIM first.

```
if (TIM_GetITStatus(TIMx, TIM_IT_Update) != RESET)
{
    TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
    TIM_Cmd(TIMx,ENABLE);
    TIM_SetAutoreload(TIMx,NewCnt);
}
```

➤ If monopulse mode is not configured, do not turn off TIM when manually modifying the CNT value.

### 3.3.4 TIMDMABurst function use precautions

### Problem description:

When the Advanced & General TIM configures DBL to be non-0, a DMA request is generated, and DMA carries data only once.

|  | 15       | 14 | 13       | . 12 | 11 | 10 | 9  | 8        | . 7      | 6 | 5        | . 4 | 3  | 2  | 1  | 0 |
|--|----------|----|----------|------|----|----|----|----------|----------|---|----------|-----|----|----|----|---|
|  | Reserved |    | DBL[4:0] |      |    |    |    | Reserved |          |   | DBA[4:0] |     |    |    |    |   |
|  |          |    | rw       | rw   | rw | rw | rw |          | Reserved | l | rw       | rw  | rw | rw | rw |   |

Bits 15:13 Reserved, must be kept at reset value.

```
Bits 12:8 DBL[4:0]: DMA burst length
```

This 5-bit vector defines the number of DMA transfers (the timer recognizes a burst transfer when a read or a write access is done to the TIMx\_DMAR address).

00000: 1 transfer, 00001: 2 transfers, 00010: 3 transfers, ... 10001: 18 transfers.

Bits 7:5 Reserved, must be kept at reset value.

### Bits 4:0 DBA[4:0]: DMA base address

This 5-bit vector defines the base-address for DMA transfers (when read/write access are done through the TIMx\_DMAR address). DBA is defined as an offset starting from the address of the TIMx\_CR1 register.

Example: 00000: TIMx\_CR1, 00001: TIMx\_CR2, 00010: TIMx\_SMCR,



#### Root cause:

The function mechanism of TIMDMABurst is different from SXX32F103.

#### For SXX32F103:

When the Advanced & General TIM configures DBL to non-0, a DMA request is generated and DMA carries (DBL+1) data

#### For MH2103C:

When the Advanced & General TIM configures DBL to be non-0, one DMA request is generated and one data is transferred by DMA

#### Solution:

The software can configure TIM to send different DMA request signals to different DMA channels at the same time, and each DMA channel carries a data to achieve the TIMDMABurst effect of SXX32F103.

#### 3.4 CAN module

### $3.\,4.\,1$ Precautions for configuring a filter in hibernation mode

### Problem description:

The CAN module first initializes the filter and then initializes the controller, resulting in the phenomenon that data can be sent but cannot be received.

The code is as follows:

#### Root cause:

MH2103A When configuring the CAN filter, ensure that the CAN is in non-sleep mode.

### Solution:

To solve this problem, initialize the CAN controller and then configure the filter.



### 3.4.2 The sending timestamp function is different

### Problem description:

Due to the different design, MH2103A uses CAN to send the timestamp, and the TIME[15:0] field in the CAN\_TDTxR register is different from that in SXX32F103 to fill the message location.

#### For SXX32F103:

TIME[7:0] as the seventh byte and TIME[15:8] as the eighth byte replace the data written to CAN\_TDHxR[31:16] (DATA6[7:0] and DATA7[7:0]).

| Bit 31:16 | TIME[15:0]: Message time stamp This field contains the value of the 16-bit timer at the time the packet is sent.                                                                                                                                                                                                                                                                                                                                                                                                                 |
|-----------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Bit 15:9  | Reserve the bit.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| Bit 8     | TGT: Transmit global time This bit is only valid if CAN is in time-triggered communication mode, i.e. the TTCM bit of the CAN MCR register is "1". 0; Do not send timestamp TIME[15:0]; 1: Send timestamp TIME[15:0]. In a message of length 8, the timestamp TIME[15:0] is the last two bytes sent:  TIME[7:0] as the seventh byte and TIME[15:8] as the eighth byte replace the data written to CAN_TDHxR [31:16] (DATA6[7:0] and DATA7[7:0]). In order to send out 2 bytes of the timestamp, the DLC must be programmed to 8. |
| Bit 7:4   | Reserve the bit.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| Bit 3:0   | <b>DLC[15:0]</b> : Send Data length code This field specifies the data length of the data message or the data length of the remote frame request. A message contains 0 to 8 bytes of data, which is determined by the DLC.                                                                                                                                                                                                                                                                                                       |





#### For MH2103A:

TIME[7:0] as the eighth byte and TIME[15:8] as the seventh byte replace the data written to CAN\_TDHxR[31:16] (DATA6[7:0] and DATA7[7:0]).



### 3.4.3 When TXFP is 1, the FIFO priority is different

### Problem description:

Due to different designs, the TXFP is set to 1 for the MH2103A. When multiple packets are waiting to be sent, the priority of sending packets is different from that of the SXX32F103.

The initial CAN configuration is as follows:



```
//CAN unit Settings
CAN_InitStructure.CAN_TTCM=DISABLE;
                                                   // Non-time triggered communication mode
                                                   // Software automatically offline management
CAN_InitStructure.CAN_ABOM=DISABLE;
CAN InitStructure.CAN AWUM=DISABLE;
                                                   // SLEEP mode awakened by software (clear sleep bit of CAN->MCR)
CAN InitStructure.CAN NART=ENABLE;
                                                   // Disable automatic packet transmission
                                                   // The packet is not locked. The new packet overwrites the old one
CAN InitStructure.CAN RFLM=DISABLE;
CAN InitStructure. CAN TXFP=ENABLE;
CAN_InitStructure.CAN_Mode= mode;
                                                  // mode Settings: mode:0, common mode; 1, loop mode;
// Set the baud rate
CAN InitStructure.CAN SJW=tsjw;
                                             The TXFP configuration is set to 1
CAN InitStructure.CAN BS1=tbs1;
CAN InitStructure.CAN BS2=tbs2;
CAN_InitStructure.CAN_Prescaler=brp;
CAN_Init(CAN1, &CAN_InitStructure);
```

The interface for sending packets can be defined as follows:

```
void CAN TransmitTest (CAN TypeDef* CANx, CanTxMsg* TxMessage, uint8 t TxMailbox)
1
    uint8_t transmit_mailbox = 0;
    transmit mailbox = TxMailbox;
                                                                  Email address
    CANx->sTxMailBox[transmit mailbox].TIR &= 1;
    if (TxMessage->IDE == CAN_Id_Standard)
      CANx->sTxMailBox[transmit mailbox].TIR |= ((TxMessage->StdId << 21) | \
                                                   TxMessage->RTR);
    }
    else
    1
      CANx->sTxMailBox[transmit mailbox].TIR |= ((TxMessage->ExtId << 3) | \
                                                   TxMessage->IDE | \
                                                   TxMessage->RTR);
    TxMessage->DLC &= (uint8_t)0x0000000F;
    CANx->sTxMailBox[transmit mailbox].TDTR &= (uint32 t) 0xFFFFFFF0;
    CANx->sTxMailBox[transmit_mailbox].TDTR |= TxMessage->DLC;
    CANx->sTxMailBox[transmit mailbox].TDLR = (((uint32 t)TxMessage->Data[3] << 24) |
                                              ((uint32 t)TxMessage->Data[2] << 16) |
                                              ((uint32_t)TxMessage->Data[1] << 8) |
                                              ((uint32 t)TxMessage->Data[0]));
    CANx->sTxMailBox[transmit mailbox].TDHR = (((uint32 t)TxMessage->Data[7] << 24) |
                                              ((uint32 t)TxMessage->Data[6] << 16) |
                                              ((uint32_t)TxMessage->Data[5] << 8)
                                              ((uint32 t)TxMessage->Data[4]));
    CANx->sTxMailBox[transmit mailbox].TIR |= 1;
```

The filling sequence of the sent packets is as follows:



```
uint8_t Can_Send_Msg(uint8_t* msg,uint8_t len)
    uint8_t mbox;
    uint16 t i=0;
    CanTxMsg TxMessage;
                                     // Set the extension identifier
    TxMessage.ExtId=0x00;
    TxMessage.IDE=CAN Id Standard;
                                    // Standard frame
    TxMessage.RTR=CAN RTR Data;
                                     // Data frame
                                     // Length of data to be sent
    TxMessage.DLC=len;
    for(i=0;i<len;i++)</pre>
    TxMessage.Data[i]=msg[i];
   TxMessage.StdId=0x12;
                                                 Fill mailbox 2 with ID 0x12
    CAN TransmitTest(CAN1,
                           &TxMessage, 2);
    TxMessage.StdId=0x13;
                                                   Fill mailbox 1 with ID 0x13
    CAN TransmitTest (CAN1,
                           &TxMessage, 1);
    TxMessage.StdId=0x14;
                                                  Fill mailbox 0 with ID 0x14
    CAN TransmitTest (CAN1,
                           &TxMessage, 0);
```

CAN packets are actually sent

### For SXX32F103:

If the TXFP value is 1 and multiple packets are to be sent, the order of sending packets is determined by the order of requests.

| Bit 2 | TXFP: Transmit FIFO priority When multiple packets are waiting to be sent at the same time, this bit determines the order in which these packets are sent 0: The priority is determined by the packet identifier. 1: Priority is determined by the order in which requests are sent. |
|-------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|-------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|

The sequence of messages is as follows:

| CAN channel | Transmission direction | ID number | Frame type | Frame format      | Length | Data                        |
|-------------|------------------------|-----------|------------|-------------------|--------|-----------------------------|
| ch1         | recel ve               | 0x0012    | Data frame | Standard<br>frame | 0x08   | x   5A 5B 5C 5D 5E 5F 60 61 |
| ch1         | recel ve               | 0x0013    | Data frame | Standard<br>frame | 0x08   | x   5A 5B 5C 5D 5E 5F 60 61 |
| ch1         | recei ve               | 0x0014    | Data frame | Standard<br>frame | 0x08   | x   5A 5B 5C 5D 5E 5F 60 61 |

#### For MH2103A:

When the TXFP value is 1 and multiple packets are to be sent, the order of sending packets is determined by the priority of the mailbox number. Email priority: Email 0> Email 1> Email 2.

The sequence of messages is as follows:



### 3.5 FLASH module

3.5.1 There is a probability Fault when calling the FLASH\_EraseOptionBytes() interface to erase option bytes

Problem description:



When the program uses the FLASH\_EraseOptionBytes() interface to erase the option byte area, there is a probability that the HardFault\_Handler exception will be interrupted.

#### Solution:

MH2103A modifies the FLASH\_EraseOptionBytes() interface as follows:

Replace the following instructions with the SetStrt() interface:

```
FLASH->CR |= CR OPTER Set;
FLASH->CR |= CR_STRT_Set;
FLASH WaitForLastOperation(EraseTimeout);
FLASH_Status FLASH_EraseOptionBytes(void)
  uint16_t rdptmp = RDP_Key;
  FLASH_Status status = FLASH_COMPLETE;
  /* Get the actual read protection Option Byte value */
  if (FLASH GetReadOutProtectionStatus() != RESET)
    rdptmp = 0x00;
  }
  /* Wait for last operation to be completed */
  status = FLASH WaitForLastOperation(EraseTimeout);
  if(status == FLASH COMPLETE)
    /* Authorize the small information block programming */
    FLASH->OPTKEYR = FLASH KEY1;
    FLASH->OPTKEYR = FLASH KEY2;
    /st if the previous operation is completed, proceed to erase the option bytes ^st/
     FLASH->CR |= CR_OPTER_Set;
      FLASH->CR |= CR STRT Set;
     /* Wait for last operation to be completed */
      status = FLASH_WaitForLastOperation(EraseTimeout);
    SetStrt();
    if(status == FLASH COMPLETE)
      /st if the erase operation is completed, disable the OPTER Bit st/
      FLASH->CR &= CR_OPTER_Reset;
      /* Enable the Option Bytes Programming operation */
      FLASH->CR |= CR OPTPG Set;
      /* Restore the last read protection Option Byte value */
      OB->RDP = (uint16 t)rdptmp;
      /* Wait for last operation to be completed */
      status = FLASH_WaitForLastOperation(ProgramTimeout);
      if (status != FLASH TIMEOUT)
        /st if the program operation is completed, disable the OPTPG Bit st/
        FLASH->CR &= CR OPTPG Reset;
    1
    else
```

The SetStrt() interface is shown in the following figure, which is implemented in mh32f10x\_flash.c.



```
#if defined( CC_ARM)
__ASM_void SetStrt(void)
    MOV
           RO, PC
   LDR
           R1, [R0,#16]
           R1, [R0,#32]
   LDR
   LDR
          R0, =0x40022010
          R1, =0x60
   LDR
   STR
          R1, [R0]
   NOP
   NOP
   NOP
   NOP
   NOP
   NOP
FLAGLABLE
           R1, =0x4002200C
   LDR
   LDR
           R2, [R1]
   AND
           R2, #0x01
   CMP
           R2, #0x00
   BNE
          FLAGLABLE
           1r
}
#elif defined(__ICCARM__)
void SetStrt(void)
{
     ASM ("MOV
                  RO, PC\n"
         "LDR
                 R1, [R0,#16]\n"
         "LDR
                 R1, [R0,#32]\n"
         "LDR
                 R0, =0x40022010\n"
         "LDR
                 R1, =0x60\n"
         "STR
                  R1, [R0]\n"
         "NOP\n"
         "NOP\n"
         "NOP\n"
         "NOP\n"
         "NOP\n"
         "NOP\n"
         "FLAGLABLE:\n"
         "LDR R1, =0x4002200C\n"
                  R2, [R1]\n"
         "LDR
                 R2, R2, #0x01\n"
         "AND
         "CMP
                 R2, #0x00\n"
         "BNE
                  FLAGLABLE\n"
         "BX
                  lr");
#elif defined(__GNUC__)
void SetStrt(void)
   asm("MOV
                   RO, PC");
   asm("LDR
                  R1, [R0,#16]");
                  R1, [R0,#32]");
   asm("LDR
```



#### 3.6 USART module

### 3.6.1 Smart card mode has no clock output

### Problem description:

```
When only CLK is configured in smart card mode, there is no clock output. The
configuration is as follows:
void UART_Configuration(uint32_t bound)
    GPIO InitTypeDef GPIO InitStructure;
    USART_InitTypeDef USART_InitStructure;
    USART_ClockInitTypeDef USART_ClockInitStruct;
    RCC APB2PeriphClockCmd(RCC APB2Periph USART1, ENABLE);
    RCC APB2PeriphClockCmd (RCC APB2Periph GPIOA, ENABLE);
    GPIO InitStructure.GPIO Pin = GPIO Pin 8;
    GPIO InitStructure.GPIO Speed = GPIO Speed 50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO Init(GPIOA, &GPIO InitStructure);
    USART ClockInitStruct.USART Clock = USART Clock Enable;
    USART ClockInitStruct.USART CPHA = USART CPOL Low;
    USART ClockInitStruct.USART CPOL =USART CPHA 1Edge;
    USART ClockInitStruct.USART LastBit = USART LastBit Disable;
                                         Configure clock frequency division
   USART SetPrescaler(USART TEST, 8);
                                                The smartcard mode was enabled
   USART SmartCardCmd(USART TEST,ENABLE);
    USART ClockInit (USART TEST, &USART ClockInitStruct);
}
```

#### Root cause:

The clock output condition of the MH2103AUSART smart card mode is different from that of the SXX32F103, and is also controlled by the send enable TE.

### Solution:

After the clock frequency division is configured and the smart card mode is enabled, set the transmit TE function to 1.

As shown in the picture below:



```
void UART Configuration (uint32 t bound)
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;
    USART_ClockInitTypeDef USART_ClockInitStruct;
    RCC APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO InitStructure.GPIO Mode = GPIO Mode AF PP;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    USART_ClockInitStruct.USART_Clock = USART_Clock_Enable;
    USART ClockInitStruct.USART CPHA = USART CPOL Low;
    USART_ClockInitStruct.USART_CPOL =USART_CPHA_1Edge;
    USART_ClockInitStruct.USART_LastBit = USART_LastBit_Disable;
    USART_SetPrescaler(USART_TEST, 8);
   USART_SmartCardCmd(USART_TEST, ENABLE);
    USART ClockInit(USART TEST, &USART ClockInitStruct);
    USART_InitStructure.USART_BaudRate = bound;
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;
USART_InitStructure.USART_StopBits = USART_StopBits_1;
    USART_InitStructure.USART_Parity = USART_Parity_No;
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
    USART_InitStructure.USART_Mode = USART_Mode_Tx;
    USART_Init(USART_TEST, &USART_InitStructure);
                                                            Enabling TE
```

3.6.2 When DMA is used to send data, USARTTC flags are used to determine whether all data is sent. Precautions

#### Problem description:

When DMA is used to send data larger than 2Byte, the USARTTC flag is used to confirm whether the data is sent. For example:

USART1 is configured normally, and RE and TE are enabled. As shown in the following picture

```
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
RCC APB2PeriphClockCmd (RCC APB2Periph GPIOA, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
GPIO InitStructure.GPIO Speed = GPIO Speed 50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO InitStructure.GPIO Mode = GPIO Mode IN FLOATING;
GPIO Init(GPIOA, &GPIO InitStructure);
USART_InitStructure.USART_BaudRate = bound;
USART_InitStructure.USART_WordLength = USART_WordLength_8b;
USART_InitStructure.USART_StopBits = USART_StopBits_1;
USART_InitStructure.USART_Parity = USART_Parity_No;
USART InitStructure.USART HardwareFlowControl = USART HardwareFlowControl None;
USART InitStructure.USART Mode = USART Mode Rx | USART Mode Tx;
USART_Init(USART1, &USART_InitStructure);
USART_Cmd(USART1, ENABLE);
```

 $\,\succ\,$  Configure and enable the DMA channel to carry data. As shown in the following picture



```
DMA_DeInit(DMA1_Channel4);

DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&USART1->DR;

DMA_InitStructure.DMA_MemoryBaseAddr = (u32)pdat;

DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;

DMA_InitStructure.DMA_BufferSize = length;

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;

DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;

DMA_InitStructure.DMA_MemoryDataSize = DMA_PeripheralDataSize_Byte;

DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;

DMA_InitStructure.DMA_MOMA_Priority = DMA_Priority_Low;

DMA_InitStructure.DMA_MOMA_MOMA_Priority_Low;

DMA_InitStructure.DMA_MOMA_InitStructure);

USART_DMACmd(USART1, USART_DMAReq_Tx, ENABLE);

DMA_Cmd(DMA1_Channel4, ENABLE);
```

> After DMATC is set, determine the USARTTC flag. As shown in the following picture

```
while(DMA_GetFlagStatus(DMA1_FLAG_TC4) != SET);
while(USART GetFlagStatus(USART1, USART FLAG TC) != SET);
```

Detect indicating signals and data sent by USARTTX pins. As shown in the following picture

### For SXX32F103:



### For MH2103A:



#### Solution:

Before configuring DMA, disable TE. After DMA is configured, TE is enabled to send data. As shown in the following picture



```
USART1->CR1 &= ~BIT(3); → Disable TE
DMA_DeInit(DMA1_Channel4);
DMA InitStructure.DMA PeripheralBaseAddr = (u32) &USART1->DR;
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)pdat;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;

DMA_InitStructure.DMA_BufferSize = length;

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
                                              = DMA_DIR_PeripheralDST;
DMA InitStructure.DMA PeripheralDataSize = DMA PeripheralDataSize Byte;
DMA_InitStructure.DMA_MemoryDataSize = DMA_PeripheralDataSize_Byte;
DMA InitStructure.DMA Mode
                                              = DMA Mode Normal;
DMA_InitStructure.DMA_Priority = DMA_Priority_Low;
DMA InitStructure.DMA M2M
                                                = DMA M2M Disable;
DMA Init(DMA1 Channel4, &DMA InitStructure);
USART DMACmd (USART1, USART DMAReg Tx, ENABLE);
DMA Cmd (DMA1 Channel4, ENABLE);
USART1->CR1 \mid= BIT(3); — Enable TE
```

### 3.7 SPI module

3.7.1 As the Master, this section describes the precautions for sending and receiving data

### in full-duplex mode

### Problem description:

When communicating with SPI as Master, the received data is abnormal.

For example:

The SPI is configured in Master two-wire full-duplex mode. As shown in the following picture

```
SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;
SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;
SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;
SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
SPI_InitStructure.SPI_CRCPolynomial = 7;
SPI_Init(SPIx, &SPI_InitStructure);
```

> Send and receive data according to the following figure.



```
// Send Data0, Datal; Do not operate to receive DR
SPI I2S SendData(SPIx, Data0);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG TXE) == RESET);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG BSY) == RESET);
SPI I2S SendData(SPIx, Datal);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG TXE) == RESET);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG BSY) == RESET);
// Read the DR Once to clear the previously received data
while (SPI_I2S_GetFlagStatus(SPIx, SPI_I2S_FLAG_RXNE) == RESET);
SPI I2S ReceiveData(SPIx);
// Send data Data2; And read DR, get the received data receive
SPI I2S SendData(SPIx, Data2);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG TXE) == RESET);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG BSY) == RESET);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG RXNE) == RESET);
receive = SPI I2S ReceiveData(SPIx);
```

MH2103A sends Data2 The received data is abnormal data.

#### Root cause:

The MH2103ASPI has a 16Byte receiving FIFO. The data received by Data1 is cached in the receiving FIFO, and the operation of reading the DR Once does not clear all the useless data in the FIFO, resulting in the data received by the subsequent sending Data2 is not the expected data.

#### Solution:

As a Master, the DR Is read every time the data is sent, ensuring that no residual data in the FIFO can be resolved.

As shown in the picture below:

```
// Send Datao, Datal; The DE is read once after each data transmission
SPI I2S SendData(SPIx, Data0);
while (SPI_I2S_GetFlagStatus(SPIx, SPI_I2S_FLAG_TXE) == RESET);
while (SPI_I2S_GetFlagStatus(SPIx, SPI_I2S_FLAG_BSY) == RESET);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG RXNE) == RESET);
SPI I2S ReceiveData(SPIx);
SPI I2S SendData(SPIx, Datal);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG TXE) == RESET);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG BSY) == RESET);
while (SPI_I2S_GetFlagStatus(SPIx, SPI_I2S_FLAG_RXNE) == RESET);
SPI I2S ReceiveData(SPIx);
// Send data Data2; And read the DR To get the received data receive
SPI I2S SendData(SPIx, Data2);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG TXE) == RESET);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG BSY) == RESET);
while (SPI I2S GetFlagStatus(SPIx, SPI I2S FLAG RXNE) == RESET);
receive = SPI I2S ReceiveData(SPIx);
```



### 3.7.2 If the SPE is closed in BUSY state, data may not be sent or received

### Problem description:

When the SPI is operated, if the software does not determine the non-busy status of the SPE when it is turned off, data may fail to be sent and received. Operation as shown below:

The MH2103A can normally send 0xAA, but cannot send 0x55.

```
SPI_I2S_SendData(SPI1,0xAA);
while(SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_BSY) == RESET);
SPI_Cmd(SPI1, DISABLE);
SPI_Cmd(SPI1, ENABLE);
while(SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_TXE) == RESET);
SPI_I2S_SendData(SPI1,0x55);
```

#### Root cause:

MH2103ASPI When the SPE device is in BUSY state, shutting down the SPE device takes effect when data is sent (when the SPE device is not in BUSY state), but ignoring the enabling action does not take effect.

### Solution:

Before shutting down the SPE, check whether the SPE is in BUSY state.

You can shut down the SPE only when the SPE is not BUSY.

### 3.7.3 Using DMA to transfer data configuration differences

#### Problem description:

When the SPI uses DMA to transmit data, the data received is incorrect.

### Solution:

When the SPI uses DMA to transmit data, both the peripheral DAM and the DMA channel must be enabled/disabled.

As shown in the picture below:

```
DMA_Cmd(FLASH_SPI_RX_DMA_CHANNEL, ENABLE);

DMA_Cmd(FLASH_SPI_TX_DMA_CHANNEL, ENABLE);

SPI_I2S_DMACmd(FLASH_SPI_MASTER, SPI_I2S_DMAReq_Tx, ENABLE);

SPI_I2S_DMACmd(FLASH_SPI_MASTER, SPI_I2S_DMAReq_Rx, ENABLE);

SPI_I2S_DMACmd(FLASH_SPI_MASTER, SPI_I2S_DMAReq_Tx, DISABLE);

SPI_I2S_DMACmd(FLASH_SPI_MASTER, SPI_I2S_DMAReq_Rx, DISABLE);

DMA_Cmd(FLASH_SPI_TX_DMA_CHANNEL, DISABLE);

DMA_Cmd(FLASH_SPI_RX_DMA_CHANNEL, DISABLE);
```



### 3.8 USB module

### 3.8.1 USB use precautions

### Problem description:

When USB is configured in Slave mode, the Host sends requests from different endpoints or from the same endpoint in different directions in a short period of time, and there is a probability of crash.

#### Root cause:

The update mechanism of EP\_ID and DIR in USB\_ISTR(Interrupt status register) of MH2103AUSB is different from that of SXX32F103.

#### Solution:

void CTR\_LP(void)

Modify the CTR\_LP() interface, as shown in the following figure. After modification, it is compatible with SXX32F103.

```
uint32_t i = 0;
uint16_t nstr = 0;
__IO uint16_t wEFVal = 0;

/* stay in loop while pending interrupts */

while (((wIstr = _GetISTR()) & ISTR_CTR) != 0)
  for(i = 0;i < 8; i++)
        nstr = _GetENDPOINT(i);
if(nstr & (EP_CTR_RX|EP_CTR_TX))
              EPindex = i;
if(nstr & EP_CTR_RX)
                   wIstr |= ISTR_DIR;
              if (nstr & EP_CTR_TX)
                   wIstr &= ISTR_DIR;
     /* extract highest priority endpoint number */
EPindex = (uint8_t) (wIstr & ISTR_EP_ID);
  if (EPindex == 0)
     /\!\!\!\!\!\!^\star Decode and service control endpoint interrupt ^*/\!\!\!\!
         calling related service routine *
      /* (Setup0_Process, In0_Process, Out0_Process) */
      /* save RX & TX status */
/* and set both to NAK */
        SaveRState = _GetENDPOINT(ENDPO);
SaveTState = SaveRState & EPTX_STAT;
SaveRState &= EPRX_STAT;
        _SetEPRxTxStatus(ENDPO,EP_RX_NAK,EP_TX_NAK);
      /* DIR bit = origin of the interrupt */
      if ((wIstr & ISTR_DIR) == 0)
        /* DIR = 0 */
```



### 3.9 DMA module

### 3.9.1 DMA usage precautions

### Problem description:

When data is being transferred by a Channel in the DMA module, modifying the CNT value (data transfer quantity) of another Channel may not take effect.

#### Root cause:

The CNT(Data Transfer quantity) modification of the MH2103ADMAChannel takes effect under different conditions than that of the SXX32F103.

#### Solution:

The software prevents the DMA Channel from modifying the CNT value (the amount of data transferred) of another Channel while it is moving data.

### 3.10 IIC module

# 3. 10. 1 Send start bit & Stop bit as Master Note Problem description:

As shown in the figure below, the MH2103C start Hold time  $(t_{h(STA)})$  &stop Hold time  $(t_{SU(STO)})$  are different from SXX32F103.

When the peer end is an I/O analog Slave, check whether the peer end has requirements on  $t_{h(STA)}$  and  $t_{SU(STO)}$ .



### For SXX32F103:

 $t_{h(STA)}$  and  $t_{SU(STO)}$  equals the high level time of the SCL.

### For MH2103C:

In standard mode,  $t_{h(STA)} \& t_{SU(STO)}$  equals (FREQ\*4+8)\* $T_{APB1}$ °

In fast mode,  $t_{h(STA)} \& t_{SU(STO)}$  equals (FREQ+8) $*T_{APB1}$ .



### Root cause:

The design is different.

### Solution:

If you have requirements for  $t_{h(STA)}$  and  $t_{SU(STO)}$  time, you can adjust the time by modifying FREQ or modifying APB1CLK.

## 4. ISP, emulator, offline burner use precautions

### 4.1 ISP tool

MH2103A supports the ISP tools commonly used in the market today. The programmer includes STMFlashLoader, FlyMcu, and STM32CubeProgrammer and so on.

STMFI ashLoader:



FlyMcu:





### STM32CubeProgrammer:



MH2103A does not support mcuisp V0.993.





### 4.2 Emulator

Emulators supported by MH2103A include ST-LINKV2, JLINK, ARM Emulator, etc.

The MH2103A does not support ST-LINKV3.

### 4.2.1 Precautions for using Keil.SXX32F1xx\_DFP.2.3.0 or later pack

### Problem description:

When using ULINK2 or CMSIS-DAP emulator and using pack package Keil.SXX32F1xx \_DFP.2.3.0 or later (including version 2.3.0), the program cannot be downloaded. The following information is displayed during the download.



### Solution:

Deselect the enable option in pack, as shown in the following figure.





### 4.3 The downloader

At present, there are many downloaders on the market, and the following lists the precautions for the use of offline downloaders: MH2103A fully supported downloaders: 金雕脱机下载器(Jindiaooffline downloaders)



The MH2103A does not support SmartPRO T9000-PLUS and XELTEKSUPERPRO 6100N.



### 4.3.1 The chip is burned abnormally in the read/write protection state

### Problem description:

When the chip equipped with read/write protection is burned by using the off-line burner such as positive atom MINI and positive atom P100, the burning failure occurs.

### Solution:

After the failure, power on or off the MCU again or Reset the external reset. Burn again to burn successfully.

Note: If the chip is empty, using the above burner, there will be no burning failure problem.

### 4.3.2 Failed to burn using the WizPro200ST8 programmer

### Problem description:

The burning failure occurred with the WizPro200ST8 programmer.

#### Solution:

Disconnect the Reset pin and chip of WizPro200ST8 programmer; Only connected VDD, SWCLK, SWDIO can be burned normally.



## Historical version

| Edi ti on | <b>Al</b> ter                                                     | Time |
|-----------|-------------------------------------------------------------------|------|
| 1.00      | Initial version                                                   |      |
| 1.01      | Added notes related to system functions, FLASH,<br>USART, SPI/IIS |      |
| 1.02      | Added SPI related precautions                                     |      |
| 1.03      | Added ADC and TIM related notes                                   |      |
| 1.04      | Added precautions for non-32bit alignment access to the APB bus   |      |
| 1.05      | Added Precautions for interrupt controller                        |      |
| 1.06      | Added notes for different modes of dual ADCs                      |      |
| 1. 07     | Added notes for ISP, emulator, and downloader                     |      |
| 1. 08     | Added different model labels                                      |      |
| 1.09      | Added the precautions for hardware migration                      |      |
| 1.10      | Add precautions for the use of system modules, CAN and DMA        |      |
| 1. 11     | Added TIM, IIC use precautions                                    |      |