2022-06-22 15:13:41 +03:00
# EIP-4844 -- Polynomial Commitments
## Table of contents
<!-- TOC -->
<!-- START doctoc generated TOC please keep comment here to allow auto update -->
<!-- DON'T EDIT THIS SECTION, INSTEAD RE - RUN doctoc TO UPDATE -->
- [Introduction ](#introduction )
- [Custom types ](#custom-types )
- [Constants ](#constants )
- [Preset ](#preset )
- [Trusted setup ](#trusted-setup )
- [Helper functions ](#helper-functions )
2022-09-26 16:39:16 +03:00
- [Bit-reversal permutation ](#bit-reversal-permutation )
- [`is_power_of_two` ](#is_power_of_two )
- [`reverse_bits` ](#reverse_bits )
- [`bit_reversal_permutation` ](#bit_reversal_permutation )
2022-06-22 15:13:41 +03:00
- [BLS12-381 helpers ](#bls12-381-helpers )
2022-09-26 18:57:00 +03:00
- [`bytes_to_bls_field` ](#bytes_to_bls_field )
2022-06-22 15:13:41 +03:00
- [`bls_modular_inverse` ](#bls_modular_inverse )
- [`div` ](#div )
2022-09-19 20:16:19 +01:00
- [`g1_lincomb` ](#g1_lincomb )
- [`vector_lincomb` ](#vector_lincomb )
2022-06-22 15:13:41 +03:00
- [KZG ](#kzg )
2022-07-13 13:13:30 +03:00
- [`blob_to_kzg_commitment` ](#blob_to_kzg_commitment )
2022-06-22 15:13:41 +03:00
- [`verify_kzg_proof` ](#verify_kzg_proof )
2022-07-13 13:13:30 +03:00
- [`compute_kzg_proof` ](#compute_kzg_proof )
2022-06-22 15:13:41 +03:00
- [Polynomials ](#polynomials )
- [`evaluate_polynomial_in_evaluation_form` ](#evaluate_polynomial_in_evaluation_form )
<!-- END doctoc generated TOC please keep comment here to allow auto update -->
<!-- /TOC -->
## Introduction
This document specifies basic polynomial operations and KZG polynomial commitment operations as they are needed for the EIP-4844 specification. The implementations are not optimized for performance, but readability. All practical implementations should optimize the polynomial operations.
## Custom types
| Name | SSZ equivalent | Description |
| - | - | - |
2022-07-13 13:12:31 +03:00
| `G1Point` | `Bytes48` | |
| `G2Point` | `Bytes96` | |
2022-06-22 15:13:41 +03:00
| `BLSFieldElement` | `uint256` | `x < BLS_MODULUS` |
| `KZGCommitment` | `Bytes48` | Same as BLS standard "is valid pubkey" check but also allows `0x00..00` for point-at-infinity |
| `KZGProof` | `Bytes48` | Same as for `KZGCommitment` |
## Constants
| Name | Value | Notes |
| - | - | - |
| `BLS_MODULUS` | `52435875175126190479447740508185965837690552500527637822603658699938581184513` | Scalar field modulus of BLS12-381 |
| `ROOTS_OF_UNITY` | `Vector[BLSFieldElement, FIELD_ELEMENTS_PER_BLOB]` | Roots of unity of order FIELD_ELEMENTS_PER_BLOB over the BLS12-381 field |
## Preset
### Trusted setup
The trusted setup is part of the preset: during testing a `minimal` insecure variant may be used,
but reusing the `mainnet` settings in public networks is a critical security requirement.
| Name | Value |
| - | - |
2022-07-13 13:12:31 +03:00
| `KZG_SETUP_G1` | `Vector[G1Point, FIELD_ELEMENTS_PER_BLOB]` , contents TBD |
2022-06-22 15:13:41 +03:00
| `KZG_SETUP_G2` | `Vector[G2Point, FIELD_ELEMENTS_PER_BLOB]` , contents TBD |
| `KZG_SETUP_LAGRANGE` | `Vector[KZGCommitment, FIELD_ELEMENTS_PER_BLOB]` , contents TBD |
## Helper functions
2022-09-26 16:39:16 +03:00
### Bit-reversal permutation
All polynomials (which are always given in Lagrange form) should be interpreted as being in
bit-reversal permutation. In practice, clients can implement this by storing the lists
`KZG_SETUP_LAGRANGE` and `ROOTS_OF_UNITY` in bit-reversal permutation, so these functions only
have to be called once at startup.
#### `is_power_of_two`
```python
def is_power_of_two(value: int) -> bool:
"""
Check if ``value` ` is a power of two integer.
"""
return (value > 0) and (value & (value - 1) == 0)
```
#### `reverse_bits`
```python
def reverse_bits(n: int, order: int) -> int:
"""
Reverse the bit order of an integer n
"""
assert is_power_of_two(order)
# Convert n to binary with the same number of bits as "order" - 1, then reverse its bit order
return int(('{:0' + str(order.bit_length() - 1) + 'b}').format(n)[::-1], 2)
```
#### `bit_reversal_permutation`
```python
2022-09-28 12:22:53 +08:00
def bit_reversal_permutation(sequence: Sequence[T]) -> Sequence[T]:
2022-09-26 16:39:16 +03:00
"""
2022-09-27 12:13:56 +01:00
Return a copy with bit-reversed permutation. The permutation is an involution (inverts itself).
2022-09-26 16:39:16 +03:00
The input and output are a sequence of generic type ``T` ` objects.
"""
2022-09-28 12:22:53 +08:00
return [sequence[reverse_bits(i, len(sequence))] for i in range(len(sequence))]
2022-09-26 16:39:16 +03:00
```
2022-06-22 15:13:41 +03:00
### BLS12-381 helpers
2022-09-26 18:57:00 +03:00
#### `bytes_to_bls_field`
```python
def bytes_to_bls_field(b: Bytes32) -> BLSFieldElement:
"""
Convert bytes to a BLS field scalar. The output is not uniform over the BLS field.
"""
return int.from_bytes(b, "little") % BLS_MODULUS
```
2022-06-22 15:13:41 +03:00
#### `bls_modular_inverse`
```python
def bls_modular_inverse(x: BLSFieldElement) -> BLSFieldElement:
"""
2022-06-22 15:19:24 +03:00
Compute the modular inverse of x
2022-06-22 15:13:41 +03:00
i.e. return y such that x * y % BLS_MODULUS == 1 and return 0 for x == 0
"""
2022-06-22 15:19:24 +03:00
return pow(x, -1, BLS_MODULUS) if x != 0 else 0
2022-06-22 15:13:41 +03:00
```
#### `div`
```python
2022-06-23 18:40:09 +08:00
def div(x: BLSFieldElement, y: BLSFieldElement) -> BLSFieldElement:
2022-06-22 15:13:41 +03:00
"""Divide two field elements: `x` by `y` """
2022-07-13 13:12:31 +03:00
return (int(x) * int(bls_modular_inverse(y))) % BLS_MODULUS
2022-06-22 15:13:41 +03:00
```
2022-09-19 20:16:19 +01:00
#### `g1_lincomb`
2022-06-22 15:13:41 +03:00
```python
2022-09-19 20:16:19 +01:00
def g1_lincomb(points: Sequence[KZGCommitment], scalars: Sequence[BLSFieldElement]) -> KZGCommitment:
2022-06-22 15:13:41 +03:00
"""
BLS multiscalar multiplication. This function can be optimized using Pippenger's algorithm and variants.
"""
2022-07-13 13:12:31 +03:00
assert len(points) == len(scalars)
result = bls.Z1
2022-06-22 15:13:41 +03:00
for x, a in zip(points, scalars):
2022-07-13 13:12:31 +03:00
result = bls.add(result, bls.multiply(bls.bytes48_to_G1(x), a))
return KZGCommitment(bls.G1_to_bytes48(result))
```
2022-09-19 20:16:19 +01:00
#### `vector_lincomb`
2022-07-13 13:12:31 +03:00
```python
2022-09-19 20:16:19 +01:00
def vector_lincomb(vectors: Sequence[Sequence[BLSFieldElement]],
2022-07-13 13:12:31 +03:00
scalars: Sequence[BLSFieldElement]) -> Sequence[BLSFieldElement]:
"""
Given a list of ``vectors` `, interpret it as a 2D matrix and compute the linear combination
of each column with `scalars` : return the resulting vector.
"""
result = [0] * len(vectors[0])
for v, s in zip(vectors, scalars):
for i, x in enumerate(v):
result[i] = (result[i] + int(s) * int(x)) % BLS_MODULUS
return [BLSFieldElement(x) for x in result]
2022-06-22 15:13:41 +03:00
```
### KZG
KZG core functions. These are also defined in EIP-4844 execution specs.
2022-07-13 13:12:31 +03:00
#### `blob_to_kzg_commitment`
2022-06-22 15:13:41 +03:00
```python
2022-07-13 13:12:31 +03:00
def blob_to_kzg_commitment(blob: Blob) -> KZGCommitment:
2022-09-26 16:39:16 +03:00
return g1_lincomb(bit_reversal_permutation(KZG_SETUP_LAGRANGE), blob)
2022-06-22 15:13:41 +03:00
```
#### `verify_kzg_proof`
```python
def verify_kzg_proof(polynomial_kzg: KZGCommitment,
2022-07-13 13:12:31 +03:00
z: BLSFieldElement,
2022-06-22 15:13:41 +03:00
y: BLSFieldElement,
2022-07-13 13:12:31 +03:00
kzg_proof: KZGProof) -> bool:
2022-06-23 18:40:09 +08:00
"""
2022-07-13 13:12:31 +03:00
Verify KZG proof that ``p(z) == y`` where ``p(z)`` is the polynomial represented by ``polynomial_kzg` `.
2022-06-23 18:40:09 +08:00
"""
2022-07-13 13:12:31 +03:00
# Verify: P - y = Q * (X - z)
X_minus_z = bls.add(bls.bytes96_to_G2(KZG_SETUP_G2[1]), bls.multiply(bls.G2, BLS_MODULUS - z))
P_minus_y = bls.add(bls.bytes48_to_G1(polynomial_kzg), bls.multiply(bls.G1, BLS_MODULUS - y))
2022-06-22 15:13:41 +03:00
return bls.pairing_check([
[P_minus_y, bls.neg(bls.G2)],
2022-07-13 13:12:31 +03:00
[bls.bytes48_to_G1(kzg_proof), X_minus_z]
2022-06-22 15:13:41 +03:00
])
```
2022-07-13 13:12:31 +03:00
#### `compute_kzg_proof`
```python
def compute_kzg_proof(polynomial: Sequence[BLSFieldElement], z: BLSFieldElement) -> KZGProof:
2022-09-26 16:39:16 +03:00
"""
Compute KZG proof at point `z` with `polynomial` being in evaluation form
"""
2022-07-13 13:12:31 +03:00
# To avoid SSZ overflow/underflow, convert element into int
polynomial = [int(i) for i in polynomial]
z = int(z)
# Shift our polynomial first (in evaluation form we can't handle the division remainder)
y = evaluate_polynomial_in_evaluation_form(polynomial, z)
polynomial_shifted = [(p - int(y)) % BLS_MODULUS for p in polynomial]
# Make sure we won't divide by zero during division
assert z not in ROOTS_OF_UNITY
2022-09-26 16:39:16 +03:00
denominator_poly = [(x - z) % BLS_MODULUS for x in bit_reversal_permutation(ROOTS_OF_UNITY)]
2022-07-13 13:12:31 +03:00
# Calculate quotient polynomial by doing point-by-point division
quotient_polynomial = [div(a, b) for a, b in zip(polynomial_shifted, denominator_poly)]
2022-09-26 16:39:16 +03:00
return KZGProof(g1_lincomb(bit_reversal_permutation(KZG_SETUP_LAGRANGE), quotient_polynomial))
2022-07-13 13:12:31 +03:00
```
2022-06-22 15:13:41 +03:00
### Polynomials
#### `evaluate_polynomial_in_evaluation_form`
```python
2022-07-13 13:12:31 +03:00
def evaluate_polynomial_in_evaluation_form(polynomial: Sequence[BLSFieldElement],
z: BLSFieldElement) -> BLSFieldElement:
2022-06-22 15:13:41 +03:00
"""
2022-07-13 13:12:31 +03:00
Evaluate a polynomial (in evaluation form) at an arbitrary point `z`
2022-06-22 15:13:41 +03:00
Uses the barycentric formula:
2022-07-13 13:12:31 +03:00
f(z) = (1 - z**WIDTH) / WIDTH * sum_(i=0)^WIDTH (f(DOMAIN[i]) * DOMAIN[i]) / (z - DOMAIN[i])
2022-06-22 15:13:41 +03:00
"""
2022-07-13 13:12:31 +03:00
width = len(polynomial)
2022-06-22 15:13:41 +03:00
assert width == FIELD_ELEMENTS_PER_BLOB
inverse_width = bls_modular_inverse(width)
2022-07-13 13:12:31 +03:00
# Make sure we won't divide by zero during division
assert z not in ROOTS_OF_UNITY
2022-06-22 15:13:41 +03:00
2022-09-26 16:39:16 +03:00
roots_of_unity_brp = bit_reversal_permutation(ROOTS_OF_UNITY)
2022-07-13 13:12:31 +03:00
result = 0
for i in range(width):
2022-09-26 16:39:16 +03:00
result += div(int(polynomial[i]) * int(roots_of_unity_brp[i]), (z - roots_of_unity_brp[i]))
2022-07-13 13:12:31 +03:00
result = result * (pow(z, width, BLS_MODULUS) - 1) * inverse_width % BLS_MODULUS
return result
2022-06-22 15:13:41 +03:00
```
2022-07-13 13:12:31 +03:00