123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202 |
- /**
- * @file rc4.c
- * @brief RC4 encryption algorithm
- *
- * @section License
- *
- * SPDX-License-Identifier: GPL-2.0-or-later
- *
- * Copyright (C) 2010-2023 Oryx Embedded SARL. All rights reserved.
- *
- * This file is part of CycloneCRYPTO Open.
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version 2
- * of the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software Foundation,
- * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- *
- * @author Oryx Embedded SARL (www.oryx-embedded.com)
- * @version 2.2.4
- **/
- //Switch to the appropriate trace level
- #define TRACE_LEVEL CRYPTO_TRACE_LEVEL
- //Dependencies
- #include "core/crypto.h"
- #include "cipher/rc4.h"
- //Check crypto library configuration
- #if (RC4_SUPPORT == ENABLED)
- //Common interface for encryption algorithms
- const CipherAlgo rc4CipherAlgo =
- {
- "RC4",
- sizeof(Rc4Context),
- CIPHER_ALGO_TYPE_STREAM,
- 0,
- (CipherAlgoInit) rc4Init,
- (CipherAlgoEncryptStream) rc4Cipher,
- (CipherAlgoDecryptStream) rc4Cipher,
- NULL,
- NULL,
- (CipherAlgoDeinit) rc4Deinit
- };
- /**
- * @brief Initialize an RC4 context using the supplied key
- * @param[in] context Pointer to the RC4 context to initialize
- * @param[in] key Pointer to the key
- * @param[in] length Length of the key
- * @return Error code
- **/
- error_t rc4Init(Rc4Context *context, const uint8_t *key, size_t length)
- {
- uint_t i;
- uint_t j;
- uint8_t temp;
- //Check parameters
- if(context == NULL || key == NULL)
- return WINDOWS_ERROR_INVALID_PARAMETER;
- //Clear context
- context->i = 0;
- context->j = 0;
- //Initialize the S array with identity permutation
- for(i = 0; i < 256; i++)
- {
- context->s[i] = i;
- }
- //S is then processed for 256 iterations
- for(i = 0, j = 0; i < 256; i++)
- {
- //Randomize the permutations using the supplied key
- j = (j + context->s[i] + key[i % length]) % 256;
- //Swap the values of S[i] and S[j]
- temp = context->s[i];
- context->s[i] = context->s[j];
- context->s[j] = temp;
- }
- //RC4 context successfully initialized
- return SUCCESS;
- }
- /**
- * @brief Encrypt/decrypt data with the RC4 algorithm
- * @param[in] context Pointer to the RC4 context
- * @param[in] input Pointer to the data to encrypt/decrypt
- * @param[in] output Pointer to the resulting data
- * @param[in] length Length of the input data
- **/
- void rc4Cipher(Rc4Context *context, const uint8_t *input, uint8_t *output,
- size_t length)
- {
- uint8_t temp;
- //Restore context
- uint_t i = context->i;
- uint_t j = context->j;
- uint8_t *s = context->s;
- //Encryption loop
- while(length > 0)
- {
- //Adjust indices
- i = (i + 1) % 256;
- j = (j + s[i]) % 256;
- //Swap the values of S[i] and S[j]
- temp = s[i];
- s[i] = s[j];
- s[j] = temp;
- //Valid input and output?
- if(input != NULL && output != NULL)
- {
- //XOR the input data with the RC4 stream
- *output = *input ^ s[(s[i] + s[j]) % 256];
- //Increment data pointers
- input++;
- output++;
- }
- //Remaining bytes to process
- length--;
- }
- //Save context
- context->i = i;
- context->j = j;
- }
- /**
- * @brief Release RC4 context
- * @param[in] context Pointer to the RC4 context
- **/
- void rc4Deinit(Rc4Context *context)
- {
- //Clear RC4 context
- osMemset(context, 0, sizeof(Rc4Context));
- }
- #endif
- void rc4_encryption(uint8_t *data, size_t data_length, const uint8_t *key, size_t key_length) {
- uint8_t S[256];
- uint8_t T[256];
- uint8_t temp;
- size_t i, j, k;
- // 初始化S盒和T盒
- for (i = 0; i < 256; i++) {
- S[i] = i;
- T[i] = key[i % key_length];
- }
- // 初始置换
- j = 0;
- for (i = 0; i < 256; i++) {
- j = (j + S[i] + T[i]) % 256;
- temp = S[i];
- S[i] = S[j];
- S[j] = temp;
- }
- // 生成伪随机数流并加密/解密数据
- i = 0;
- j = 0;
- for (k = 0; k < data_length; k++) {
- i = (i + 1) % 256;
- j = (j + S[i]) % 256;
- temp = S[i];
- S[i] = S[j];
- S[j] = temp;
- uint8_t key_byte = S[(S[i] + S[j]) % 256];
- data[k] ^= key_byte;
- }
- }
- void rc4_decrypt(uint8_t *data, size_t data_length, const uint8_t *key, size_t key_length) {
- rc4_encryption(data, data_length, key, key_length);
- }
|