Запись и чтение OTP: различия между версиями

Материал из MIK32 микроконтроллер
Нет описания правки
Нет описания правки
Строка 57: Строка 57:
void SystemClock_Config(void);
void SystemClock_Config(void);
static void OTP_Init(void);
static void OTP_Init(void);
</syntaxhighlight>Функция main должно выглядеть примерно так:
</syntaxhighlight>Запись осуществляется при 8В на выводе VPRG. Читать можно сразу после операции записи при высоком программирующем напряжении, но будет достаточно и 3.3В.
 
Функции для записи:
 
* void HAL_OTP_WriteTestColumn - запись массива данных в тестовый столбец;
* void HAL_OTP_WriteTestRow - запись массива данных в тестовую строку;
* void HAL_OTP_WriteTestBit - запись бита в тестовую ячейку на пересечении тестовой строки и тестового столбца;
* void HAL_OTP_Write - запись массива данных в основной массив OTP.
 
Функции для чтения:
 
* void HAL_OTP_ReadTestColumn(OTP_HandleTypeDef *hotp, uint8_t Address, uint32_t DataRead[], uint32_t DataLength);
* uint32_t HAL_OTP_ReadTestRow(OTP_HandleTypeDef *hotp);
* uint32_t HAL_OTP_ReadTestBit(OTP_HandleTypeDef *hotp);
* void HAL_OTP_Read(OTP_HandleTypeDef *hotp, uint8_t Address, uint32_t DataRead[], uint32_t DataLength);
 
Функция main должно выглядеть примерно так:

Версия от 10:04, 14 февраля 2023

В примере будут записаны данные в основной массив OTP, тестовый столбец и тестовую строку, после чего они будут считаны.

Работа с конфигуратором (В разработке)

Для начала настроем в конфигураторе тактирование mik32, например, от внешнего кварца 32МГц. Затем настроем делители шины. Так как OTP тактируется от шины APB_M_CLK, то зададим делители AHB_DIV и APB_M_DIV. В данном примере оставим делители по умолчанию. В итоге вкладка с тактированием должна выглядеть так:

(Картинка тактирования из конфигуратора. В работе)

Затем перейдем к настройке самого OTP. Для этого откроем вкладку OTP и нажмем включить.

После этого появится одна настройка - режим чтения. При выборе "чтения в 2 этапа" происходит автоматическое инкрементирование адреса OTPA после чтения, вводятся такты ожидания интерфейса APB. При выборе "чтения в 3 этапа" автоматического инкрементирования OTPA не происходит, такты ожидания не вводятся, но требуется опрос флага готовности BSY. Выберем, например, чтение в 3 этапа.

В итоге настройки таймера в конфигураторе должны выглядеть примерно так:

(Объяснение работы с конфигуратором. В разработке)

Нажимаем кнопку генерации. В итоге у нас появится проект для PlatformIo. Далее работа идет в visual studio code.

Использование библиотеки HAL_OTP

В сгенерированном проекте в файле main.c должна быть функция OTP_Init, в которой будут заданы настройки для OTP. Выглядит она так:

static void OTP_Init(void)
{
    hotp.Instance = OTP;

    hotp.ReadMode = OPT_READ_3STAGES;

    HAL_OTP_Init(&hotp);
}

Кроме этого в функции SystemClock_Config приведены настройки для тактирования. Убедитесь что в PeriphClkInit.PMClockAPB_M присутствует PM_CLOCK_OTP_CONTROLLER_M. Сама функция должна выглядеть примерно так:

void SystemClock_Config(void)
{
    RCC_OscInitTypeDef RCC_OscInit = {0};
    RCC_PeriphCLKInitTypeDef PeriphClkInit = {0};

    RCC_OscInit.OscillatorEnable = RCC_OSCILLATORTYPE_OSC32K | RCC_OSCILLATORTYPE_OSC32M;   
    RCC_OscInit.OscillatorSystem = RCC_OSCILLATORTYPE_OSC32M;                          
    RCC_OscInit.AHBDivider = 0;                             
    RCC_OscInit.APBMDivider = 0;                             
    RCC_OscInit.APBPDivider = 0;                             
    RCC_OscInit.HSI32MCalibrationValue = 0;                  
    RCC_OscInit.LSI32KCalibrationValue = 0;
    HAL_RCC_OscConfig(&RCC_OscInit);

    PeriphClkInit.PMClockAHB = PMCLOCKAHB_DEFAULT;    
    PeriphClkInit.PMClockAPB_M = PMCLOCKAPB_M_DEFAULT | PM_CLOCK_WU_M | PM_CLOCK_PAD_CONFIG_M | PM_CLOCK_OTP_CONTROLLER_M;     
    PeriphClkInit.PMClockAPB_P = PMCLOCKAPB_P_DEFAULT | PM_CLOCK_UART_0_M;     
    PeriphClkInit.RTCClockSelection = RCC_RTCCLKSOURCE_NO_CLK;
    PeriphClkInit.RTCClockCPUSelection = RCC_RTCCLKCPUSOURCE_NO_CLK;
    HAL_RCC_ClockConfig(&PeriphClkInit);
}

Для демонстрации вывода текста в PeriphClkInit.PMClockAPB_P присутствует PM_CLOCK_UART_0_M. У вас его может не быть так как UART нужно включить отдельно. В начале main.c можно видеть объявление структуры с набором настроек для OTP, которую использует функция инициализации OTP_Init.

OTP_HandleTypeDef hotp;

void SystemClock_Config(void);
static void OTP_Init(void);

Запись осуществляется при 8В на выводе VPRG. Читать можно сразу после операции записи при высоком программирующем напряжении, но будет достаточно и 3.3В.

Функции для записи:

  • void HAL_OTP_WriteTestColumn - запись массива данных в тестовый столбец;
  • void HAL_OTP_WriteTestRow - запись массива данных в тестовую строку;
  • void HAL_OTP_WriteTestBit - запись бита в тестовую ячейку на пересечении тестовой строки и тестового столбца;
  • void HAL_OTP_Write - запись массива данных в основной массив OTP.

Функции для чтения:

  • void HAL_OTP_ReadTestColumn(OTP_HandleTypeDef *hotp, uint8_t Address, uint32_t DataRead[], uint32_t DataLength);
  • uint32_t HAL_OTP_ReadTestRow(OTP_HandleTypeDef *hotp);
  • uint32_t HAL_OTP_ReadTestBit(OTP_HandleTypeDef *hotp);
  • void HAL_OTP_Read(OTP_HandleTypeDef *hotp, uint8_t Address, uint32_t DataRead[], uint32_t DataLength);

Функция main должно выглядеть примерно так: