Запись и чтение OTP: различия между версиями
Андрей (обсуждение | вклад) Нет описания правки |
Дмитрий (обсуждение | вклад) (исправил библиотеки) |
||
(не показано 6 промежуточных версий 2 участников) | |||
Строка 10: | Строка 10: | ||
После этого появится одна настройка - режим чтения. При выборе "чтения в 2 этапа" происходит автоматическое инкрементирование адреса OTPA после чтения, вводятся такты ожидания интерфейса APB. При выборе "чтения в 3 этапа" автоматического инкрементирования OTPA не происходит, такты ожидания не вводятся, но требуется опрос флага готовности BSY. Выберем, например, чтение в 3 этапа. | После этого появится одна настройка - режим чтения. При выборе "чтения в 2 этапа" происходит автоматическое инкрементирование адреса OTPA после чтения, вводятся такты ожидания интерфейса APB. При выборе "чтения в 3 этапа" автоматического инкрементирования OTPA не происходит, такты ожидания не вводятся, но требуется опрос флага готовности BSY. Выберем, например, чтение в 3 этапа. | ||
В итоге настройки | В итоге настройки OTP в конфигураторе должны выглядеть примерно так: | ||
(Объяснение работы с конфигуратором. В разработке) | (Объяснение работы с конфигуратором. В разработке) | ||
Строка 17: | Строка 17: | ||
== Использование библиотеки HAL_OTP == | == Использование библиотеки HAL_OTP == | ||
В сгенерированном проекте в файле main.c должна быть функция OTP_Init, в которой будут заданы настройки для OTP. Выглядит она так:<syntaxhighlight lang="c" line="1"> | В сгенерированном проекте в файле main.c должна быть функция OTP_Init, в которой будут заданы настройки для OTP. Выглядит она так: | ||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | |||
static void OTP_Init(void) | static void OTP_Init(void) | ||
{ | { | ||
Строка 27: | Строка 29: | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
}} | |||
Кроме этого в функции SystemClock_Config приведены настройки для тактирования. Убедитесь что в PeriphClkInit.PMClockAPB_M присутствует PM_CLOCK_OTP_CONTROLLER_M. Сама функция должна выглядеть примерно так:<syntaxhighlight lang="c" line="1"> | Кроме этого в функции SystemClock_Config приведены настройки для тактирования. Убедитесь что в PeriphClkInit.PMClockAPB_M присутствует PM_CLOCK_OTP_CONTROLLER_M. Сама функция должна выглядеть примерно так: | ||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | |||
void SystemClock_Config(void) | void SystemClock_Config(void) | ||
{ | { | ||
Строка 50: | Строка 54: | ||
HAL_RCC_ClockConfig(&PeriphClkInit); | HAL_RCC_ClockConfig(&PeriphClkInit); | ||
} | } | ||
</syntaxhighlight>Для демонстрации вывода текста в PeriphClkInit.PMClockAPB_P присутствует PM_CLOCK_UART_0_M. У вас его может не быть так как UART нужно включить отдельно. | </syntaxhighlight> | ||
}} | |||
Для демонстрации вывода текста в PeriphClkInit.PMClockAPB_P присутствует PM_CLOCK_UART_0_M. У вас его может не быть так как UART нужно включить отдельно. Для этого нужно подключить библиотеки uart_lib и xprintf: | |||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | |||
#include "mik32_hal_rcc.h" | |||
#include "mik32_hal_otp.h" | |||
#include "uart_lib.h" | |||
#include "xprintf.h" | |||
</syntaxhighlight> | |||
}} | |||
Для инициализации UART в функции main после функции тактирования SystemClock_Config следует написать: | |||
<syntaxhighlight lang="c" line="1" start="1"> | |||
UART_Init(UART_0, 3333, UART_CONTROL1_TE_M | UART_CONTROL1_M_8BIT_M, 0, 0); | |||
</syntaxhighlight> | |||
Скорость UART задается делителем во втором аргументе функции. При такой записи скорость будет 9600 бод. | |||
В начале main.c можно видеть объявление структуры с набором настроек для OTP, которую использует функция инициализации OTP_Init.<syntaxhighlight lang="c" line="1"> | В начале main.c можно видеть объявление структуры с набором настроек для OTP, которую использует функция инициализации OTP_Init. | ||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | |||
OTP_HandleTypeDef hotp; | OTP_HandleTypeDef hotp; | ||
void SystemClock_Config(void); | void SystemClock_Config(void); | ||
static void OTP_Init(void); | static void OTP_Init(void); | ||
</syntaxhighlight>Запись осуществляется при 8В на выводе VPRG. Читать можно сразу после операции записи при высоком программирующем напряжении, но будет достаточно и 3.3В. | </syntaxhighlight> | ||
}} | |||
Запись осуществляется при 8В на выводе VPRG. Читать можно сразу после операции записи при высоком программирующем напряжении, но для чтения будет достаточно и 3.3В. | |||
Функции для записи: | Функции для записи: | ||
Строка 72: | Строка 97: | ||
* HAL_OTP_ReadTestBit - чтение бита из тестовой ячейки на пересечении тестовой строки и тестового столбца; | * HAL_OTP_ReadTestBit - чтение бита из тестовой ячейки на пересечении тестовой строки и тестового столбца; | ||
* HAL_OTP_Read - чтение массива данных из основного массива OTP. | * HAL_OTP_Read - чтение массива данных из основного массива OTP. | ||
[[Файл:Массив OTP.jpg|мини|Рисунок 3 - данные OTP]] | |||
Запишем такие данные: | Запишем такие данные: | ||
Строка 81: | Строка 106: | ||
Согласно спецификации Hard IP, после окончания операции записи(чтения) до начала следующей за ней операции чтения(записи) должно пройти не менее 1 мкс. | Согласно спецификации Hard IP, после окончания операции записи(чтения) до начала следующей за ней операции чтения(записи) должно пройти не менее 1 мкс. | ||
После записи прочитаем всё что записали. Функция main должна выглядеть примерно так:<syntaxhighlight lang="c" line="1"> | После записи прочитаем всё что записали. Функция main должна выглядеть примерно так: | ||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | |||
int main() | int main() | ||
{ | { | ||
SystemClock_Config(); | SystemClock_Config(); | ||
UART_Init(UART_0, 3333, UART_CONTROL1_TE_M | UART_CONTROL1_M_8BIT_M, 0, 0); | |||
OTP_Init(); | OTP_Init(); | ||
Строка 163: | Строка 192: | ||
} | } | ||
</syntaxhighlight>Убедитесь что VPRG не соединен с VCC перемычкой. Теперь можно подать на вывод VPRG 8В и прошивать. Если у вас включен UART | </syntaxhighlight> | ||
}} | |||
[[Файл:OTP вывод UART.png|мини|Рисунок 4 - OTP вывод UART]] | |||
Убедитесь что VPRG не соединен с VCC перемычкой. Теперь можно подать на вывод VPRG 8В и прошивать. Если у вас включен UART, то можно видеть вывод как на рисунке 4. |
Текущая версия от 09:54, 24 марта 2023
В примере будут записаны данные в основной массив OTP, тестовый столбец и тестовую строку, после чего они будут считаны.
Работа с конфигуратором (В разработке)
Для начала настроем в конфигураторе тактирование mik32, например, от внешнего кварца 32МГц. Затем настроем делители шины. Так как OTP тактируется от шины APB_M_CLK, то зададим делители AHB_DIV и APB_M_DIV. В данном примере оставим делители по умолчанию. В итоге вкладка с тактированием должна выглядеть так:
(Картинка тактирования из конфигуратора. В работе)
Затем перейдем к настройке самого OTP. Для этого откроем вкладку OTP и нажмем включить.
После этого появится одна настройка - режим чтения. При выборе "чтения в 2 этапа" происходит автоматическое инкрементирование адреса OTPA после чтения, вводятся такты ожидания интерфейса APB. При выборе "чтения в 3 этапа" автоматического инкрементирования OTPA не происходит, такты ожидания не вводятся, но требуется опрос флага готовности BSY. Выберем, например, чтение в 3 этапа.
В итоге настройки OTP в конфигураторе должны выглядеть примерно так:
(Объяснение работы с конфигуратором. В разработке)
Нажимаем кнопку генерации. В итоге у нас появится проект для PlatformIo. Далее работа идет в visual studio code.
Использование библиотеки HAL_OTP
В сгенерированном проекте в файле main.c должна быть функция OTP_Init, в которой будут заданы настройки для OTP. Выглядит она так:
Кроме этого в функции SystemClock_Config приведены настройки для тактирования. Убедитесь что в PeriphClkInit.PMClockAPB_M присутствует PM_CLOCK_OTP_CONTROLLER_M. Сама функция должна выглядеть примерно так:
Для демонстрации вывода текста в PeriphClkInit.PMClockAPB_P присутствует PM_CLOCK_UART_0_M. У вас его может не быть так как UART нужно включить отдельно. Для этого нужно подключить библиотеки uart_lib и xprintf:
Для инициализации UART в функции main после функции тактирования SystemClock_Config следует написать:
UART_Init(UART_0, 3333, UART_CONTROL1_TE_M | UART_CONTROL1_M_8BIT_M, 0, 0);
Скорость UART задается делителем во втором аргументе функции. При такой записи скорость будет 9600 бод.
В начале main.c можно видеть объявление структуры с набором настроек для OTP, которую использует функция инициализации OTP_Init.
Запись осуществляется при 8В на выводе VPRG. Читать можно сразу после операции записи при высоком программирующем напряжении, но для чтения будет достаточно и 3.3В.
Функции для записи:
- HAL_OTP_WriteTestColumn - запись массива данных в тестовый столбец;
- HAL_OTP_WriteTestRow - запись данных в тестовую строку;
- HAL_OTP_WriteTestBit - запись бита в тестовую ячейку на пересечении тестовой строки и тестового столбца;
- HAL_OTP_Write - запись массива данных в основной массив OTP.
Функции для чтения:
- HAL_OTP_ReadTestColumn - чтение массива данных из тестового столбца;
- HAL_OTP_ReadTestRow - чтение данных из тестовой строки;
- HAL_OTP_ReadTestBit - чтение бита из тестовой ячейки на пересечении тестовой строки и тестового столбца;
- HAL_OTP_Read - чтение массива данных из основного массива OTP.
Запишем такие данные:
- в тестовую строку - 0x12345678;
- в тестовый столбец - 0b1, 0b0, 0b1, 0b1, 0b1, 0b1, 0b0, 0b1;
- в тестовую ячейку - 0b1;
- в основной массив - 0xAAAAAAAA, 0xBBBBBBBB, 0xCCCCCCCC, 0xDDDDDDDD, 0xEEEEEEEE, 0xFFFFFFFF, 0xAAAABBBB, 0xCCCCDDDD.
Согласно спецификации Hard IP, после окончания операции записи(чтения) до начала следующей за ней операции чтения(записи) должно пройти не менее 1 мкс.
После записи прочитаем всё что записали. Функция main должна выглядеть примерно так:
Убедитесь что VPRG не соединен с VCC перемычкой. Теперь можно подать на вывод VPRG 8В и прошивать. Если у вас включен UART, то можно видеть вывод как на рисунке 4.