Запись и чтение OTP: различия между версиями
Андрей (обсуждение | вклад) Нет описания правки |
Андрей (обсуждение | вклад) Нет описания правки |
||
Строка 27: | Строка 27: | ||
} | } | ||
</syntaxhighlight>Кроме этого в функции SystemClock_Config приведены настройки для тактирования. Убедитесь что в PeriphClkInit.PMClockAPB_M присутствует PM_CLOCK_OTP_CONTROLLER_M. Сама функция должна выглядеть примерно так:<syntaxhighlight lang="c" line="1"> | </syntaxhighlight>Кроме этого в функции SystemClock_Config приведены настройки для тактирования. Убедитесь что в PeriphClkInit.PMClockAPB_M присутствует PM_CLOCK_OTP_CONTROLLER_M. Сама функция должна выглядеть примерно так:<syntaxhighlight lang="c" line="1"> | ||
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); | |||
} | |||
</syntaxhighlight><syntaxhighlight lang="c" line="1"> | |||
void SystemClock_Config(void) | void SystemClock_Config(void) | ||
{ | { |
Версия от 09:35, 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); }
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); }