Шифрование и расшифровка данных: различия между версиями
Андрей (обсуждение | вклад) Нет описания правки |
Дмитрий (обсуждение | вклад) (закоментировал и добавил UART) |
||
Строка 22: | Строка 22: | ||
В сгенерированном проекте в файле main.c должна быть функция Crypto_Init, в которой будут заданы настройки для крипто-блока. Выглядит она так: | В сгенерированном проекте в файле main.c должна быть функция Crypto_Init, в которой будут заданы настройки для крипто-блока. Выглядит она так: | ||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | <syntaxhighlight lang="c" line="1"> | ||
static void Crypto_Init(void) | static void Crypto_Init(void) | ||
Строка 34: | Строка 35: | ||
HAL_Crypto_Init(&hcrypto); | HAL_Crypto_Init(&hcrypto); | ||
} | } | ||
</syntaxhighlight>Кроме этого в функции SystemClock_Config приведены настройки для тактирования. Убедитесь что в PeriphClkInit.PMClockAHB присутствует PM_CLOCK_CRYPTO_M. Сама функция должна выглядеть примерно так: | </syntaxhighlight> | ||
}} | |||
Кроме этого в функции SystemClock_Config приведены настройки для тактирования. Убедитесь, что в PeriphClkInit.PMClockAHB присутствует PM_CLOCK_CRYPTO_M. Сама функция должна выглядеть примерно так: | |||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | <syntaxhighlight lang="c" line="1"> | ||
void SystemClock_Config(void) | void SystemClock_Config(void) | ||
Строка 58: | Строка 63: | ||
HAL_RCC_ClockConfig(&PeriphClkInit); | HAL_RCC_ClockConfig(&PeriphClkInit); | ||
} | } | ||
</syntaxhighlight> | </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_spi.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 можно видеть объявление структуры с набором настроек для крипто-блока, которую использует функция инициализации Crypto_Init. | |||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | <syntaxhighlight lang="c" line="1"> | ||
Crypto_HandleTypeDef hcrypto; | Crypto_HandleTypeDef hcrypto; | ||
Строка 66: | Строка 91: | ||
static void Crypto_Init(void); | static void Crypto_Init(void); | ||
</syntaxhighlight> | </syntaxhighlight> | ||
}} | |||
Создадим глобальный массив для ключа. Количество элементов массива можно задать с помощью макросов CRYPTO_KEY_KUZNECHIK (количество слов в ключе для алгоритма кузнечик). | Создадим глобальный массив для ключа. Количество элементов массива можно задать с помощью макросов CRYPTO_KEY_KUZNECHIK (количество слов в ключе для алгоритма кузнечик). | ||
Строка 71: | Строка 97: | ||
<syntaxhighlight lang="c" line="1"> | <syntaxhighlight lang="c" line="1"> | ||
uint32_t crypto_key[CRYPTO_KEY_KUZNECHIK] = {0x8899aabb, 0xccddeeff, 0x00112233, 0x44556677, 0xfedcba98, 0x76543210, 0x01234567, 0x89abcdef}; | uint32_t crypto_key[CRYPTO_KEY_KUZNECHIK] = {0x8899aabb, 0xccddeeff, 0x00112233, 0x44556677, 0xfedcba98, 0x76543210, 0x01234567, 0x89abcdef}; | ||
</syntaxhighlight>Для режимов шифрования ECB и CBC длина данных должна быть кратна длине блока. Иначе данные следует дополнить. | </syntaxhighlight> | ||
Для режимов шифрования ECB и CBC длина данных должна быть кратна длине блока. Иначе данные следует дополнить. | |||
Для режима CTR длинна данных может быть любой. В этом режиме дополнения не требуются. | Для режима CTR длинна данных может быть любой. В этом режиме дополнения не требуются. | ||
Синхропосылка (вектор инициализации), в соответствии с ГОСТ 34.13—2015, для разных режимов может иметь разную длину : | Синхропосылка (вектор инициализации), в соответствии с ГОСТ 34.13—2015, для разных режимов может иметь разную длину: | ||
*В режиме ECB синхропосылка не используется; | *В режиме ECB синхропосылка не используется; | ||
Строка 89: | Строка 117: | ||
*cipher_text - полученные зашифрованные данные; | *cipher_text - полученные зашифрованные данные; | ||
*expect_cipher_text - зашифрованные данные, которые ожидается получить. | *expect_cipher_text - зашифрованные данные, которые ожидается получить. | ||
Количество слов в данных для зашифровки должно быть кратно размеру блока (для кузнечика 4 слова или 128 бит). Если данных меньше, то их нужно дополнить вручную. ГОСТ 34.13—2015 определяет три возможные процедуры дополнения. Можно, например, дополнить остаток нулями до размера блока. | Количество слов в данных для зашифровки должно быть кратно размеру блока (для кузнечика 4 слова или 128 бит). Если данных меньше, то их нужно дополнить вручную. ГОСТ 34.13—2015 определяет три возможные процедуры дополнения. Можно, например, дополнить остаток нулями до размера блока. | ||
Внутри kuznechik_ECB_code сначала задается режим шифрования функцией HAL_Crypto_SetCipherMode ( | Внутри kuznechik_ECB_code сначала задается режим шифрования функцией HAL_Crypto_SetCipherMode (если другие режимы использовать не планируется, то этого можно не делать, так как этот режим шифрования уже задан после инициализации функцией Crypto_Init), затем устанавливается мастер-ключ функцией HAL_Crypto_SetKey. Так как ключ должен быть инициализирован в режиме шифрования, то после функции HAL_Crypto_SetKey блок находится в режиме зашифровки данных. После этого используется функция HAL_Crypto_Encode для зашифровки данных. | ||
В конце поочередно выведем каждый из массивов в UART. После этого проведем сравнение полученных данных и ожидаемых. | В конце поочередно выведем каждый из массивов в UART. После этого проведем сравнение полученных данных и ожидаемых. | ||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
Функция kuznechik_ECB_code<syntaxhighlight lang="c" line="1"> | Функция kuznechik_ECB_code<syntaxhighlight lang="c" line="1"> | ||
void kuznechik_ECB_code() | void kuznechik_ECB_code() | ||
Строка 178: | Строка 206: | ||
xprintf("\n"); | xprintf("\n"); | ||
} | } | ||
</syntaxhighlight>Функция kuznechik_ECB_decode аналогична, но в ней используются другие массивы: | </syntaxhighlight> | ||
}} | |||
Функция kuznechik_ECB_decode аналогична, но в ней используются другие массивы: | |||
* plain_text - полученные расшифрованные данные; | *plain_text - полученные расшифрованные данные; | ||
*cipher_text - зашифрованные данные; | *cipher_text - зашифрованные данные; | ||
*expect_plain_text - расшифрованные данные, которые ожидается получить. | *expect_plain_text - расшифрованные данные, которые ожидается получить. | ||
Строка 186: | Строка 216: | ||
Для расшифровки данных используется функция HAL_Crypto_Decode. | Для расшифровки данных используется функция HAL_Crypto_Decode. | ||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
Функция kuznechik_ECB_decode<syntaxhighlight lang="c" line="1"> | Функция kuznechik_ECB_decode<syntaxhighlight lang="c" line="1"> | ||
void kuznechik_ECB_decode() | void kuznechik_ECB_decode() | ||
Строка 267: | Строка 298: | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
Функция main | }} | ||
Функция main. | |||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | <syntaxhighlight lang="c" line="1"> | ||
int main() | int main() | ||
Строка 287: | Строка 322: | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
}} | |||
[[Файл:Вывод в UART. Режим ECB.png|мини|Вывод в UART]] | [[Файл:Вывод в UART. Режим ECB.png|мини|Вывод в UART]] | ||
Если у вас включен UART или подключен common.h, или объявлен #define MIK32_CRYPTO_DEBUG, то можно посмотреть вывод xprintf. Вывод в UART изображен на рисунке. | Если у вас включен UART или подключен common.h, или объявлен #define MIK32_CRYPTO_DEBUG, то можно посмотреть вывод xprintf. Вывод в UART изображен на рисунке. | ||
Строка 306: | Строка 342: | ||
Количество слов в данных для зашифровки должно быть кратно размеру блока (для кузнечика 4 слова или 128 бит). Если данных меньше, то их нужно дополнить вручную. | Количество слов в данных для зашифровки должно быть кратно размеру блока (для кузнечика 4 слова или 128 бит). Если данных меньше, то их нужно дополнить вручную. | ||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
Функция kuznechik_CBC_code<syntaxhighlight lang="c" line="1"> | Функция kuznechik_CBC_code<syntaxhighlight lang="c" line="1"> | ||
void kuznechik_CBC_code() | void kuznechik_CBC_code() | ||
Строка 390: | Строка 427: | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
}} | |||
Функция kuznechik_CBC_decode для расшифровки. | |||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | |||
void kuznechik_CBC_decode() | void kuznechik_CBC_decode() | ||
{ | { | ||
Строка 478: | Строка 518: | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
}} | |||
функция main. | |||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | |||
int main() | int main() | ||
{ | { | ||
Строка 500: | Строка 543: | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
}} | |||
[[Файл:Вывод в UART. CBC.png|мини|Вывод в UART. CBC]] | [[Файл:Вывод в UART. CBC.png|мини|Вывод в UART. CBC]] | ||
Если у вас включен UART или подключен common.h, или объявлен #define MIK32_CRYPTO_DEBUG, то можно посмотреть вывод xprintf. Вывод в UART изображен на рисунке. | Если у вас включен UART или подключен common.h, или объявлен #define MIK32_CRYPTO_DEBUG, то можно посмотреть вывод xprintf. Вывод в UART изображен на рисунке. | ||
Строка 506: | Строка 550: | ||
Количество слов в данных может быть любым. Если данных меньше, то их нужно дополнить вручную. | Количество слов в данных может быть любым. Если данных меньше, то их нужно дополнить вручную. | ||
Функция kuznechik_CTR_code<syntaxhighlight lang="c" line="1"> | Функция kuznechik_CTR_code. | ||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | |||
void kuznechik_CTR_code() | void kuznechik_CTR_code() | ||
{ | { | ||
Строка 591: | Строка 637: | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
Функция kuznechik_CTR_decode <syntaxhighlight lang="c" line="1"> | }} | ||
Функция kuznechik_CTR_decode. | |||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | |||
void kuznechik_CTR_decode() | void kuznechik_CTR_decode() | ||
{ | { | ||
Строка 676: | Строка 727: | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
Функция main | }} | ||
Функция main. | |||
{{#spoiler:show=Развернуть код|hide=Свернуть код| | |||
<syntaxhighlight lang="c" line="1"> | <syntaxhighlight lang="c" line="1"> | ||
int main() | int main() | ||
Строка 697: | Строка 752: | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
}} | |||
[[Файл:Вывод в UART. CTR.png|мини|Вывод в UART. CTR]] | [[Файл:Вывод в UART. CTR.png|мини|Вывод в UART. CTR]] | ||
Если у вас включен UART или подключен common.h, или объявлен #define MIK32_CRYPTO_DEBUG, то можно посмотреть вывод xprintf. Вывод в UART изображен на рисунке. | Если у вас включен UART или подключен common.h, или объявлен #define MIK32_CRYPTO_DEBUG, то можно посмотреть вывод xprintf. Вывод в UART изображен на рисунке. |
Версия от 16:00, 20 марта 2023
В примере будет зашифрованы и расшифрованы данные алгоритмом кузнечик. Данный процесс будет рассмотрен с различными режимами шифрования. Шифрования для алгоритмов «Магма» и AES128 полностью аналогичны, за исключением разрядности обрабатываемых данных.
Работа с конфигуратором (В разработке)
Для начала настроем в конфигураторе тактирование mik32, например, от внешнего кварца 32МГц. Затем настроем делители шины. Так как крипто-блок тактируется от шины AHB_CLK, то зададим делитель AHB_DIV. В данном примере оставим делитель по умолчанию. В итоге вкладка с тактированием должна выглядеть так:
(Картинка тактирования из конфигуратора. В работе)
Затем перейдем к настройке самого крипто-блока. Для этого откроем вкладку крипто-блок и нажмем включить. После этого появятся несколько настроек.
Зададим им следующие значения:
- Алгоритм шифрования - Кузнечик;
- Режим шифрования - ECB;
- В перестановке слова - нет перестановки;
- Порядок загрузки/выгрузки - От старшего слова к младшему.
В итоге настройки крипто-блока в конфигураторе должны выглядеть как на рисунке.
Нажимаем кнопку сохранения и генерации. В итоге у нас появится проект для PlatformIo. Далее работа идет в visual studio code.
Использование библиотеки HAL_Crypto
В сгенерированном проекте в файле main.c должна быть функция Crypto_Init, в которой будут заданы настройки для крипто-блока. Выглядит она так:
Кроме этого в функции SystemClock_Config приведены настройки для тактирования. Убедитесь, что в PeriphClkInit.PMClockAHB присутствует PM_CLOCK_CRYPTO_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 можно видеть объявление структуры с набором настроек для крипто-блока, которую использует функция инициализации Crypto_Init.
Создадим глобальный массив для ключа. Количество элементов массива можно задать с помощью макросов CRYPTO_KEY_KUZNECHIK (количество слов в ключе для алгоритма кузнечик).
uint32_t crypto_key[CRYPTO_KEY_KUZNECHIK] = {0x8899aabb, 0xccddeeff, 0x00112233, 0x44556677, 0xfedcba98, 0x76543210, 0x01234567, 0x89abcdef};
Для режимов шифрования ECB и CBC длина данных должна быть кратна длине блока. Иначе данные следует дополнить.
Для режима CTR длинна данных может быть любой. В этом режиме дополнения не требуются.
Синхропосылка (вектор инициализации), в соответствии с ГОСТ 34.13—2015, для разных режимов может иметь разную длину:
- В режиме ECB синхропосылка не используется;
- В режиме CBC синхропосылка должна быть длинной m*z, где m - длина блока (128 бит для алгоритма Кузнечик), z - целое число. Поддерживается только случай z=1, m=n, где n - длина блока (4 слова для Кузнечика и AES, 2 слова для Магмы);
- В режиме CTR в регистр INIT следует записывать значение счетчика состоящего из синхропосылки, длинна которой равна половине длине блока, и такого же количества нулей. В итоге размер синхропосылки будет равен размеру блока.
Режим простой замены (Electronic Codebook, ЕСВ)
Теперь напишем функции для зашифровки и расшифровки данных в режиме шифрования ECB - kuznechik_ECB_code и kuznechik_ECB_decode соответственно.
В функции kuznechik_ECB_code будет три массива данных:
- plain_text - незашифрованные данные;
- cipher_text - полученные зашифрованные данные;
- expect_cipher_text - зашифрованные данные, которые ожидается получить.
Количество слов в данных для зашифровки должно быть кратно размеру блока (для кузнечика 4 слова или 128 бит). Если данных меньше, то их нужно дополнить вручную. ГОСТ 34.13—2015 определяет три возможные процедуры дополнения. Можно, например, дополнить остаток нулями до размера блока.
Внутри kuznechik_ECB_code сначала задается режим шифрования функцией HAL_Crypto_SetCipherMode (если другие режимы использовать не планируется, то этого можно не делать, так как этот режим шифрования уже задан после инициализации функцией Crypto_Init), затем устанавливается мастер-ключ функцией HAL_Crypto_SetKey. Так как ключ должен быть инициализирован в режиме шифрования, то после функции HAL_Crypto_SetKey блок находится в режиме зашифровки данных. После этого используется функция HAL_Crypto_Encode для зашифровки данных.
В конце поочередно выведем каждый из массивов в UART. После этого проведем сравнение полученных данных и ожидаемых.
Функция kuznechik_ECB_decode аналогична, но в ней используются другие массивы:
- plain_text - полученные расшифрованные данные;
- cipher_text - зашифрованные данные;
- expect_plain_text - расшифрованные данные, которые ожидается получить.
Для расшифровки данных используется функция HAL_Crypto_Decode.
Функция main.
Если у вас включен UART или подключен common.h, или объявлен #define MIK32_CRYPTO_DEBUG, то можно посмотреть вывод xprintf. Вывод в UART изображен на рисунке.
Функции для режимов CBC и CTR аналогичны, но содержат другие данные в массивах.
Отличие этих режимов в том, что они перед каждой новой расшифровкой или зашифровкой требуют записи вектора инициализации с помощью функции HAL_Crypto_SetINIT.
Режим гаммирования (Counter, CTR)
Размер массива для вектора инициализации можно задать с помощью макросов:
- IV_LENGTH_KUZNECHIK_CBC
- IV_LENGTH_MAGMA_CBC
- IV_LENGTH_AES_CBC
- IV_LENGTH_KUZNECHIK_CTR
- IV_LENGTH_MAGMA_CTR
- IV_LENGTH_AES_CTR
Количество слов в данных для зашифровки должно быть кратно размеру блока (для кузнечика 4 слова или 128 бит). Если данных меньше, то их нужно дополнить вручную.
Функция kuznechik_CBC_decode для расшифровки.
функция main.
Если у вас включен UART или подключен common.h, или объявлен #define MIK32_CRYPTO_DEBUG, то можно посмотреть вывод xprintf. Вывод в UART изображен на рисунке.
Режим простой замены с зацеплением (Cipher Block Chaining, СВС)
Количество слов в данных может быть любым. Если данных меньше, то их нужно дополнить вручную.
Функция kuznechik_CTR_code.
Функция kuznechik_CTR_decode.
Функция main.
Если у вас включен UART или подключен common.h, или объявлен #define MIK32_CRYPTO_DEBUG, то можно посмотреть вывод xprintf. Вывод в UART изображен на рисунке.