use crate::object_runtime::ObjectRuntime;
use crate::NativesCostTable;
use fastcrypto::error::{FastCryptoError, FastCryptoResult};
use fastcrypto::groups::{
bls12381 as bls, FromTrustedByteArray, GroupElement, HashToGroupElement, MultiScalarMul,
Pairing,
};
use fastcrypto::serde_helpers::ToFromByteArray;
use move_binary_format::errors::{PartialVMError, PartialVMResult};
use move_core_types::gas_algebra::InternalGas;
use move_core_types::vm_status::StatusCode;
use move_vm_runtime::native_charge_gas_early_exit;
use move_vm_runtime::native_functions::NativeContext;
use move_vm_types::{
loaded_data::runtime_types::Type,
natives::function::NativeResult,
pop_arg,
values::{Value, VectorRef},
};
use smallvec::smallvec;
use std::collections::VecDeque;
pub const NOT_SUPPORTED_ERROR: u64 = 0;
pub const INVALID_INPUT_ERROR: u64 = 1;
pub const INPUT_TOO_LONG_ERROR: u64 = 2;
fn is_supported(context: &NativeContext) -> PartialVMResult<bool> {
Ok(context
.extensions()
.get::<ObjectRuntime>()?
.protocol_config
.enable_group_ops_native_functions())
}
fn is_msm_supported(context: &NativeContext) -> PartialVMResult<bool> {
Ok(context
.extensions()
.get::<ObjectRuntime>()?
.protocol_config
.enable_group_ops_native_function_msm())
}
fn is_uncompressed_g1_supported(context: &NativeContext) -> PartialVMResult<bool> {
Ok(context
.extensions()
.get::<ObjectRuntime>()?
.protocol_config
.uncompressed_g1_group_elements())
}
fn v2_native_charge(context: &NativeContext, cost: InternalGas) -> PartialVMResult<InternalGas> {
Ok(
if context
.extensions()
.get::<ObjectRuntime>()?
.protocol_config
.native_charging_v2()
{
context.gas_used()
} else {
cost
},
)
}
fn map_op_result(
context: &NativeContext,
cost: InternalGas,
result: FastCryptoResult<Vec<u8>>,
) -> PartialVMResult<NativeResult> {
match result {
Ok(bytes) => Ok(NativeResult::ok(
v2_native_charge(context, cost)?,
smallvec![Value::vector_u8(bytes)],
)),
Err(_) => Ok(NativeResult::err(
v2_native_charge(context, cost)?,
INVALID_INPUT_ERROR,
)),
}
}
#[derive(Clone)]
pub struct GroupOpsCostParams {
pub bls12381_decode_scalar_cost: Option<InternalGas>,
pub bls12381_decode_g1_cost: Option<InternalGas>,
pub bls12381_decode_g2_cost: Option<InternalGas>,
pub bls12381_decode_gt_cost: Option<InternalGas>,
pub bls12381_scalar_add_cost: Option<InternalGas>,
pub bls12381_g1_add_cost: Option<InternalGas>,
pub bls12381_g2_add_cost: Option<InternalGas>,
pub bls12381_gt_add_cost: Option<InternalGas>,
pub bls12381_scalar_sub_cost: Option<InternalGas>,
pub bls12381_g1_sub_cost: Option<InternalGas>,
pub bls12381_g2_sub_cost: Option<InternalGas>,
pub bls12381_gt_sub_cost: Option<InternalGas>,
pub bls12381_scalar_mul_cost: Option<InternalGas>,
pub bls12381_g1_mul_cost: Option<InternalGas>,
pub bls12381_g2_mul_cost: Option<InternalGas>,
pub bls12381_gt_mul_cost: Option<InternalGas>,
pub bls12381_scalar_div_cost: Option<InternalGas>,
pub bls12381_g1_div_cost: Option<InternalGas>,
pub bls12381_g2_div_cost: Option<InternalGas>,
pub bls12381_gt_div_cost: Option<InternalGas>,
pub bls12381_g1_hash_to_base_cost: Option<InternalGas>,
pub bls12381_g2_hash_to_base_cost: Option<InternalGas>,
pub bls12381_g1_hash_to_cost_per_byte: Option<InternalGas>,
pub bls12381_g2_hash_to_cost_per_byte: Option<InternalGas>,
pub bls12381_g1_msm_base_cost: Option<InternalGas>,
pub bls12381_g2_msm_base_cost: Option<InternalGas>,
pub bls12381_g1_msm_base_cost_per_input: Option<InternalGas>,
pub bls12381_g2_msm_base_cost_per_input: Option<InternalGas>,
pub bls12381_msm_max_len: Option<u32>,
pub bls12381_pairing_cost: Option<InternalGas>,
pub bls12381_g1_to_uncompressed_g1_cost: Option<InternalGas>,
pub bls12381_uncompressed_g1_to_g1_cost: Option<InternalGas>,
pub bls12381_uncompressed_g1_sum_base_cost: Option<InternalGas>,
pub bls12381_uncompressed_g1_sum_cost_per_term: Option<InternalGas>,
pub bls12381_uncompressed_g1_sum_max_terms: Option<u64>,
}
macro_rules! native_charge_gas_early_exit_option {
($native_context:ident, $cost:expr) => {{
use move_binary_format::errors::PartialVMError;
use move_core_types::vm_status::StatusCode;
native_charge_gas_early_exit!(
$native_context,
$cost.ok_or_else(|| {
PartialVMError::new(StatusCode::UNKNOWN_INVARIANT_VIOLATION_ERROR)
.with_message("Gas cost for group ops is missing".to_string())
})?
);
}};
}
#[repr(u8)]
enum Groups {
BLS12381Scalar = 0,
BLS12381G1 = 1,
BLS12381G2 = 2,
BLS12381GT = 3,
BLS12381UncompressedG1 = 4,
}
impl Groups {
fn from_u8(value: u8) -> Option<Self> {
match value {
0 => Some(Groups::BLS12381Scalar),
1 => Some(Groups::BLS12381G1),
2 => Some(Groups::BLS12381G2),
3 => Some(Groups::BLS12381GT),
4 => Some(Groups::BLS12381UncompressedG1),
_ => None,
}
}
}
fn parse_untrusted<G: ToFromByteArray<S> + FromTrustedByteArray<S>, const S: usize>(
e: &[u8],
) -> FastCryptoResult<G> {
G::from_byte_array(e.try_into().map_err(|_| FastCryptoError::InvalidInput)?)
}
fn parse_trusted<G: ToFromByteArray<S> + FromTrustedByteArray<S>, const S: usize>(
e: &[u8],
) -> FastCryptoResult<G> {
G::from_trusted_byte_array(e.try_into().map_err(|_| FastCryptoError::InvalidInput)?)
}
fn binary_op_diff<
G1: ToFromByteArray<S1> + FromTrustedByteArray<S1>,
G2: ToFromByteArray<S2> + FromTrustedByteArray<S2>,
const S1: usize,
const S2: usize,
>(
op: impl Fn(G1, G2) -> FastCryptoResult<G2>,
a1: &[u8],
a2: &[u8],
) -> FastCryptoResult<Vec<u8>> {
let e1 = parse_trusted::<G1, S1>(a1)?;
let e2 = parse_trusted::<G2, S2>(a2)?;
let result = op(e1, e2)?;
Ok(result.to_byte_array().to_vec())
}
fn binary_op<G: ToFromByteArray<S> + FromTrustedByteArray<S>, const S: usize>(
op: impl Fn(G, G) -> FastCryptoResult<G>,
a1: &[u8],
a2: &[u8],
) -> FastCryptoResult<Vec<u8>> {
binary_op_diff::<G, G, S, S>(op, a1, a2)
}
pub fn internal_validate(
context: &mut NativeContext,
ty_args: Vec<Type>,
mut args: VecDeque<Value>,
) -> PartialVMResult<NativeResult> {
debug_assert!(ty_args.is_empty());
debug_assert!(args.len() == 2);
let cost = context.gas_used();
if !is_supported(context)? {
return Ok(NativeResult::err(cost, NOT_SUPPORTED_ERROR));
}
let bytes_ref = pop_arg!(args, VectorRef);
let bytes = bytes_ref.as_bytes_ref();
let group_type = pop_arg!(args, u8);
let cost_params = &context
.extensions()
.get::<NativesCostTable>()?
.group_ops_cost_params
.clone();
let result = match Groups::from_u8(group_type) {
Some(Groups::BLS12381Scalar) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_decode_scalar_cost);
parse_untrusted::<bls::Scalar, { bls::Scalar::BYTE_LENGTH }>(&bytes).is_ok()
}
Some(Groups::BLS12381G1) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_decode_g1_cost);
parse_untrusted::<bls::G1Element, { bls::G1Element::BYTE_LENGTH }>(&bytes).is_ok()
}
Some(Groups::BLS12381G2) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_decode_g2_cost);
parse_untrusted::<bls::G2Element, { bls::G2Element::BYTE_LENGTH }>(&bytes).is_ok()
}
_ => false,
};
Ok(NativeResult::ok(
v2_native_charge(context, cost)?,
smallvec![Value::bool(result)],
))
}
pub fn internal_add(
context: &mut NativeContext,
ty_args: Vec<Type>,
mut args: VecDeque<Value>,
) -> PartialVMResult<NativeResult> {
debug_assert!(ty_args.is_empty());
debug_assert!(args.len() == 3);
let cost = context.gas_used();
if !is_supported(context)? {
return Ok(NativeResult::err(cost, NOT_SUPPORTED_ERROR));
}
let e2_ref = pop_arg!(args, VectorRef);
let e2 = e2_ref.as_bytes_ref();
let e1_ref = pop_arg!(args, VectorRef);
let e1 = e1_ref.as_bytes_ref();
let group_type = pop_arg!(args, u8);
let cost_params = &context
.extensions()
.get::<NativesCostTable>()?
.group_ops_cost_params
.clone();
let result = match Groups::from_u8(group_type) {
Some(Groups::BLS12381Scalar) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_scalar_add_cost);
binary_op::<bls::Scalar, { bls::Scalar::BYTE_LENGTH }>(|a, b| Ok(a + b), &e1, &e2)
}
Some(Groups::BLS12381G1) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_g1_add_cost);
binary_op::<bls::G1Element, { bls::G1Element::BYTE_LENGTH }>(|a, b| Ok(a + b), &e1, &e2)
}
Some(Groups::BLS12381G2) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_g2_add_cost);
binary_op::<bls::G2Element, { bls::G2Element::BYTE_LENGTH }>(|a, b| Ok(a + b), &e1, &e2)
}
Some(Groups::BLS12381GT) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_gt_add_cost);
binary_op::<bls::GTElement, { bls::GTElement::BYTE_LENGTH }>(|a, b| Ok(a + b), &e1, &e2)
}
_ => Err(FastCryptoError::InvalidInput),
};
map_op_result(context, cost, result)
}
pub fn internal_sub(
context: &mut NativeContext,
ty_args: Vec<Type>,
mut args: VecDeque<Value>,
) -> PartialVMResult<NativeResult> {
debug_assert!(ty_args.is_empty());
debug_assert!(args.len() == 3);
let cost = context.gas_used();
if !is_supported(context)? {
return Ok(NativeResult::err(cost, NOT_SUPPORTED_ERROR));
}
let e2_ref = pop_arg!(args, VectorRef);
let e2 = e2_ref.as_bytes_ref();
let e1_ref = pop_arg!(args, VectorRef);
let e1 = e1_ref.as_bytes_ref();
let group_type = pop_arg!(args, u8);
let cost_params = &context
.extensions()
.get::<NativesCostTable>()?
.group_ops_cost_params
.clone();
let result = match Groups::from_u8(group_type) {
Some(Groups::BLS12381Scalar) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_scalar_sub_cost);
binary_op::<bls::Scalar, { bls::Scalar::BYTE_LENGTH }>(|a, b| Ok(a - b), &e1, &e2)
}
Some(Groups::BLS12381G1) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_g1_sub_cost);
binary_op::<bls::G1Element, { bls::G1Element::BYTE_LENGTH }>(|a, b| Ok(a - b), &e1, &e2)
}
Some(Groups::BLS12381G2) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_g2_sub_cost);
binary_op::<bls::G2Element, { bls::G2Element::BYTE_LENGTH }>(|a, b| Ok(a - b), &e1, &e2)
}
Some(Groups::BLS12381GT) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_gt_sub_cost);
binary_op::<bls::GTElement, { bls::GTElement::BYTE_LENGTH }>(|a, b| Ok(a - b), &e1, &e2)
}
_ => Err(FastCryptoError::InvalidInput),
};
map_op_result(context, cost, result)
}
pub fn internal_mul(
context: &mut NativeContext,
ty_args: Vec<Type>,
mut args: VecDeque<Value>,
) -> PartialVMResult<NativeResult> {
debug_assert!(ty_args.is_empty());
debug_assert!(args.len() == 3);
let cost = context.gas_used();
if !is_supported(context)? {
return Ok(NativeResult::err(cost, NOT_SUPPORTED_ERROR));
}
let e2_ref = pop_arg!(args, VectorRef);
let e2 = e2_ref.as_bytes_ref();
let e1_ref = pop_arg!(args, VectorRef);
let e1 = e1_ref.as_bytes_ref();
let group_type = pop_arg!(args, u8);
let cost_params = &context
.extensions()
.get::<NativesCostTable>()?
.group_ops_cost_params
.clone();
let result = match Groups::from_u8(group_type) {
Some(Groups::BLS12381Scalar) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_scalar_mul_cost);
binary_op::<bls::Scalar, { bls::Scalar::BYTE_LENGTH }>(|a, b| Ok(b * a), &e1, &e2)
}
Some(Groups::BLS12381G1) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_g1_mul_cost);
binary_op_diff::<
bls::Scalar,
bls::G1Element,
{ bls::Scalar::BYTE_LENGTH },
{ bls::G1Element::BYTE_LENGTH },
>(|a, b| Ok(b * a), &e1, &e2)
}
Some(Groups::BLS12381G2) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_g2_mul_cost);
binary_op_diff::<
bls::Scalar,
bls::G2Element,
{ bls::Scalar::BYTE_LENGTH },
{ bls::G2Element::BYTE_LENGTH },
>(|a, b| Ok(b * a), &e1, &e2)
}
Some(Groups::BLS12381GT) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_gt_mul_cost);
binary_op_diff::<
bls::Scalar,
bls::GTElement,
{ bls::Scalar::BYTE_LENGTH },
{ bls::GTElement::BYTE_LENGTH },
>(|a, b| Ok(b * a), &e1, &e2)
}
_ => Err(FastCryptoError::InvalidInput),
};
map_op_result(context, cost, result)
}
pub fn internal_div(
context: &mut NativeContext,
ty_args: Vec<Type>,
mut args: VecDeque<Value>,
) -> PartialVMResult<NativeResult> {
debug_assert!(ty_args.is_empty());
debug_assert!(args.len() == 3);
let cost = context.gas_used();
if !is_supported(context)? {
return Ok(NativeResult::err(cost, NOT_SUPPORTED_ERROR));
}
let e2_ref = pop_arg!(args, VectorRef);
let e2 = e2_ref.as_bytes_ref();
let e1_ref = pop_arg!(args, VectorRef);
let e1 = e1_ref.as_bytes_ref();
let group_type = pop_arg!(args, u8);
let cost_params = &context
.extensions()
.get::<NativesCostTable>()?
.group_ops_cost_params
.clone();
let result = match Groups::from_u8(group_type) {
Some(Groups::BLS12381Scalar) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_scalar_div_cost);
binary_op::<bls::Scalar, { bls::Scalar::BYTE_LENGTH }>(|a, b| b / a, &e1, &e2)
}
Some(Groups::BLS12381G1) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_g1_div_cost);
binary_op_diff::<
bls::Scalar,
bls::G1Element,
{ bls::Scalar::BYTE_LENGTH },
{ bls::G1Element::BYTE_LENGTH },
>(|a, b| b / a, &e1, &e2)
}
Some(Groups::BLS12381G2) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_g2_div_cost);
binary_op_diff::<
bls::Scalar,
bls::G2Element,
{ bls::Scalar::BYTE_LENGTH },
{ bls::G2Element::BYTE_LENGTH },
>(|a, b| b / a, &e1, &e2)
}
Some(Groups::BLS12381GT) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_gt_div_cost);
binary_op_diff::<
bls::Scalar,
bls::GTElement,
{ bls::Scalar::BYTE_LENGTH },
{ bls::GTElement::BYTE_LENGTH },
>(|a, b| b / a, &e1, &e2)
}
_ => Err(FastCryptoError::InvalidInput),
};
map_op_result(context, cost, result)
}
pub fn internal_hash_to(
context: &mut NativeContext,
ty_args: Vec<Type>,
mut args: VecDeque<Value>,
) -> PartialVMResult<NativeResult> {
debug_assert!(ty_args.is_empty());
debug_assert!(args.len() == 2);
let cost = context.gas_used();
if !is_supported(context)? {
return Ok(NativeResult::err(cost, NOT_SUPPORTED_ERROR));
}
let m_ref = pop_arg!(args, VectorRef);
let m = m_ref.as_bytes_ref();
let group_type = pop_arg!(args, u8);
if m.is_empty() {
return Ok(NativeResult::err(cost, INVALID_INPUT_ERROR));
}
let cost_params = &context
.extensions()
.get::<NativesCostTable>()?
.group_ops_cost_params
.clone();
let result = match Groups::from_u8(group_type) {
Some(Groups::BLS12381G1) => {
native_charge_gas_early_exit_option!(
context,
cost_params
.bls12381_g1_hash_to_base_cost
.and_then(|base_cost| cost_params
.bls12381_g1_hash_to_cost_per_byte
.map(|per_byte| base_cost + per_byte * (m.len() as u64).into()))
);
Ok(bls::G1Element::hash_to_group_element(&m)
.to_byte_array()
.to_vec())
}
Some(Groups::BLS12381G2) => {
native_charge_gas_early_exit_option!(
context,
cost_params
.bls12381_g2_hash_to_base_cost
.and_then(|base_cost| cost_params
.bls12381_g2_hash_to_cost_per_byte
.map(|per_byte| base_cost + per_byte * (m.len() as u64).into()))
);
Ok(bls::G2Element::hash_to_group_element(&m)
.to_byte_array()
.to_vec())
}
_ => Err(FastCryptoError::InvalidInput),
};
map_op_result(context, cost, result)
}
fn msm_num_of_additions(n: u64) -> u64 {
debug_assert!(n > 0);
let wbits = (64 - n.leading_zeros() - 1) as u64;
let window_size = match wbits {
0 => 1,
1..=4 => 2,
5..=12 => wbits - 2,
_ => wbits - 3,
};
let num_of_windows = 255 / window_size + if 255 % window_size == 0 { 0 } else { 1 };
num_of_windows * (n + (1 << window_size) + 1)
}
#[test]
fn test_msm_factor() {
assert_eq!(msm_num_of_additions(1), 1020);
assert_eq!(msm_num_of_additions(2), 896);
assert_eq!(msm_num_of_additions(3), 1024);
assert_eq!(msm_num_of_additions(4), 1152);
assert_eq!(msm_num_of_additions(32), 3485);
}
fn multi_scalar_mul<G, const SCALAR_SIZE: usize, const POINT_SIZE: usize>(
context: &mut NativeContext,
scalar_decode_cost: Option<InternalGas>,
point_decode_cost: Option<InternalGas>,
base_cost: Option<InternalGas>,
base_cost_per_addition: Option<InternalGas>,
max_len: u32,
scalars: &[u8],
points: &[u8],
) -> PartialVMResult<NativeResult>
where
G: GroupElement
+ ToFromByteArray<POINT_SIZE>
+ FromTrustedByteArray<POINT_SIZE>
+ MultiScalarMul,
G::ScalarType: ToFromByteArray<SCALAR_SIZE> + FromTrustedByteArray<SCALAR_SIZE>,
{
if points.is_empty()
|| scalars.is_empty()
|| scalars.len() % SCALAR_SIZE != 0
|| points.len() % POINT_SIZE != 0
|| points.len() / POINT_SIZE != scalars.len() / SCALAR_SIZE
{
return Ok(NativeResult::err(context.gas_used(), INVALID_INPUT_ERROR));
}
if points.len() / POINT_SIZE > max_len as usize {
return Ok(NativeResult::err(context.gas_used(), INPUT_TOO_LONG_ERROR));
}
native_charge_gas_early_exit_option!(
context,
scalar_decode_cost.map(|cost| cost * ((scalars.len() / SCALAR_SIZE) as u64).into())
);
let scalars = scalars
.chunks(SCALAR_SIZE)
.map(parse_trusted::<G::ScalarType, { SCALAR_SIZE }>)
.collect::<Result<Vec<_>, _>>();
native_charge_gas_early_exit_option!(
context,
point_decode_cost.map(|cost| cost * ((points.len() / POINT_SIZE) as u64).into())
);
let points = points
.chunks(POINT_SIZE)
.map(parse_trusted::<G, { POINT_SIZE }>)
.collect::<Result<Vec<_>, _>>();
if let (Ok(scalars), Ok(points)) = (scalars, points) {
let num_of_additions = msm_num_of_additions(scalars.len() as u64);
native_charge_gas_early_exit_option!(
context,
base_cost.and_then(|base| base_cost_per_addition
.map(|per_addition| base + per_addition * num_of_additions.into()))
);
let r = G::multi_scalar_mul(&scalars, &points)
.expect("Already checked the lengths of the vectors");
Ok(NativeResult::ok(
context.gas_used(),
smallvec![Value::vector_u8(r.to_byte_array().to_vec())],
))
} else {
Ok(NativeResult::err(context.gas_used(), INVALID_INPUT_ERROR))
}
}
pub fn internal_multi_scalar_mul(
context: &mut NativeContext,
ty_args: Vec<Type>,
mut args: VecDeque<Value>,
) -> PartialVMResult<NativeResult> {
debug_assert!(ty_args.is_empty());
debug_assert!(args.len() == 3);
let cost = context.gas_used();
if !is_msm_supported(context)? {
return Ok(NativeResult::err(cost, NOT_SUPPORTED_ERROR));
}
let elements_ref = pop_arg!(args, VectorRef);
let elements = elements_ref.as_bytes_ref();
let scalars_ref = pop_arg!(args, VectorRef);
let scalars = scalars_ref.as_bytes_ref();
let group_type = pop_arg!(args, u8);
let cost_params = &context
.extensions()
.get::<NativesCostTable>()?
.group_ops_cost_params
.clone();
let max_len = cost_params.bls12381_msm_max_len.ok_or_else(|| {
PartialVMError::new(StatusCode::UNKNOWN_INVARIANT_VIOLATION_ERROR)
.with_message("Max len for MSM is not set".to_string())
})?;
match Groups::from_u8(group_type) {
Some(Groups::BLS12381G1) => multi_scalar_mul::<
bls::G1Element,
{ bls::Scalar::BYTE_LENGTH },
{ bls::G1Element::BYTE_LENGTH },
>(
context,
cost_params.bls12381_decode_scalar_cost,
cost_params.bls12381_decode_g1_cost,
cost_params.bls12381_g1_msm_base_cost,
cost_params.bls12381_g1_msm_base_cost_per_input,
max_len,
scalars.as_ref(),
elements.as_ref(),
),
Some(Groups::BLS12381G2) => multi_scalar_mul::<
bls::G2Element,
{ bls::Scalar::BYTE_LENGTH },
{ bls::G2Element::BYTE_LENGTH },
>(
context,
cost_params.bls12381_decode_scalar_cost,
cost_params.bls12381_decode_g2_cost,
cost_params.bls12381_g2_msm_base_cost,
cost_params.bls12381_g2_msm_base_cost_per_input,
max_len,
scalars.as_ref(),
elements.as_ref(),
),
_ => Ok(NativeResult::err(
v2_native_charge(context, cost)?,
INVALID_INPUT_ERROR,
)),
}
}
pub fn internal_pairing(
context: &mut NativeContext,
ty_args: Vec<Type>,
mut args: VecDeque<Value>,
) -> PartialVMResult<NativeResult> {
debug_assert!(ty_args.is_empty());
debug_assert!(args.len() == 3);
let cost = context.gas_used();
if !is_supported(context)? {
return Ok(NativeResult::err(cost, NOT_SUPPORTED_ERROR));
}
let e2_ref = pop_arg!(args, VectorRef);
let e2 = e2_ref.as_bytes_ref();
let e1_ref = pop_arg!(args, VectorRef);
let e1 = e1_ref.as_bytes_ref();
let group_type = pop_arg!(args, u8);
let cost_params = &context
.extensions()
.get::<NativesCostTable>()?
.group_ops_cost_params
.clone();
let result = match Groups::from_u8(group_type) {
Some(Groups::BLS12381G1) => {
native_charge_gas_early_exit_option!(context, cost_params.bls12381_pairing_cost);
parse_trusted::<bls::G1Element, { bls::G1Element::BYTE_LENGTH }>(&e1).and_then(|e1| {
parse_trusted::<bls::G2Element, { bls::G2Element::BYTE_LENGTH }>(&e2).map(|e2| {
let e3 = e1.pairing(&e2);
e3.to_byte_array().to_vec()
})
})
}
_ => Err(FastCryptoError::InvalidInput),
};
map_op_result(context, cost, result)
}
pub fn internal_convert(
context: &mut NativeContext,
ty_args: Vec<Type>,
mut args: VecDeque<Value>,
) -> PartialVMResult<NativeResult> {
debug_assert!(ty_args.is_empty());
debug_assert!(args.len() == 3);
let cost = context.gas_used();
if !(is_uncompressed_g1_supported(context))? {
return Ok(NativeResult::err(cost, NOT_SUPPORTED_ERROR));
}
let e_ref = pop_arg!(args, VectorRef);
let e = e_ref.as_bytes_ref();
let to_type = pop_arg!(args, u8);
let from_type = pop_arg!(args, u8);
let cost_params = &context
.extensions()
.get::<NativesCostTable>()?
.group_ops_cost_params
.clone();
let result = match (Groups::from_u8(from_type), Groups::from_u8(to_type)) {
(Some(Groups::BLS12381UncompressedG1), Some(Groups::BLS12381G1)) => {
native_charge_gas_early_exit_option!(
context,
cost_params.bls12381_uncompressed_g1_to_g1_cost
);
e.to_vec()
.try_into()
.map_err(|_| FastCryptoError::InvalidInput)
.map(bls::G1ElementUncompressed::from_trusted_byte_array)
.and_then(|e| bls::G1Element::try_from(&e))
.map(|e| e.to_byte_array().to_vec())
}
(Some(Groups::BLS12381G1), Some(Groups::BLS12381UncompressedG1)) => {
native_charge_gas_early_exit_option!(
context,
cost_params.bls12381_g1_to_uncompressed_g1_cost
);
parse_trusted::<bls::G1Element, { bls::G1Element::BYTE_LENGTH }>(&e)
.map(|e| bls::G1ElementUncompressed::from(&e))
.map(|e| e.into_byte_array().to_vec())
}
_ => Err(FastCryptoError::InvalidInput),
};
map_op_result(context, cost, result)
}
pub fn internal_sum(
context: &mut NativeContext,
ty_args: Vec<Type>,
mut args: VecDeque<Value>,
) -> PartialVMResult<NativeResult> {
debug_assert!(ty_args.is_empty());
debug_assert!(args.len() == 2);
let cost = context.gas_used();
if !(is_uncompressed_g1_supported(context))? {
return Ok(NativeResult::err(cost, NOT_SUPPORTED_ERROR));
}
let cost_params = &context
.extensions()
.get::<NativesCostTable>()?
.group_ops_cost_params
.clone();
let inputs = pop_arg!(args, VectorRef);
let group_type = pop_arg!(args, u8);
let length = inputs
.len(&Type::Vector(Box::new(Type::U8)))?
.value_as::<u64>()?;
let result = match Groups::from_u8(group_type) {
Some(Groups::BLS12381UncompressedG1) => {
let max_terms = cost_params
.bls12381_uncompressed_g1_sum_max_terms
.ok_or_else(|| {
PartialVMError::new(StatusCode::UNKNOWN_INVARIANT_VIOLATION_ERROR)
.with_message("Max number of terms is not set".to_string())
})?;
if length > max_terms {
return Ok(NativeResult::err(cost, INPUT_TOO_LONG_ERROR));
}
native_charge_gas_early_exit_option!(
context,
cost_params
.bls12381_uncompressed_g1_sum_base_cost
.and_then(|base| cost_params
.bls12381_uncompressed_g1_sum_cost_per_term
.map(|per_term| base + per_term * length.into()))
);
(0..length)
.map(|i| {
inputs
.borrow_elem(i as usize, &Type::Vector(Box::new(Type::U8)))
.and_then(Value::value_as::<VectorRef>)
.map_err(|_| FastCryptoError::InvalidInput)
.and_then(|v| {
v.as_bytes_ref()
.to_vec()
.try_into()
.map_err(|_| FastCryptoError::InvalidInput)
})
.map(bls::G1ElementUncompressed::from_trusted_byte_array)
})
.collect::<FastCryptoResult<Vec<_>>>()
.and_then(|e| bls::G1ElementUncompressed::sum(&e))
.map(|e| bls::G1ElementUncompressed::from(&e))
.map(|e| e.into_byte_array().to_vec())
}
_ => Err(FastCryptoError::InvalidInput),
};
map_op_result(context, cost, result)
}