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

Материал из MIK32 микроконтроллер
Нет описания правки
(исправил библиотеки)
 
(не показано 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 и в main.c или def_list.h есть #define MIK32_OTP_DEBUG, то можно видеть вывод как на рисунке.
</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.
Рисунок 3 - данные OTP

Запишем такие данные:

  • в тестовую строку - 0x12345678;
  • в тестовый столбец - 0b1, 0b0, 0b1, 0b1, 0b1, 0b1, 0b0, 0b1;
  • в тестовую ячейку - 0b1;
  • в основной массив - 0xAAAAAAAA, 0xBBBBBBBB, 0xCCCCCCCC, 0xDDDDDDDD, 0xEEEEEEEE, 0xFFFFFFFF, 0xAAAABBBB, 0xCCCCDDDD.

Согласно спецификации Hard IP, после окончания операции записи(чтения) до начала следующей за ней операции чтения(записи) должно пройти не менее 1 мкс.

После записи прочитаем всё что записали. Функция main должна выглядеть примерно так:

Рисунок 4 - OTP вывод UART

Убедитесь что VPRG не соединен с VCC перемычкой. Теперь можно подать на вывод VPRG 8В и прошивать. Если у вас включен UART, то можно видеть вывод как на рисунке 4.