/* * Copyright (C) 2013 Freie Universität Berlin, Computer Systems & Telematics * * 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. */ /** * @ingroup sys_crypto * @{ * * @file skipjack.h * @brief Headers for the implementation of the SkipJack cipher-algorithm * * @author Freie Universitaet Berlin, Computer Systems & Telematics * @author Nicolai Schmittberger * @author Zakaria Kasmi */ #ifndef SKIPJACK_H_ #define SKIPJACK_H_ #include "crypto/ciphers.h" #ifdef __cplusplus extern "C" { #endif #define F(addr) /*CRYPTO_TABLE_ACCESS( &SJ_F[addr])*/ (SJ_F[addr]) // G-Permutation: 4 round feistel structure #define G(key, b, bLeft, bRight) \ ( \ bLeft = b, \ bRight = (b >> 8), \ bLeft ^= F(bRight ^ key[0]), \ bRight ^= F(bLeft ^ key[1]), \ bLeft ^= F(bRight ^ key[2]), \ bRight ^= F(bLeft ^ key[3]), \ ((bRight << 8) | bLeft)) #define G_INV(key, b, bLeft, bRight) \ ( bLeft = b, \ bRight = (b >> 8), \ bRight ^= F(bLeft ^ key[3]), \ bLeft ^= F(bRight ^ key[2]), \ bRight ^= F(bLeft ^ key[1]), \ bLeft ^= F(bRight ^ key[0]), \ ((bRight << 8) | bLeft)) // A-RULE: #define RULE_A(skey, w1, w2, w3, w4, counter, tmp, bLeft, bRight ) { \ tmp = w4; \ w4 = w3; \ w3 = w2; \ w2 = G(skey, w1, bLeft, bRight); \ w1 = ((tmp ^ w2) ^ counter); \ counter++; \ skey += 4; } #define RULE_A_INV(skey, w1, w2, w3, w4, counter, tmp, bLeft, bRight) { \ tmp = w4; \ w4 = (w1 ^ w2 ^ counter); \ w1 = G_INV(skey, w2, bLeft, bRight); \ w2 = w3; \ w3 = tmp; \ counter--; \ skey -= 4; } \ // B-RULE: #define RULE_B(skey, w1, w2, w3, w4, counter, tmp, bLeft, bRight ) { \ tmp = w1; \ w1 = w4; \ w4 = w3; \ w3 = (tmp ^ w2 ^ counter); \ w2 = G(skey, tmp, bLeft, bRight); \ counter++; \ skey += 4; } #define RULE_B_INV(skey, w1, w2, w3, w4, counter, tmp, bLeft, bRight ) { \ tmp = w1; \ w1 = G_INV(skey, w2, bLeft, bRight); \ w2 = (w1 ^ w3 ^ counter); \ w3 = w4; \ w4 = tmp; \ counter--; \ skey -= 4; } /** * @brief The cipher_context_t adapted for SkipJack * @typedef skipjack_context_t */ typedef struct { // 2 times keysize. makes unrolling keystream easier / efficient uint8_t skey [ 20 ]; } skipjack_context_t; /** * @brief Initialize the SkipJack-BlockCipher context. * * @param context structure to hold the opaque data from this * initialization call. It should be passed to future * invocations of this module which use this particular * key. * @param blockSize size of the block in bytes. * @param keySize key size in bytes * @param key pointer to the key * * @return Whether initialization was successful. The command may be * unsuccessful if the key size or blockSize are not valid. */ int skipjack_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. * * @param context holds the module specific opaque data related to the * key (perhaps key expansions). * @param plainBlock a plaintext block of blockSize * @param cipherBlock the resulting ciphertext block of blockSize * * @return Whether the encryption was successful. Possible failure reasons * include not calling init(). */ int skipjack_encrypt(cipher_context_t *context, uint8_t *plainBlock, uint8_t *cipherBlock); /** * @brief Decrypts a single block (of blockSize) using the passed context. * * @param context holds the module specific opaque data related to the * key (perhaps key expansions). * @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() */ int skipjack_decrypt(cipher_context_t *context, uint8_t *cipherBlock, uint8_t *plainBlock); /** * @brief Sets up the context to use the passed key for usage with SkipJack * 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 skipjack_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. In the case where the * cipher operates with multiple block sizes, this will pick one * particular size (deterministically). */ uint8_t skipjack_get_preferred_block_size(void); /** * Interface to access the functions * */ extern block_cipher_interface_t skipjack_interface; #ifdef __cplusplus } #endif /** @} */ #endif /* SKIPJACK_H_ */