parent
eb6bd444c4
commit
0e2011e085
13 changed files with 1028 additions and 79 deletions
@ -0,0 +1,348 @@ |
||||
#! /usr/bin/pythonw |
||||
# The Keccak sponge function, designed by Guido Bertoni, Joan Daemen, |
||||
# questions, please refer to our website: http://keccak.noekeon.org/ |
||||
# |
||||
# Implementation by Renaud Bauvin, |
||||
# hereby denoted as "the implementer". |
||||
# |
||||
# To the extent possible under law, the implementer has waived all copyright |
||||
# and related or neighboring rights to the source code in this file. |
||||
# http://creativecommons.org/publicdomain/zero/1.0/ |
||||
|
||||
import math |
||||
|
||||
class KeccakError(Exception): |
||||
"""Class of error used in the Keccak implementation |
||||
|
||||
Use: raise KeccakError.KeccakError("Text to be displayed")""" |
||||
|
||||
def __init__(self, value): |
||||
self.value = value |
||||
def __str__(self): |
||||
return repr(self.value) |
||||
|
||||
|
||||
class Keccak: |
||||
""" |
||||
Class implementing the Keccak sponge function |
||||
""" |
||||
def __init__(self, b=1600): |
||||
"""Constructor: |
||||
|
||||
b: parameter b, must be 25, 50, 100, 200, 400, 800 or 1600 (default value)""" |
||||
self.setB(b) |
||||
|
||||
def setB(self,b): |
||||
"""Set the value of the parameter b (and thus w,l and nr) |
||||
|
||||
b: parameter b, must be choosen among [25, 50, 100, 200, 400, 800, 1600] |
||||
""" |
||||
|
||||
if b not in [25, 50, 100, 200, 400, 800, 1600]: |
||||
raise KeccakError.KeccakError('b value not supported - use 25, 50, 100, 200, 400, 800 or 1600') |
||||
|
||||
# Update all the parameters based on the used value of b |
||||
self.b=b |
||||
self.w=b//25 |
||||
self.l=int(math.log(self.w,2)) |
||||
self.nr=12+2*self.l |
||||
|
||||
# Constants |
||||
|
||||
## Round constants |
||||
RC=[0x0000000000000001, |
||||
0x0000000000008082, |
||||
0x800000000000808A, |
||||
0x8000000080008000, |
||||
0x000000000000808B, |
||||
0x0000000080000001, |
||||
0x8000000080008081, |
||||
0x8000000000008009, |
||||
0x000000000000008A, |
||||
0x0000000000000088, |
||||
0x0000000080008009, |
||||
0x000000008000000A, |
||||
0x000000008000808B, |
||||
0x800000000000008B, |
||||
0x8000000000008089, |
||||
0x8000000000008003, |
||||
0x8000000000008002, |
||||
0x8000000000000080, |
||||
0x000000000000800A, |
||||
0x800000008000000A, |
||||
0x8000000080008081, |
||||
0x8000000000008080, |
||||
0x0000000080000001, |
||||
0x8000000080008008] |
||||
|
||||
## Rotation offsets |
||||
r=[[0, 36, 3, 41, 18] , |
||||
[1, 44, 10, 45, 2] , |
||||
[62, 6, 43, 15, 61] , |
||||
[28, 55, 25, 21, 56] , |
||||
[27, 20, 39, 8, 14] ] |
||||
|
||||
## Generic utility functions |
||||
|
||||
def rot(self,x,n): |
||||
"""Bitwise rotation (to the left) of n bits considering the \ |
||||
string of bits is w bits long""" |
||||
|
||||
n = n%self.w |
||||
return ((x>>(self.w-n))+(x<<n))%(1<<self.w) |
||||
|
||||
def fromHexStringToLane(self, string): |
||||
"""Convert a string of bytes written in hexadecimal to a lane value""" |
||||
|
||||
#Check that the string has an even number of characters i.e. whole number of bytes |
||||
if len(string)%2!=0: |
||||
raise KeccakError.KeccakError("The provided string does not end with a full byte") |
||||
|
||||
#Perform the modification |
||||
temp='' |
||||
nrBytes=len(string)//2 |
||||
for i in range(nrBytes): |
||||
offset=(nrBytes-i-1)*2 |
||||
temp+=string[offset:offset+2] |
||||
return int(temp, 16) |
||||
|
||||
def fromLaneToHexString(self, lane): |
||||
"""Convert a lane value to a string of bytes written in hexadecimal""" |
||||
|
||||
laneHexBE = (("%%0%dX" % (self.w//4)) % lane) |
||||
#Perform the modification |
||||
temp='' |
||||
nrBytes=len(laneHexBE)//2 |
||||
for i in range(nrBytes): |
||||
offset=(nrBytes-i-1)*2 |
||||
temp+=laneHexBE[offset:offset+2] |
||||
return temp.upper() |
||||
|
||||
def printState(self, state, info): |
||||
"""Print on screen the state of the sponge function preceded by \ |
||||
string info |
||||
|
||||
state: state of the sponge function |
||||
info: a string of characters used as identifier""" |
||||
|
||||
print("Current value of state: %s" % (info)) |
||||
for y in range(5): |
||||
line=[] |
||||
for x in range(5): |
||||
line.append(hex(state[x][y])) |
||||
print('\t%s' % line) |
||||
|
||||
### Conversion functions String <-> Table (and vice-versa) |
||||
|
||||
def convertStrToTable(self,string): |
||||
|
||||
|
||||
#Check that input paramaters |
||||
if self.w%8!= 0: |
||||
raise KeccakError("w is not a multiple of 8") |
||||
if len(string)!=2*(self.b)//8: |
||||
raise KeccakError.KeccakError("string can't be divided in 25 blocks of w bits\ |
||||
i.e. string must have exactly b bits") |
||||
|
||||
#Convert |
||||
output=[[0,0,0,0,0], |
||||
[0,0,0,0,0], |
||||
[0,0,0,0,0], |
||||
[0,0,0,0,0], |
||||
[0,0,0,0,0]] |
||||
for x in range(5): |
||||
for y in range(5): |
||||
offset=2*((5*y+x)*self.w)//8 |
||||
output[x][y]=self.fromHexStringToLane(string[offset:offset+(2*self.w//8)]) |
||||
return output |
||||
|
||||
def convertTableToStr(self,table): |
||||
|
||||
#Check input format |
||||
if self.w%8!= 0: |
||||
raise KeccakError.KeccakError("w is not a multiple of 8") |
||||
if (len(table)!=5) or (False in [len(row)==5 for row in table]): |
||||
raise KeccakError.KeccakError("table must b") |
||||
|
||||
#Convert |
||||
output=['']*25 |
||||
for x in range(5): |
||||
for y in range(5): |
||||
output[5*y+x]=self.fromLaneToHexString(table[x][y]) |
||||
output =''.join(output).upper() |
||||
return output |
||||
|
||||
def Round(self,A,RCfixed): |
||||
"""Perform one round of computation as defined in the Keccak-f permutation |
||||
|
||||
""" |
||||
|
||||
#Initialisation of temporary variables |
||||
B=[[0,0,0,0,0], |
||||
[0,0,0,0,0], |
||||
[0,0,0,0,0], |
||||
[0,0,0,0,0], |
||||
[0,0,0,0,0]] |
||||
C= [0,0,0,0,0] |
||||
D= [0,0,0,0,0] |
||||
|
||||
#Theta step |
||||
for x in range(5): |
||||
C[x] = A[x][0]^A[x][1]^A[x][2]^A[x][3]^A[x][4] |
||||
|
||||
for x in range(5): |
||||
D[x] = C[(x-1)%5]^self.rot(C[(x+1)%5],1) |
||||
|
||||
for x in range(5): |
||||
for y in range(5): |
||||
A[x][y] = A[x][y]^D[x] |
||||
|
||||
#Rho and Pi steps |
||||
for x in range(5): |
||||
for y in range(5): |
||||
B[y][(2*x+3*y)%5] = self.rot(A[x][y], self.r[x][y]) |
||||
|
||||
#Chi step |
||||
for x in range(5): |
||||
for y in range(5): |
||||
A[x][y] = B[x][y]^((~B[(x+1)%5][y]) & B[(x+2)%5][y]) |
||||
|
||||
#Iota step |
||||
A[0][0] = A[0][0]^RCfixed |
||||
|
||||
return A |
||||
|
||||
def KeccakF(self,A, verbose=False): |
||||
"""Perform Keccak-f function on the state A |
||||
|
||||
verbose: a boolean flag activating the printing of intermediate computations |
||||
""" |
||||
|
||||
if verbose: |
||||
self.printState(A,"Before first round") |
||||
|
||||
for i in range(self.nr): |
||||
#NB: result is truncated to lane size |
||||
A = self.Round(A,self.RC[i]%(1<<self.w)) |
||||
|
||||
if verbose: |
||||
self.printState(A,"Satus end of round #%d/%d" % (i+1,self.nr)) |
||||
|
||||
return A |
||||
|
||||
### Padding rule |
||||
|
||||
def pad10star1(self, M, n): |
||||
"""Pad M with the pad10*1 padding rule to reach a length multiple of r bits |
||||
|
||||
M: message pair (length in bits, string of hex characters ('9AFC...') |
||||
n: length in bits (must be a multiple of 8) |
||||
Example: pad10star1([60, 'BA594E0FB9EBBD30'],8) returns 'BA594E0FB9EBBD93' |
||||
""" |
||||
|
||||
[my_string_length, my_string]=M |
||||
|
||||
# Check the parameter n |
||||
if n%8!=0: |
||||
raise KeccakError.KeccakError("n must be a multiple of 8") |
||||
|
||||
# Check the length of the provided string |
||||
if len(my_string)%2!=0: |
||||
#Pad with one '0' to reach correct length (don't know test |
||||
#vectors coding) |
||||
my_string=my_string+'0' |
||||
if my_string_length>(len(my_string)//2*8): |
||||
raise KeccakError.KeccakError("the string is too short to contain the number of bits announced") |
||||
|
||||
nr_bytes_filled=my_string_length//8 |
||||
nbr_bits_filled=my_string_length%8 |
||||
l = my_string_length % n |
||||
if ((n-8) <= l <= (n-2)): |
||||
if (nbr_bits_filled == 0): |
||||
my_byte = 0 |
||||
else: |
||||
my_byte=int(my_string[nr_bytes_filled*2:nr_bytes_filled*2+2],16) |
||||
my_byte=(my_byte>>(8-nbr_bits_filled)) |
||||
my_byte=my_byte+2**(nbr_bits_filled)+2**7 |
||||
my_byte="%02X" % my_byte |
||||
my_string=my_string[0:nr_bytes_filled*2]+my_byte |
||||
else: |
||||
if (nbr_bits_filled == 0): |
||||
my_byte = 0 |
||||
else: |
||||
my_byte=int(my_string[nr_bytes_filled*2:nr_bytes_filled*2+2],16) |
||||
my_byte=(my_byte>>(8-nbr_bits_filled)) |
||||
my_byte=my_byte+2**(nbr_bits_filled) |
||||
my_byte="%02X" % my_byte |
||||
my_string=my_string[0:nr_bytes_filled*2]+my_byte |
||||
while((8*len(my_string)//2)%n < (n-8)): |
||||
my_string=my_string+'00' |
||||
my_string = my_string+'80' |
||||
|
||||
return my_string |
||||
|
||||
def Keccak(self,M,r=1024,c=512,n=1024,verbose=False): |
||||
"""Compute the Keccak[r,c,d] sponge function on message M |
||||
|
||||
M: message pair (length in bits, string of hex characters ('9AFC...') |
||||
r: bitrate in bits (defautl: 1024) |
||||
c: capacity in bits (default: 576) |
||||
n: length of output in bits (default: 1024), |
||||
verbose: print the details of computations(default:False) |
||||
""" |
||||
|
||||
#Check the inputs |
||||
if (r<0) or (r%8!=0): |
||||
raise KeccakError.KeccakError('r must be a multiple of 8 in this implementation') |
||||
if (n%8!=0): |
||||
raise KeccakError.KeccakError('outputLength must be a multiple of 8') |
||||
self.setB(r+c) |
||||
|
||||
if verbose: |
||||
print("Create a Keccak function with (r=%d, c=%d (i.e. w=%d))" % (r,c,(r+c)//25)) |
||||
|
||||
#Compute lane length (in bits) |
||||
w=(r+c)//25 |
||||
|
||||
# Initialisation of state |
||||
S=[[0,0,0,0,0], |
||||
[0,0,0,0,0], |
||||
[0,0,0,0,0], |
||||
[0,0,0,0,0], |
||||
[0,0,0,0,0]] |
||||
|
||||
#Padding of messages |
||||
P = self.pad10star1(M, r) |
||||
|
||||
if verbose: |
||||
print("String ready to be absorbed: %s (will be completed by %d x '00')" % (P, c//8)) |
||||
|
||||
#Absorbing phase |
||||
for i in range((len(P)*8//2)//r): |
||||
Pi=self.convertStrToTable(P[i*(2*r//8):(i+1)*(2*r//8)]+'00'*(c//8)) |
||||
|
||||
for y in range(5): |
||||
for x in range(5): |
||||
S[x][y] = S[x][y]^Pi[x][y] |
||||
S = self.KeccakF(S, verbose) |
||||
|
||||
if verbose: |
||||
print("Value after absorption : %s" % (self.convertTableToStr(S))) |
||||
|
||||
#Squeezing phase |
||||
Z = '' |
||||
outputLength = n |
||||
while outputLength>0: |
||||
string=self.convertTableToStr(S) |
||||
Z = Z + string[:r*2//8] |
||||
outputLength -= r |
||||
if outputLength>0: |
||||
S = self.KeccakF(S, verbose) |
||||
|
||||
# NB: done by block of length r, could have to be cut if outputLength |
||||
# is not a multiple of r |
||||
|
||||
if verbose: |
||||
print("Value after squeezing : %s" % (self.convertTableToStr(S))) |
||||
|
||||
return Z[:2*n//8] |
Loading…
Reference in new issue