2013-11-28 18:12:40 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2013 Freie Universität Berlin, Computer Systems & Telematics
|
|
|
|
*
|
2014-08-23 15:43:13 +02:00
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
2013-11-28 18:12:40 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup sys_crypto
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file rc5.h
|
|
|
|
* @brief Headers for the implementation of the RC5 Cipher-Algorithm
|
|
|
|
*
|
|
|
|
* @author Freie Universitaet Berlin, Computer Systems & Telematics
|
|
|
|
* @author Nicolai Schmittberger <nicolai.schmittberger@fu-berlin.de>
|
|
|
|
* @author Zakaria Kasmi <zkasmi@inf.fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "crypto/ciphers.h"
|
|
|
|
|
|
|
|
#ifndef RC5_H_
|
|
|
|
#define RC5_H_
|
|
|
|
|
2014-10-10 11:51:11 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
2013-11-28 18:12:40 +01:00
|
|
|
|
|
|
|
#define RC5_32_P 0xB7E15163L
|
|
|
|
#define RC5_32_Q 0x9E3779B9L
|
|
|
|
#define RC5_32_MASK 0xffffffffL
|
|
|
|
|
|
|
|
#define RC5_ROUNDS 12
|
|
|
|
|
|
|
|
#define rotl32(a,b) fastrol32((a), (b))
|
|
|
|
#define rotr32(a,b) fastror32((a), (b))
|
|
|
|
#define rol32(a, n) ( a = (((a) << (n)) | ((a) >> (32-(n)))))
|
|
|
|
#define ror32(a, n) ( a = (((a) >> (n)) | ((a) << (32-(n)))))
|
|
|
|
#define fastrol32 rol32
|
|
|
|
#define fastror32 ror32
|
|
|
|
|
|
|
|
// convert a 4byte char array to a unsigned long
|
|
|
|
// [assumes least significant byte of char array is first]
|
|
|
|
#define c2l(c,l) (l =((unsigned long)(*((c)))), \
|
|
|
|
l|=((unsigned long)(*((c+1))))<< 8L, \
|
|
|
|
l|=((unsigned long)(*((c+2))))<<16L, \
|
|
|
|
l|=((unsigned long)(*((c+3))))<<24L)
|
|
|
|
|
|
|
|
// convert an unsigned long to a 4 byte char array
|
|
|
|
// [assumes least significant byte of char array is first]
|
|
|
|
#define l2c(l,c) (*((c)) =(unsigned char)(((l) )&0xff), \
|
|
|
|
*((c+1))=(unsigned char)(((l)>> 8L)&0xff), \
|
|
|
|
*((c+2))=(unsigned char)(((l)>>16L)&0xff), \
|
|
|
|
*((c+3))=(unsigned char)(((l)>>24L)&0xff))
|
|
|
|
|
|
|
|
// 2 * (ROUNDS +1) * 4
|
|
|
|
// 2 * 13 * 4 = 104 bytes
|
|
|
|
/**
|
|
|
|
* @brief the cipher_context_t adapted for RC5
|
|
|
|
*/
|
|
|
|
typedef struct {
|
2014-12-06 11:55:28 +01:00
|
|
|
/** @cond INTERNAL */
|
2013-11-28 18:12:40 +01:00
|
|
|
uint32_t skey [2 * (RC5_ROUNDS + 1)];
|
2014-12-06 11:55:28 +01:00
|
|
|
/** @endcond */
|
2013-11-28 18:12:40 +01:00
|
|
|
} rc5_context_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize the BlockCipher.
|
|
|
|
*
|
|
|
|
* @param context the cipher_context_t-struct to save the initialization
|
|
|
|
* of the cipher in
|
|
|
|
* @param blockSize the used blocksize - this must match the
|
|
|
|
* cipher-blocksize
|
|
|
|
* @param keySize the size of the key
|
|
|
|
* @param key a pointer to the key
|
|
|
|
*
|
|
|
|
* @return Whether initialization was successful. The command may be
|
|
|
|
* unsuccessful if the key size or blockSize are not valid for the
|
|
|
|
* given cipher implementation.
|
|
|
|
*/
|
|
|
|
int rc5_init(cipher_context_t *context, uint8_t blockSize, uint8_t keySize,
|
|
|
|
uint8_t *key);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Encrypts a single block (of blockSize) using the passed context.
|
|
|
|
*
|
|
|
|
* PROLOGUE: 24 cycles
|
|
|
|
* INIT: 48 cycles
|
|
|
|
* LOOP: 1680 cycles (12 + fastrol [= 42] + 16) * 2 * RC5_ROUNDS
|
|
|
|
* CLOSE: 24 cycles
|
|
|
|
* =====================
|
|
|
|
* 1776 cycles (avg case)
|
|
|
|
*
|
|
|
|
* @param context the cipher_context_t-struct to save the updated key in
|
2014-11-30 22:34:50 +01:00
|
|
|
* @param plain_block a plaintext block of blockSize
|
|
|
|
* @param cipher_block the resulting ciphertext block of blockSize
|
2013-11-28 18:12:40 +01:00
|
|
|
*
|
|
|
|
* @return Whether the encryption was successful. Possible failure reasons
|
|
|
|
* include not calling init().
|
|
|
|
*/
|
2014-11-30 22:34:50 +01:00
|
|
|
int rc5_encrypt(cipher_context_t *context, uint8_t *plain_block, uint8_t *cipher_block);
|
2013-11-28 18:12:40 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Decrypts a single block (of blockSize) using the key and the
|
|
|
|
* keySize.
|
|
|
|
*
|
|
|
|
* @param context the cipher_context_t-struct to use for this decryption
|
|
|
|
* @param cipherBlock a ciphertext block of blockSize
|
|
|
|
* @param plainBlock the resulting plaintext block of blockSize
|
|
|
|
*
|
|
|
|
* @return Whether the decryption was successful. Possible failure reasons
|
|
|
|
* include not calling init() or an unimplimented decrypt function.
|
|
|
|
*/
|
|
|
|
int rc5_decrypt(cipher_context_t *context, uint8_t *cipherBlock,
|
|
|
|
uint8_t *plainBlock);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Sets up the key for usage with RC5
|
|
|
|
* Performs the key expansion on the real secret.
|
|
|
|
*
|
|
|
|
* @param context the cipher_context_t-struct to save the updated key in
|
|
|
|
* @param key a pointer to the secret key
|
|
|
|
* @param keysize the length of the secret key
|
|
|
|
*
|
|
|
|
* @return SUCCESS
|
|
|
|
*/
|
|
|
|
int rc5_setup_key(cipher_context_t *context, uint8_t *key, uint8_t keysize);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Returns the preferred block size that this cipher operates with.
|
|
|
|
* It is always safe to call this function before the init() call has
|
|
|
|
* been made.
|
|
|
|
*
|
|
|
|
* @return the preferred block size for this cipher.
|
|
|
|
*/
|
|
|
|
uint8_t rc5_get_preferred_block_size(void);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Interface to access the functions
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
extern block_cipher_interface_t rc5_interface;
|
|
|
|
|
2014-10-10 11:51:11 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-11-28 18:12:40 +01:00
|
|
|
/** @} */
|
|
|
|
#endif /* RC5_H_ */
|