Skip to main content
Version: 3.3.0

MSM - Multi Scalar Multiplication

Overview

Multi-Scalar Multiplication (MSM) is a fundamental operation in elliptic curve cryptography and zero-knowledge proofs. It is defined as:

MSM(a,G)=j=0n1ajGjMSM(a, G) = \sum_{j=0}^{n-1} a_j \cdot G_j

Where:

  • GjGG_j \in G are points from an elliptic curve group.
  • a0,,ana_0, \ldots, a_n are scalars.
  • MSM(a,G)GMSM(a, G) \in G is the result, a single elliptic curve point.

MSM is inherently parallelizable, making it a critical operation for optimizing performance in cryptographic protocols like zk-SNARKs. Accelerating MSM can significantly reduce the time required for proof generation.

Accelerating MSM is crucial to a ZK protocol's performance due to the large percent of run time they take when generating proofs. You can learn more about how MSMs work from this video and from our resource list on Ingopedia.

C++ API

MSMConfig Struct

The MSMConfig struct configures the MSM operation. It allows customization of parameters like the number of precomputed points, the window bitsize (c), and memory management. Here's the configuration structure:

struct MSMConfig {
icicleStreamHandle stream;
int precompute_factor;
int c;
int bitsize;
int batch_size;
bool are_points_shared_in_batch;
bool are_scalars_on_device;
bool are_scalars_montgomery_form;
bool are_points_on_device;
bool are_points_montgomery_form;
bool are_results_on_device;
bool is_async;
ConfigExtension* ext;
};

Default Configuration

You can obtain a default MSMConfig using:

  static MSMConfig default_msm_config()
{
MSMConfig config = {
nullptr, // stream
1, // precompute_factor
0, // c
0, // bitsize
1, // batch_size
true, // are_points_shared_in_batch
false, // are_scalars_on_device
false, // are_scalars_montgomery_form
false, // are_points_on_device
false, // are_points_montgomery_form
false, // are_results_on_device
false, // is_async
nullptr, // ext
};
return config;
}

msm Function

The msm function computes the MSM operation:

template <typename S, typename A, typename P>
eIcicleError msm(const S* scalars, const A* bases, int msm_size, const MSMConfig& config, P* results);
note

The API is template and can work with all ICICLE curves (if corresponding lib is linked), including G2 groups.

Batched MSM

The MSM supports batch mode - running multiple MSMs in parallel. It's always better to use the batch mode instead of running single msms in serial as long as there is enough memory available. We support running a batch of MSMs that share the same points as well as a batch of MSMs that use different points.

Config fields are_points_shared_in_batch and batch_size are used to configure msm for batch mode.

G2 MSM

for G2 MSM, use the same msm api with the G2 types.

note

Supported curves have types for both G1 and G2.

Precompution

What It Does:

  • The function computes a set of additional points derived from the original base points. These precomputed points are stored and later reused during the MSM computation.
  • Purpose: By precomputing and storing these points, the MSM operation can reduce the number of operations needed at runtime, which can significantly speed up the calculation.

When to Use:

  • Memory vs. Speed Trade-off: Precomputation increases the memory footprint because additional points are stored, but it reduces the computational effort during MSM, making the process faster.
  • Best for Repeated Use: It’s especially beneficial when the same set of base points is used multiple times in different MSM operations.
template <typename A>
eIcicleError msm_precompute_bases(const A* input_bases, int bases_size, const MSMConfig& config, A* output_bases);
note

User is allocating the output_bases (on host or device memory) and later use it as bases when calling msm.

Rust and Go bindings

The Rust and Go bindings provide equivalent functionality for their respective environments. Refer to their documentation for details on usage.

CUDA backend MSM

This section describes the CUDA msm implementation and how to customize it (optional).

Algorithm description

We follow the bucket method algorithm. The GPU implementation consists of four phases:

  1. Preparation phase - The scalars are split into smaller scalars of c bits each. These are the bucket indices. The points are grouped according to their corresponding bucket index and the buckets are sorted by size.
  2. Accumulation phase - Each bucket accumulates all of its points using a single thread. More than one thread is assigned to large buckets, in proportion to their size. A bucket is considered large if its size is above the large bucket threshold that is determined by the large_bucket_factor parameter. The large bucket threshold is the expected average bucket size times the large_bucket_factor parameter.
  3. Buckets Reduction phase - bucket results are multiplied by their corresponding bucket number and each bucket module is reduced to a small number of final results. By default, this is done by an iterative algorithm which is highly parallel. Setting is_big_triangle to true will switch this phase to the running sum algorithm described in the above YouTube talk which is much less parallel.
  4. Final accumulation phase - The final results from the last phase are accumulated using the double-and-add algorithm.

Configuring CUDA msm

Use ConfigExtension object to pass backend specific configuration. CUDA specific msm configuration:

ConfigExtension ext;
ext.set("large_bucket_factor", 15);
// use the config-extension in the msm config for the backend to see.
msm_config.ext = &ext;
// call msm
msm(..., config,...); // msm backend is reading the config-extension

Choosing optimal parameters

is_big_triangle should be false in almost all cases. It might provide better results only for very small MSMs (smaller than 2^8) with a large batch (larger than 100) but this should be tested per scenario. Large buckets exist in two cases:

  1. When the scalar distribution isn't uniform.
  2. When c does not divide the scalar bit-size.

large_bucket_factor that is equal to 10 yields good results for most cases, but it's best to fine tune this parameter per c and per scalar distribution. The two most important parameters for performance are c and the precompute_factor. They affect the number of EC additions as well as the memory size. When the points are not known in advance we cannot use precomputation. In this case the best c value is usually around log2(msmSize)4log_2(msmSize) - 4. However, in most protocols the points are known in advanced and precomputation can be used unless limited by memory. Usually it's best to use maximum precomputation (such that we end up with only a single bucket module) combined we a c value around log2(msmSize)1log_2(msmSize) - 1.

Memory usage estimation

The main memory requirements of the MSM are the following:

  • Scalars - sizeof(scalar_t) * msm_size * batch_size
  • Scalar indices - ~6 * sizeof(unsigned) * nof_bucket_modules * msm_size * batch_size
  • Points - sizeof(affine_t) * msm_size * precomp_factor * batch_size
  • Buckets - sizeof(projective_t) * nof_bucket_modules * 2^c * batch_size

where nof_bucket_modules = ceil(ceil(bitsize / c) / precompute_factor)

During the MSM computation first the memory for scalars and scalar indices is allocated, then the indices are freed and points and buckets are allocated. This is why a good estimation for the required memory is the following formula:

max(scalars+scalarIndices,scalars+points+buckets)max(scalars + scalarIndices, scalars + points + buckets)

This gives a good approximation within 10% of the actual required memory for most cases.

Example parameters

Here is a useful table showing optimal parameters for different MSMs. They are optimal for BLS12-377 curve when running on NVIDIA GeForce RTX 3090 Ti. This is the configuration used:

Here are the parameters and the results for the different cases:

MSM sizeBatch sizePrecompute factorcMemory estimation (GB)Actual memory (GB)Single MSM time (ms)
101190.002270.002779.2
10123110.002590.002721.76
101000170.941.090.051
10100023112.592.740.025
1511110.0110.0199.9
15116160.0610.0652.4
151001111.911.920.84
1510019146.326.610.56
1811140.1280.12814.4
18115170.400.425.9
2211171.641.6568
22113215.675.9454
2411186.586.61232
24172112.413.4199

The optimal values can vary per GPU and per curve. It is best to try a few combinations until you get the best results for your specific case.