mirror of
https://github.com/codex-storage/constantine.git
synced 2025-01-19 15:23:06 +00:00
e5612f5705
* unoptimized msm * MSM: reorder loops * add a signed windowed recoding technique * improve wNAF table access * use batchAffine * revamp EC tests * MSM signed digit support * refactor MSM: recode signed ahead of time * missing test vector * refactor allocs and Alloca sideeffect * add an endomorphism threshold * Add Jacobian extended coordinates * refactor recodings, prepare for parallelizable on-the-fly signed recoding * recoding changes, introduce proper NAF for pairings * more pairings refactoring, introduce miller accumulator for EVM * some optim to the addchain miller loop * start optimizing multi-pairing * finish multi-miller loop refactoring * minor tuning * MSM: signed encoding suitable for parallelism (no precompute) * cleanup signed window encoding * add prefetching * add metering * properly init result to infinity * comment on prefetching * introduce vartime inversion for batch additions * fix JacExt infinity conversion * add batchAffine for MSM, though slower than JacExtended at the moment * add a batch affine scheduler for MSM * Add Multi-Scalar-Multiplication endomorphism acceleration * some tuning * signed integer fixes + 32-bit + tuning * Some more tuning * common msm bench + don't use affine for c < 9 * nit
217 lines
6.0 KiB
Nim
217 lines
6.0 KiB
Nim
# Constantine
|
|
# Copyright (c) 2018-2019 Status Research & Development GmbH
|
|
# Copyright (c) 2020-Present Mamy André-Ratsimbazafy
|
|
# Licensed and distributed under either of
|
|
# * MIT license (license terms in the root directory or at http://opensource.org/licenses/MIT).
|
|
# * Apache v2 license (license terms in the root directory or at http://www.apache.org/licenses/LICENSE-2.0).
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
import
|
|
# Standard library
|
|
std/[unittest, times, os, strutils],
|
|
# 3rd party
|
|
pkg/jsony,
|
|
# Internals
|
|
../constantine/platforms/abstractions,
|
|
../constantine/math/config/curves,
|
|
../constantine/math/extension_fields,
|
|
../constantine/math/io/[io_bigints, io_ec],
|
|
../constantine/math/ec_shortweierstrass,
|
|
../constantine/hash_to_curve/hash_to_curve,
|
|
../constantine/hashes
|
|
|
|
# Serialization
|
|
# --------------------------------------------------------------------------
|
|
|
|
type
|
|
FieldDesc = object
|
|
m: string
|
|
p: string
|
|
|
|
MapDesc = object
|
|
name: string
|
|
|
|
HashToCurveTest[EC: ECP_ShortW_Aff] = object
|
|
L: string
|
|
Z: string
|
|
ciphersuite: string
|
|
curve: string
|
|
dst: string
|
|
expand: string
|
|
field: FieldDesc
|
|
hash: string
|
|
k: string
|
|
map: MapDesc
|
|
randomOracle: bool
|
|
vectors: seq[TestVector[EC]]
|
|
|
|
TestVector*[EC: ECP_ShortW_Aff] = object
|
|
P: EC
|
|
Q0, Q1: EC
|
|
msg: string
|
|
u: seq[string]
|
|
|
|
EC_G1_hex = object
|
|
x: string
|
|
y: string
|
|
|
|
Fp2_hex = string
|
|
|
|
EC_G2_hex = object
|
|
x: Fp2_hex
|
|
y: Fp2_hex
|
|
|
|
const
|
|
TestVectorsDir* =
|
|
currentSourcePath.rsplit(DirSep, 1)[0] / "protocol_hash_to_curve"
|
|
|
|
proc parseHook*(src: string, pos: var int, value: var ECP_ShortW_Aff) =
|
|
# Note when nim-serialization was used:
|
|
# When ECP_ShortW_Aff[Fp[Foo], G1]
|
|
# and ECP_ShortW_Aff[Fp[Foo], G2]
|
|
# are generated in the same file (i.e. twists and base curve are both on Fp)
|
|
# this creates bad codegen, in the C code, the `value`parameter gets the wrong type
|
|
# TODO: upstream
|
|
when ECP_ShortW_Aff.F is Fp:
|
|
var P: EC_G1_hex
|
|
parseHook(src, pos, P)
|
|
let ok = value.fromHex(P.x, P.y)
|
|
doAssert ok, "\nDeserialization error on G1 for\n" &
|
|
" P.x: " & P.x & "\n" &
|
|
" P.y: " & P.x & "\n"
|
|
elif ECP_ShortW_Aff.F is Fp2:
|
|
var P: EC_G2_hex
|
|
parseHook(src, pos, P)
|
|
let Px = P.x.split(',')
|
|
let Py = P.y.split(',')
|
|
|
|
let ok = value.fromHex(Px[0], Px[1], Py[0], Py[1])
|
|
doAssert ok, "\nDeserialization error on G2 for\n" &
|
|
" P.x0: " & Px[0] & "\n" &
|
|
" P.x1: " & Px[1] & "\n" &
|
|
" P.y0: " & Py[0] & "\n" &
|
|
" P.y1: " & Py[1] & "\n"
|
|
else:
|
|
{.error: "Not Implemented".}
|
|
|
|
proc loadVectors(TestType: typedesc, filename: string): TestType =
|
|
let content = readFile(TestVectorsDir/filename)
|
|
result = content.fromJson(TestType)
|
|
|
|
# Testing
|
|
# ------------------------------------------------------------------------
|
|
|
|
proc run_hash_to_curve_test(
|
|
EC: typedesc,
|
|
spec_version: string,
|
|
filename: string
|
|
) =
|
|
|
|
when EC.G == G1:
|
|
const G1_or_G2 = "G1"
|
|
else:
|
|
const G1_or_G2 = "G2"
|
|
let vec = loadVectors(HashToCurveTest[ECP_ShortW_Aff[EC.F, EC.G]], filename)
|
|
|
|
let testSuiteDesc = "Hash to Curve " & $EC.F.C & " " & G1_or_G2 & " - official specs " & spec_version & " test vectors"
|
|
|
|
suite testSuiteDesc & " [" & $WordBitWidth & "-bit words]":
|
|
|
|
doAssert vec.hash == "sha256"
|
|
doAssert vec.k == "0x80" # 128
|
|
|
|
for i in 0 ..< vec.vectors.len:
|
|
test "test " & $i & " - msg: \'" & vec.vectors[i].msg & "\'":
|
|
var P{.noInit.}: EC
|
|
sha256.hashToCurve(
|
|
k = 128,
|
|
output = P,
|
|
augmentation = "",
|
|
message = vec.vectors[i].msg,
|
|
domainSepTag = vec.dst
|
|
)
|
|
|
|
var P_ref: EC
|
|
P_ref.fromAffine(vec.vectors[i].P)
|
|
|
|
doAssert: bool(P == P_ref)
|
|
|
|
proc run_hash_to_curve_svdw_test(
|
|
EC: typedesc,
|
|
spec_version: string,
|
|
filename: string
|
|
) =
|
|
|
|
when EC.G == G1:
|
|
const G1_or_G2 = "G1"
|
|
else:
|
|
const G1_or_G2 = "G2"
|
|
let vec = loadVectors(HashToCurveTest[ECP_ShortW_Aff[EC.F, EC.G]], filename)
|
|
|
|
let testSuiteDesc = "Hash to Curve " & $EC.F.C & " " & G1_or_G2 & " - official specs " & spec_version & " test vectors"
|
|
|
|
suite testSuiteDesc & " [" & $WordBitWidth & "-bit words]":
|
|
|
|
doAssert vec.hash == "sha256"
|
|
doAssert vec.k == "0x80" # 128
|
|
|
|
for i in 0 ..< vec.vectors.len:
|
|
test "test " & $i & " - msg: \'" & vec.vectors[i].msg & "\'":
|
|
var P{.noInit.}: EC
|
|
sha256.hashToCurve_svdw(
|
|
k = 128,
|
|
output = P,
|
|
augmentation = "",
|
|
message = vec.vectors[i].msg,
|
|
domainSepTag = vec.dst
|
|
)
|
|
|
|
var P_ref: EC
|
|
P_ref.fromAffine(vec.vectors[i].P)
|
|
|
|
doAssert: bool(P == P_ref)
|
|
|
|
echo "\n------------------------------------------------------\n"
|
|
echo "Hash-to-curve" & '\n'
|
|
|
|
# Hash-to-curve v8 to latest
|
|
# https://github.com/cfrg/draft-irtf-cfrg-hash-to-curve/blob/draft-irtf-cfrg-hash-to-curve-10/poc/vectors/BLS12381G2_XMD:SHA-256_SSWU_RO_.json
|
|
run_hash_to_curve_test(
|
|
ECP_ShortW_Prj[Fp[BLS12_381], G1],
|
|
"v8",
|
|
"tv_h2c_v8_BLS12_381_hash_to_G1_SHA256_SSWU_RO.json"
|
|
)
|
|
|
|
run_hash_to_curve_test(
|
|
ECP_ShortW_Prj[Fp2[BLS12_381], G2],
|
|
"v8",
|
|
"tv_h2c_v8_BLS12_381_hash_to_G2_SHA256_SSWU_RO.json"
|
|
)
|
|
|
|
# Hash-to-curve v7 (different domain separation tag)
|
|
# https://github.com/cfrg/draft-irtf-cfrg-hash-to-curve/blob/draft-irtf-cfrg-hash-to-curve-07/poc/vectors/BLS12381G2_XMD:SHA-256_SSWU_RO_.json
|
|
run_hash_to_curve_test(
|
|
ECP_ShortW_Prj[Fp[BLS12_381], G1],
|
|
"v7",
|
|
"tv_h2c_v7_BLS12_381_hash_to_G1_SHA256_SSWU_RO.json"
|
|
)
|
|
|
|
run_hash_to_curve_test(
|
|
ECP_ShortW_Prj[Fp2[BLS12_381], G2],
|
|
"v7",
|
|
"tv_h2c_v7_BLS12_381_hash_to_G2_SHA256_SSWU_RO.json"
|
|
)
|
|
|
|
# With the slower universal SVDW mapping instead of SSWU
|
|
run_hash_to_curve_svdw_test(
|
|
ECP_ShortW_Jac[Fp[BLS12_381], G1],
|
|
"v7 (SVDW)",
|
|
"tv_h2c_v7_BLS12_381_hash_to_G1_SHA256_SVDW_RO.json"
|
|
)
|
|
|
|
run_hash_to_curve_svdw_test(
|
|
ECP_ShortW_Jac[Fp2[BLS12_381], G2],
|
|
"v7 (SVDW)",
|
|
"tv_h2c_v7_BLS12_381_hash_to_G2_SHA256_SVDW_RO.json"
|
|
)
|