The integration is still experimental, and some breaking changes may occur.
Hashi’s core functionality is to propagate block headers across multiple chains, enabling cross-chain state verification. On EVM-based chains, this can be done in a trustless manner because the EVM exposes the last 256 block headers. However, this approach is not feasible on Solana, as the Solana VM only provides slot hashes, which do not include commitments that would allow verification of account states (as opposed to the state_root on Ethereum).
Nevertheless, Solana’s VM does allow reading account states (such as data, address, lamports, owner, and rent_epoch). To make use of this capability, we developed a program called Snapshotter. Snapshotter allows the propagation of the hash of each subscribed account’s state to other chains via Hashi reporters.
Technically, whenever the subscribe
function is called, the hash of the specified account’s state is added to a Merkle tree. Once a predefined batch size is reached (i.e., a certain number of subscriptions have occurred), anyone can call calculate_root
to compute and store the Merkle root on-chain (AKA accounts_root
). At that point, any Hashi-compatible reporter (This is an example of a Hashi-compatible reporter) can invoke dispatch_root
to propagate the root to other blockchains, thereby synchronizing the account state across multiple ecosystems. Finally, on the destination chain, you can verify an account simply by verifying a Merkle proof.
Suppose you need to read the Solana System Program account data from Base. The first step is to let the Snapshotter know that you want the System Program account’s hash to be included in the next accounts_root
. To do this, simply call subscribe
. Full example here.
Once the number of subscribed accounts reaches BATCH_SIZE, anyone can call calculate_root
, which computes the Merkle root of the new batch and stores it so reporters can easily retrieve it when broadcasting to other chains. Because each new batch’s root (batch_accounts_root
) must be combined with the existing accounts_root
(to preserve previously included accounts), we implemented a custom “Batch Merkle Tree.” This specialized Merkle tree allows us to merge all batch_accounts_root
values into a single accounts_root
without losing any historical data. Full example here.
Once the accounts_root
is updated, you can transmit it to other chains simply by calling dispatch_root
. In this example, the Wormhole reporter is used to dispatch the root, but any compatible reporter will work. Full example here.
Once the accounts_root
is successfully stored in the corresponding adapter(s) on Base, you can read the account data by calling HashiProver.verifyForeignSolanaAccount
from any contract.
For a complete example of how to read a Solana account data, refer to this example.