This guide explains how to estimate gas costs for Fully Homomorphic Encryption (FHE) operations in your smart contracts on Zama's fhEVM. Understanding gas consumption is critical for designing efficient confidential smart contracts.
FHE operations in fhEVM are computationally intensive, resulting in higher gas costs compared to standard Ethereum operations. This is due to the complex mathematical operations required to ensure privacy and security.
Native Gas:
Standard gas used for operations on the underlying EVM chain.
On fhEVM, native gas consumption is approximately 20% higher than in mocked environments.
FHEGas:
Represents gas consumed by FHE-specific computations.
A new synthetic kind of gas consumed by FHE-specific computations.
FHEGas is tracked in each block by the FHEPayment contract to prevent DDOS attacks.
If too many FHE operations are requested in the same block, the transaction will revert once the FHEGas block limit is reached.
FHEGas is consistent across both mocked and real fhEVM environments.
Note: Gas values provided are approximate and may vary based on network conditions, implementation details, and contract complexity.
To monitor gas usage during development, use the following tools:
getFHEGasFromTxReceipt
:
Extracts FHEGas consumption from a transaction receipt.
Works only in mocked fhEVM environments, but gives the exact same value as in non-mocked environments.
Import as: import { getFHEGasFromTxReceipt } from "../coprocessorUtils";
.gasUsed
from ethers.js transaction receipt:
Standard ethers.js transaction receipt property that returns the native gas used.
In mocked mode, this value underestimates real native gas usage by ~20%.
Works in both mocked and real fhEVM environments, as it's a standard Ethereum transaction property.
The following code demonstrates how to measure both FHEGas and native gas during a transaction:
The current devnet has a FHEGas limit of 10,000,000 per block. Here's what you need to know:
If you send a transaction that exceeds this limit or if the FHEGas block limit is exceeded, depending on other previous transaction in same block:
The transaction will revert
Any native gas fees (but not FHEGas) will still be charged
You should do one of the following:
Reduce the number of FHE operations in your transaction
Wait for the next block when the FHEGas limit resets
Split your operations across multiple transactions
ebool
)and
/or
/xor
26,000
not
30,000
Gas costs increase with the bit-width of the encrypted integer type. Below are the detailed costs for various operations on encrypted types.
euint4
)add
/sub
65,000
add
/sub
(scalar)
65,000
mul
150,000
mul
(scalar)
88,000
div
(scalar)
139,000
rem
(scalar)
286,000
and
/or
/xor
32,000
shr
/shl
116,000
shr
/shl
(scalar)
35,000
rotr
/rotl
116,000
rotr
/rotl
(scalar)
35,000
eq
/ne
51,000
ge
/gt
/le
/lt
70,000
min
/max
121,000
min
/max
(scalar)
121,000
neg
60,000
not
33,000
select
45,000
euint8
)add
/sub
94,000
add
/sub
(scalar)
94,000
mul
197,000
mul
(scalar)
159,000
div
(scalar)
238,000
rem
(scalar)
460,000
and
/or
/xor
34,000
shr
/shl
133,000
shr
/shl
(scalar)
35,000
rotr
/rotl
133,000
rotr
/rotl
(scalar)
35,000
eq
/ne
53,000
ge
/gt
/le
/lt
82,000
min
/max
128,000
min
/max
(scalar)
128,000
neg
95,000
not
34,000
select
47,000
randEuint8()
100,000
euint16
)add
/sub
133,000
add
/sub
(scalar)
133,000
mul
262,000
mul
(scalar)
208,000
div
(scalar)
314,000
rem
(scalar)
622,000
and
/or
/xor
34,000
shr
/shl
153,000
shr
/shl
(scalar)
35,000
rotr
/rotl
153,000
rotr
/rotl
(scalar)
35,000
eq
/ne
54,000
ge
/gt
/le
/lt
105,000
min
/max
153,000
min
/max
(scalar)
150,000
neg
131,000
not
35,000
select
47,000
randEuint16()
100,000
euint32
)add
/sub
162,000
add
/sub
(scalar)
162,000
mul
359,000
mul
(scalar)
264,000
div
(scalar)
398,000
rem
(scalar)
805,000
and
/or
/xor
35,000
shr
/shl
183,000
shr
/shl
(scalar)
35,000
rotr
/rotl
183,000
rotr
/rotl
(scalar)
35,000
eq
/ne
82,000
ge
/gt
/le
/lt
128,000
min
/max
183,000
min
/max
(scalar)
164,000
neg
160,000
not
36,000
select
50,000
randEuint32()
100,000
euint64
)add
/sub
188,000
add
/sub
(scalar)
188,000
mul
641,000
mul
(scalar)
356,000
div
(scalar)
584,000
rem
(scalar)
1,095,000
and
/or
/xor
38,000
shr
/shl
227,000
shr
/shl
(scalar)
38,000
rotr
/rotl
227,000
rotr
/rotl
(scalar)
38,000
eq
/ne
86,000
ge
/gt
/le
/lt
156,000
min
/max
210,000
min
/max
(scalar)
192,000
neg
199,000
not
37,000
select
53,000
randEuint64()
100,000
euint128
)add
/sub
218,000
add
/sub
(scalar)
218,000
mul
1,145,000
mul
(scalar)
480,000
div
(scalar)
857,000
rem
(scalar)
1,499,000
and
/or
/xor
41,000
shr
/shl
282,000
shr
/shl
(scalar)
41,000
rotr
/rotl
282,000
rotr
/rotl
(scalar)
41,000
eq
/ne
88,000
ge
/gt
/le
/lt
190,000
min
/max
241,000
min
/max
(scalar)
225,000
neg
248,000
not
38,000
select
70,000
euint256
)add
/sub
253,000
add
/sub
(scalar)
253,000
mul
2,045,000
mul
(scalar)
647,000
div
(scalar)
1,258,000
rem
(scalar)
2,052,000
and
/or
/xor
44,000
shr
/shl
350,000
shr
/shl
(scalar)
44,000
rotr
/rotl
350,000
rotr
/rotl
(scalar)
44,000
eq
/ne
100,000
ge
/gt
/le
/lt
231,000
min
/max
277,000
min
/max
(scalar)
264,000
neg
309,000
not
39,000
select
90,000
eq
/ne
90,000
cast
200
trivialEncrypt
(basic)
100-800
trivialEncrypt
(extended)
1,600-6,400
randBounded
100,000
select
43,000-300,000
rand
100,000-400,000
To resolve a failed transaction due to gas limits:
Open MetaMask and go to Settings
Navigate to Advanced Settings
Enable "Customize transaction nonce"
When resending the transaction:
Use the same nonce as the failed transaction
Set an appropriate gas limit under 10M
Adjust other parameters as needed
This allows you to "replace" the failed transaction with a valid one using the correct gas parameters.