3 * email: daniel.otte@rub.de
12 #include <avr/pgmspace.h>
21 void gamma(uint32_t* a){
24 a[1] ^= ~((a[3]) | (a[2]));
27 tmp=a[3]; a[3]=a[0]; a[0]=tmp;
28 a[2] ^= a[0] ^ a[1] ^ a[3];
30 a[1] ^= ~((a[3]) | (a[2]));
34 #define ROTL32(a,n) (((a)<<n)|((a)>>(32-n)))
35 #define ROTR32(a,n) (((a)>>n)|((a)<<(32-n)))
38 void pi1(uint32_t* a){
39 a[1] = ROTL32(a[1], 1);
40 a[2] = ROTL32(a[2], 5);
41 a[3] = ROTL32(a[3], 2);
45 void pi2(uint32_t* a){
46 a[1] = ROTR32(a[1], 1);
47 a[2] = ROTR32(a[2], 5);
48 a[3] = ROTR32(a[3], 2);
52 void theta(uint32_t* k, uint32_t* a){
55 temp = a[0] ^ a[2]; temp ^= ROTR32(temp, 8) ^ ROTL32(temp, 8);
64 temp = a[1] ^ a[3]; temp ^= ROTR32(temp, 8) ^ ROTL32(temp, 8);
71 void noekeon_round(uint32_t* key, uint32_t* state, uint8_t const1, uint8_t const2){
72 ((uint8_t*)state)[RC_POS] ^= const1;
74 ((uint8_t*)state)[RC_POS] ^= const2;
80 uint8_t rc_tab[] PROGMEM = {
82 0x1B, 0x36, 0x6C, 0xD8, 0xAB, 0x4D, 0x9A,
83 0x2F, 0x5E, 0xBC, 0x63, 0xC6, 0x97, 0x35, 0x6A,
87 0xD4, 0xB3, 0x7D, 0xFA, 0xEF, 0xC5, 0x91, 0x39,
88 0x72, 0xE4, 0xD3, 0xBD, 0x61, 0xC2, 0x9F, 0x25,
92 void changendian32(void* a){
93 ((uint8_t*)a)[0] ^= ((uint8_t*)a)[3];
94 ((uint8_t*)a)[3] ^= ((uint8_t*)a)[0];
95 ((uint8_t*)a)[0] ^= ((uint8_t*)a)[3];
97 ((uint8_t*)a)[1] ^= ((uint8_t*)a)[2];
98 ((uint8_t*)a)[2] ^= ((uint8_t*)a)[1];
99 ((uint8_t*)a)[1] ^= ((uint8_t*)a)[2];
103 void changendian(void* a){
104 changendian32((uint32_t*)(&(((uint32_t*)a)[0])));
105 changendian32((uint32_t*)(&(((uint32_t*)a)[1])));
106 changendian32((uint32_t*)(&(((uint32_t*)a)[2])));
107 changendian32((uint32_t*)(&(((uint32_t*)a)[3])));
110 /******************************************************************************/
112 void noekeon_enc(void* buffer, void* key){
119 for(i=0; i<ROUND_NR; ++i){
120 noekeon_round((uint32_t*)key, (uint32_t*)buffer, rc, 0);
121 rc = pgm_read_byte(rc_tab+i);
123 ((uint8_t*)buffer)[RC_POS] ^= rc;
124 theta((uint32_t*)key, (uint32_t*)buffer);
131 void noekeon_dec(void* buffer, void* key){
141 memset(nullv, 0, 16);
142 memcpy(dkey, key, 16);
144 theta((uint32_t*)nullv, (uint32_t*)dkey);
145 uart_putstr_P(PSTR("\r\nTheta: "));
146 uart_hexdump(dkey, 16);
148 for(i=ROUND_NR-1; i>=0; --i){
149 rc = pgm_read_byte(rc_tab+i);
150 noekeon_round((uint32_t*)dkey, (uint32_t*)buffer, 0, rc);
152 theta((uint32_t*)dkey, (uint32_t*)buffer);
153 ((uint8_t*)buffer)[RC_POS] ^= 0x80;
159 void noekeon_init(void* key, noekeon_ctx_t* ctx){
162 memset(nullv, 0, 16);
163 memcpy(ctx, key, 16);
164 noekeon_enc(ctx, nullv);