1
0
Fork 0
mirror of synced 2025-09-24 04:40:05 +00:00
ZoKrates/zokrates_stdlib/stdlib/hashes/poseidon/poseidon.zok
2021-04-13 17:03:42 +02:00

59 lines
No EOL
1.6 KiB
Text

// https://eprint.iacr.org/2019/458.pdf
from "./constants.zok" import poseidon_c
from "./constants.zok" import poseidon_m
def ark<N>(field[N] state, field[497] c, u32 it) -> field[N]:
for u32 i in 0..N do
state[i] = state[i] + c[it + i]
endfor
return state
def sbox<N>(field[N] state, u32 f, u32 p, u32 r) -> field[N]:
state[0] = state[0]**5
for u32 i in 1..N do
state[i] = if ((r < f/2) || (r >= f/2 + p)) then state[i]**5 else state[i] fi
endfor
return state
def mix<N>(field[N] state, field[7][7] m) -> field[N]:
field[N] out = [0; N]
for u32 i in 0..N do
field acc = 0
for u32 j in 0..N do
acc = acc + (state[j] * m[i][j])
endfor
out[i] = acc
endfor
return out
def main<N>(field[N] inputs) -> field:
assert(N > 0 && N <= 6) // max 6 inputs
u32 t = N + 1
u32[8] rounds_p = [56, 57, 56, 60, 60, 63, 64, 63]
u32 f = 8
u32 p = rounds_p[(t - 2)]
// Constants are padded with zeroes to the maximum value calculated by
// t * (f + p) = 497, where `t` (number of inputs + 1) is a max of 7.
// This is done to keep the function generic, as resulting array size depends on `t`
// and we do not want callers passing down constants.
// This should be revisited once compiler limitations are gone.
field[497] c = poseidon_c()[t - 2]
field[7][7] m = poseidon_m()[t - 2]
field[t] state = [0; t]
for u32 i in 1..t do
state[i] = inputs[i - 1]
endfor
for u32 r in 0..f+p do
state = ark(state, c, r * t)
state = sbox(state, f, p, r)
state = mix(state, m)
endfor
return state[0]