use crate::{
crypto,
utilities::{bool_to_bytes32, right_pad},
Address, Precompile, PrecompileError, PrecompileId, PrecompileOutput, PrecompileResult,
};
use std::vec::Vec;
#[allow(dead_code)]
pub mod arkworks;
cfg_if::cfg_if! {
if #[cfg(feature = "bn")]{
pub(crate) mod substrate;
pub(crate) use substrate as crypto_backend;
} else {
pub(crate) use arkworks as crypto_backend;
}
}
pub mod add {
use super::*;
pub const ADDRESS: Address = crate::u64_to_address(6);
pub const ISTANBUL_ADD_GAS_COST: u64 = 150;
pub const ISTANBUL: Precompile =
Precompile::new(PrecompileId::Bn254Add, ADDRESS, |input, gas_limit| {
run_add(input, ISTANBUL_ADD_GAS_COST, gas_limit)
});
pub const BYZANTIUM_ADD_GAS_COST: u64 = 500;
pub const BYZANTIUM: Precompile =
Precompile::new(PrecompileId::Bn254Add, ADDRESS, |input, gas_limit| {
run_add(input, BYZANTIUM_ADD_GAS_COST, gas_limit)
});
}
pub mod mul {
use super::*;
pub const ADDRESS: Address = crate::u64_to_address(7);
pub const ISTANBUL_MUL_GAS_COST: u64 = 6_000;
pub const ISTANBUL: Precompile =
Precompile::new(PrecompileId::Bn254Mul, ADDRESS, |input, gas_limit| {
run_mul(input, ISTANBUL_MUL_GAS_COST, gas_limit)
});
pub const BYZANTIUM_MUL_GAS_COST: u64 = 40_000;
pub const BYZANTIUM: Precompile =
Precompile::new(PrecompileId::Bn254Mul, ADDRESS, |input, gas_limit| {
run_mul(input, BYZANTIUM_MUL_GAS_COST, gas_limit)
});
}
pub mod pair {
use super::*;
pub const ADDRESS: Address = crate::u64_to_address(8);
pub const ISTANBUL_PAIR_PER_POINT: u64 = 34_000;
pub const ISTANBUL_PAIR_BASE: u64 = 45_000;
pub const ISTANBUL: Precompile =
Precompile::new(PrecompileId::Bn254Pairing, ADDRESS, |input, gas_limit| {
run_pair(
input,
ISTANBUL_PAIR_PER_POINT,
ISTANBUL_PAIR_BASE,
gas_limit,
)
});
pub const BYZANTIUM_PAIR_PER_POINT: u64 = 80_000;
pub const BYZANTIUM_PAIR_BASE: u64 = 100_000;
pub const BYZANTIUM: Precompile =
Precompile::new(PrecompileId::Bn254Pairing, ADDRESS, |input, gas_limit| {
run_pair(
input,
BYZANTIUM_PAIR_PER_POINT,
BYZANTIUM_PAIR_BASE,
gas_limit,
)
});
}
const FQ_LEN: usize = 32;
const SCALAR_LEN: usize = 32;
const FQ2_LEN: usize = 2 * FQ_LEN;
const G1_LEN: usize = 2 * FQ_LEN;
const G2_LEN: usize = 2 * FQ2_LEN;
pub const ADD_INPUT_LEN: usize = 2 * G1_LEN;
pub const MUL_INPUT_LEN: usize = G1_LEN + SCALAR_LEN;
pub const PAIR_ELEMENT_LEN: usize = G1_LEN + G2_LEN;
pub fn run_add(input: &[u8], gas_cost: u64, gas_limit: u64) -> PrecompileResult {
if gas_cost > gas_limit {
return Err(PrecompileError::OutOfGas);
}
let input = right_pad::<ADD_INPUT_LEN>(input);
let p1_bytes = &input[..G1_LEN];
let p2_bytes = &input[G1_LEN..];
let output = crypto().bn254_g1_add(p1_bytes, p2_bytes)?;
Ok(PrecompileOutput::new(gas_cost, output.into()))
}
pub fn run_mul(input: &[u8], gas_cost: u64, gas_limit: u64) -> PrecompileResult {
if gas_cost > gas_limit {
return Err(PrecompileError::OutOfGas);
}
let input = right_pad::<MUL_INPUT_LEN>(input);
let point_bytes = &input[..G1_LEN];
let scalar_bytes = &input[G1_LEN..G1_LEN + SCALAR_LEN];
let output = crypto().bn254_g1_mul(point_bytes, scalar_bytes)?;
Ok(PrecompileOutput::new(gas_cost, output.into()))
}
pub fn run_pair(
input: &[u8],
pair_per_point_cost: u64,
pair_base_cost: u64,
gas_limit: u64,
) -> PrecompileResult {
let gas_used = (input.len() / PAIR_ELEMENT_LEN) as u64 * pair_per_point_cost + pair_base_cost;
if gas_used > gas_limit {
return Err(PrecompileError::OutOfGas);
}
if !input.len().is_multiple_of(PAIR_ELEMENT_LEN) {
return Err(PrecompileError::Bn254PairLength);
}
let elements = input.len() / PAIR_ELEMENT_LEN;
let mut points = Vec::with_capacity(elements);
for idx in 0..elements {
let start = idx * PAIR_ELEMENT_LEN;
let g1_start = start;
let g2_start = start + G1_LEN;
let encoded_g1_element = &input[g1_start..g2_start];
let encoded_g2_element = &input[g2_start..g2_start + G2_LEN];
points.push((encoded_g1_element, encoded_g2_element));
}
let pairing_result = crypto().bn254_pairing_check(&points)?;
Ok(PrecompileOutput::new(
gas_used,
bool_to_bytes32(pairing_result),
))
}
#[cfg(test)]
mod tests {
use crate::{
bn254::{
add::BYZANTIUM_ADD_GAS_COST,
mul::BYZANTIUM_MUL_GAS_COST,
pair::{BYZANTIUM_PAIR_BASE, BYZANTIUM_PAIR_PER_POINT},
},
PrecompileError,
};
use primitives::hex;
use super::*;
#[test]
fn test_bn254_add() {
let input = hex::decode(
"\
18b18acfb4c2c30276db5411368e7185b311dd124691610c5d3b74034e093dc9\
063c909c4720840cb5134cb9f59fa749755796819658d32efc0d288198f37266\
07c2b7f58a84bd6145f00c9c2bc0bb1a187f20ff2c92963a88019e7c6a014eed\
06614e20c147e940f2d70da3f74c9a17df361706a4485c742bd6788478fa17d7",
)
.unwrap();
let expected = hex::decode(
"\
2243525c5efd4b9c3d3c45ac0ca3fe4dd85e830a4ce6b65fa1eeaee202839703\
301d1d33be6da8e509df21cc35964723180eed7532537db9ae5e7d48f195c915",
)
.unwrap();
let outcome = run_add(&input, BYZANTIUM_ADD_GAS_COST, 500).unwrap();
assert_eq!(outcome.bytes, expected);
let input = hex::decode(
"\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000",
)
.unwrap();
let expected = hex::decode(
"\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000",
)
.unwrap();
let outcome = run_add(&input, BYZANTIUM_ADD_GAS_COST, 500).unwrap();
assert_eq!(outcome.bytes, expected);
let input = hex::decode(
"\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000",
)
.unwrap();
let res = run_add(&input, BYZANTIUM_ADD_GAS_COST, 499);
assert!(matches!(res, Err(PrecompileError::OutOfGas)));
let input = [0u8; 0];
let expected = hex::decode(
"\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000",
)
.unwrap();
let outcome = run_add(&input, BYZANTIUM_ADD_GAS_COST, 500).unwrap();
assert_eq!(outcome.bytes, expected);
let input = hex::decode(
"\
1111111111111111111111111111111111111111111111111111111111111111\
1111111111111111111111111111111111111111111111111111111111111111\
1111111111111111111111111111111111111111111111111111111111111111\
1111111111111111111111111111111111111111111111111111111111111111",
)
.unwrap();
let res = run_add(&input, BYZANTIUM_ADD_GAS_COST, 500);
assert!(matches!(
res,
Err(PrecompileError::Bn254AffineGFailedToCreate)
));
}
#[test]
fn test_bn254_mul() {
let input = hex::decode(
"\
2bd3e6d0f3b142924f5ca7b49ce5b9d54c4703d7ae5648e61d02268b1a0a9fb7\
21611ce0a6af85915e2f1d70300909ce2e49dfad4a4619c8390cae66cefdb204\
00000000000000000000000000000000000000000000000011138ce750fa15c2",
)
.unwrap();
let expected = hex::decode(
"\
070a8d6a982153cae4be29d434e8faef8a47b274a053f5a4ee2a6c9c13c31e5c\
031b8ce914eba3a9ffb989f9cdd5b0f01943074bf4f0f315690ec3cec6981afc",
)
.unwrap();
let outcome = run_mul(&input, BYZANTIUM_MUL_GAS_COST, 40_000).unwrap();
assert_eq!(outcome.bytes, expected);
let input = hex::decode(
"\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000\
0200000000000000000000000000000000000000000000000000000000000000",
)
.unwrap();
let res = run_mul(&input, BYZANTIUM_MUL_GAS_COST, 39_999);
assert!(matches!(res, Err(PrecompileError::OutOfGas)));
let input = hex::decode(
"\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000\
0200000000000000000000000000000000000000000000000000000000000000",
)
.unwrap();
let expected = hex::decode(
"\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000",
)
.unwrap();
let outcome = run_mul(&input, BYZANTIUM_MUL_GAS_COST, 40_000).unwrap();
assert_eq!(outcome.bytes, expected);
let input = [0u8; 0];
let expected = hex::decode(
"\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000",
)
.unwrap();
let outcome = run_mul(&input, BYZANTIUM_MUL_GAS_COST, 40_000).unwrap();
assert_eq!(outcome.bytes, expected);
let input = hex::decode(
"\
1111111111111111111111111111111111111111111111111111111111111111\
1111111111111111111111111111111111111111111111111111111111111111\
0f00000000000000000000000000000000000000000000000000000000000000",
)
.unwrap();
let res = run_mul(&input, BYZANTIUM_MUL_GAS_COST, 40_000);
assert!(matches!(
res,
Err(PrecompileError::Bn254AffineGFailedToCreate)
));
}
#[test]
fn test_bn254_pair() {
let input = hex::decode(
"\
1c76476f4def4bb94541d57ebba1193381ffa7aa76ada664dd31c16024c43f59\
3034dd2920f673e204fee2811c678745fc819b55d3e9d294e45c9b03a76aef41\
209dd15ebff5d46c4bd888e51a93cf99a7329636c63514396b4a452003a35bf7\
04bf11ca01483bfa8b34b43561848d28905960114c8ac04049af4b6315a41678\
2bb8324af6cfc93537a2ad1a445cfd0ca2a71acd7ac41fadbf933c2a51be344d\
120a2a4cf30c1bf9845f20c6fe39e07ea2cce61f0c9bb048165fe5e4de877550\
111e129f1cf1097710d41c4ac70fcdfa5ba2023c6ff1cbeac322de49d1b6df7c\
2032c61a830e3c17286de9462bf242fca2883585b93870a73853face6a6bf411\
198e9393920d483a7260bfb731fb5d25f1aa493335a9e71297e485b7aef312c2\
1800deef121f1e76426a00665e5c4479674322d4f75edadd46debd5cd992f6ed\
090689d0585ff075ec9e99ad690c3395bc4b313370b38ef355acdadcd122975b\
12c85ea5db8c6deb4aab71808dcb408fe3d1e7690c43d37b4ce6cc0166fa7daa",
)
.unwrap();
let expected =
hex::decode("0000000000000000000000000000000000000000000000000000000000000001")
.unwrap();
let outcome = run_pair(
&input,
BYZANTIUM_PAIR_PER_POINT,
BYZANTIUM_PAIR_BASE,
260_000,
)
.unwrap();
assert_eq!(outcome.bytes, expected);
let input = hex::decode(
"\
1c76476f4def4bb94541d57ebba1193381ffa7aa76ada664dd31c16024c43f59\
3034dd2920f673e204fee2811c678745fc819b55d3e9d294e45c9b03a76aef41\
209dd15ebff5d46c4bd888e51a93cf99a7329636c63514396b4a452003a35bf7\
04bf11ca01483bfa8b34b43561848d28905960114c8ac04049af4b6315a41678\
2bb8324af6cfc93537a2ad1a445cfd0ca2a71acd7ac41fadbf933c2a51be344d\
120a2a4cf30c1bf9845f20c6fe39e07ea2cce61f0c9bb048165fe5e4de877550\
111e129f1cf1097710d41c4ac70fcdfa5ba2023c6ff1cbeac322de49d1b6df7c\
2032c61a830e3c17286de9462bf242fca2883585b93870a73853face6a6bf411\
198e9393920d483a7260bfb731fb5d25f1aa493335a9e71297e485b7aef312c2\
1800deef121f1e76426a00665e5c4479674322d4f75edadd46debd5cd992f6ed\
090689d0585ff075ec9e99ad690c3395bc4b313370b38ef355acdadcd122975b\
12c85ea5db8c6deb4aab71808dcb408fe3d1e7690c43d37b4ce6cc0166fa7daa",
)
.unwrap();
let res = run_pair(
&input,
BYZANTIUM_PAIR_PER_POINT,
BYZANTIUM_PAIR_BASE,
259_999,
);
assert!(matches!(res, Err(PrecompileError::OutOfGas)));
let input = [0u8; 0];
let expected =
hex::decode("0000000000000000000000000000000000000000000000000000000000000001")
.unwrap();
let outcome = run_pair(
&input,
BYZANTIUM_PAIR_PER_POINT,
BYZANTIUM_PAIR_BASE,
260_000,
)
.unwrap();
assert_eq!(outcome.bytes, expected);
let input = hex::decode(
"\
1111111111111111111111111111111111111111111111111111111111111111\
1111111111111111111111111111111111111111111111111111111111111111\
1111111111111111111111111111111111111111111111111111111111111111\
1111111111111111111111111111111111111111111111111111111111111111\
1111111111111111111111111111111111111111111111111111111111111111\
1111111111111111111111111111111111111111111111111111111111111111",
)
.unwrap();
let res = run_pair(
&input,
BYZANTIUM_PAIR_PER_POINT,
BYZANTIUM_PAIR_BASE,
260_000,
);
assert!(matches!(
res,
Err(PrecompileError::Bn254AffineGFailedToCreate)
));
let input = hex::decode(
"\
1111111111111111111111111111111111111111111111111111111111111111\
1111111111111111111111111111111111111111111111111111111111111111\
111111111111111111111111111111\
",
)
.unwrap();
let res = run_pair(
&input,
BYZANTIUM_PAIR_PER_POINT,
BYZANTIUM_PAIR_BASE,
260_000,
);
assert!(matches!(res, Err(PrecompileError::Bn254PairLength)));
let input = hex::decode(
"\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000\
209dd15ebff5d46c4bd888e51a93cf99a7329636c63514396b4a452003a35bf7\
04bf11ca01483bfa8b34b43561848d28905960114c8ac04049af4b6315a41678\
2bb8324af6cfc93537a2ad1a445cfd0ca2a71acd7ac41fadbf933c2a51be344d\
120a2a4cf30c1bf9845f20c6fe39e07ea2cce61f0c9bb048165fe5e4de877550",
)
.unwrap();
let expected =
hex::decode("0000000000000000000000000000000000000000000000000000000000000001")
.unwrap();
let outcome = run_pair(
&input,
BYZANTIUM_PAIR_PER_POINT,
BYZANTIUM_PAIR_BASE,
260_000,
)
.unwrap();
assert_eq!(outcome.bytes, expected);
let input = hex::decode(
"\
1c76476f4def4bb94541d57ebba1193381ffa7aa76ada664dd31c16024c43f59\
3034dd2920f673e204fee2811c678745fc819b55d3e9d294e45c9b03a76aef41\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000\
0000000000000000000000000000000000000000000000000000000000000000",
)
.unwrap();
let outcome = run_pair(
&input,
BYZANTIUM_PAIR_PER_POINT,
BYZANTIUM_PAIR_BASE,
260_000,
)
.unwrap();
assert_eq!(outcome.bytes, expected);
}
}