当前位置: 首页 > news >正文

Revive 中的 Precompile 合约:实现与调用机制

图片

Polkadot 在 2.0 里面引入了新的 PolkaVM 来支持智能合约的运行,并且使用Revive Pallet 兼容 EVM。通过 Resolc 的编译,solidity 代码可以在 PolkaVM 上更加高效的运行。

在一般的 EVM 执行环境下,precompile 都是一个不可缺少的部分,能够提供一些通用的功能,例如 ecrecover 方法。比如在 Moonbeam中,precompile 的合约可以参考这个文档:Canonical Contract Addresses on Moonbeam(🔗: https://docs.moonbeam.network/builders/ethereum/canonical-contracts/#ethereum-mainnet-precompiles)。Revive Pallet 同样提供了一些 precompile 的合约给开发者使用。

01

代码分析

我们在 polkadot sdk 的代码库中,打开 precompiles(🔗: https://github.com/paritytech/polkadot-sdk/tree/master/substrate/frame/revive/src/pure_precompiles) 这个目录中,我们可以看的一些 precompile 合约的源代码。以 sha256 作为参考,它的代码实现非常简单,需要实现 Precompile 这个 trait。二个参数分别是对使用过的 gas 存贮的对象和一个 u8 类型的数组。它只需要通过调用 sp_io 里面的函数,然后返回就可以了。

/// The Sha256 precompile.pub struct Sha256;impl<T: Config> Precompile<T> for Sha256 {fn execute(gas_meter: &mut GasMeter<T>, input: &[u8]) -> Result<ExecReturnValue, &'static str> {		gas_meter.charge(RuntimeCosts::HashSha256(input.len() as u32))?;		let data = sp_io::hashing::sha2_256(input).to_vec();Ok(ExecReturnValue { data, flags: ReturnFlags::empty() })	}}

对这些 precompile 的调用入口在 pure precompile.rs (🔗 https://github.com/paritytech/polkadot-sdk/blob/master/substrate/frame/revive/src/pure_precompiles.rs#L59) 源文件里面,execute 方法通过合约地址的最后一个字节来找到需要调用的合约。可以看到 Sha256 的编号是 2,而 ECRecover 的是 1.

图片

为什么只比较最后一个字节,因为我们把前 19 个字节都是 0 的分配给了 precompile 地址空间。

pub fn is_precompile(address: &H160) -> bool {    let bytes = address.as_bytes();    bytes.starts_with(&[0u8; 19]) && bytes[19] != 0 }

在运行到合约执行的时候,runtime 总是先通过地址来判断是否为 precompile 的合约。这个判断逻辑在 call (🔗: https://github.com/paritytech/polkadot-sdk/blob/master/substrate/frame/revive/src/exec.rs#L1485) 和 delegate call (🔗:https://github.com/paritytech/polkadot-sdk/blob/master/substrate/frame/revive/src/exec.rs#L1564) 最开始进行,如果是 precompile 合约,直接转到 Runtime 里面运行。不同版本的代码行数可以有区别,只需要搜索 is_precompile 就可以。

02

调用和验证

在理解了 precompile 以及他们的地址,我们就可以尝试着来调用他们。我们还是以 sha256 为例子,我们写一个简单的 solidity 合约,代码如下:合约非常简单,首先定义一个地址,然后通过 call 这个函数来调用 sha256 方法,这里可以输入一个 bytes 作为 input,并把它的结果放到链上存贮 result 里面。

// SPDX-License-Identifier: GPL-3.0 pragma solidity >=0.8.2 <0.9.0; contract Storage {    event CallPrecompile(bytes);    bytes result;    function callH256(bytes calldata input) public {        // address of precompile h256        address precompile = address(0x02);        // result        bool success;        bytes memory resultInMemory;        // just call it without selector        (success, resultInMemory) = precompile.call{value: 0}(input);        // emit the result        if (success) {            emit CallPrecompile(resultInMemory);        }        // put result in storage        result = resultInMemory;    }}

我们打开 polkaVM 的 remix,在浏览器中输入 https://remix.polkadot.io 

把这个合约加到一个源文件中,并编译和部署。得到地址后,调用 callH256 函数,测试数据可以从这个文件(🔗 https://github.com/paritytech/polkadot-sdk/blob/master/substrate/frame/revive/src/pure_precompiles/testdata/2-sha256.json)得到, 里面包含输入和输出的正确结果。 

运行结果截图如下:

图片

03

使用和扩展

本文简单的介绍了 revive 里面 precompile 具体实现和使用方法,大家可以关注 precompile 的变化,可能会包含你所需要的功能,这样就可以直接使用了,不需要自己在去实现。

如果你是在自己的链上引入了 revive,也可以根据需要自己添加 precompile 合约,把需要的 runtime 里面的功能暴露给智能合约来调用,或者将常用的算法,函数放在 precompile,提高运行效率,减少合约大小。

免责声明:由 PaperMoon 提供并包含在本文中的材料仅用于学习目的。它们不构成财务或投资建议,也不应被解读为任何商业决策的指导。我们建议读者在做出任何投资或商业相关的决定之前,进行独立研究并咨询专业人士。PaperMoon 对根据本文内容采取的任何行动不承担任何责任。

相关文章:

  • Jetpack Room 使用详解
  • 【多模态模型】跨模态智能的核心技术与应用实践
  • 【误差理论与可靠性工程】蒙特卡洛法计算电路可靠度和三极管静态工作点电压
  • 新增 29 个专业,科技成为关键赛道!
  • 服务器不能复制粘贴文件的处理方式
  • 前端面试高频算法
  • AI服务器与普通服务器之间的区别
  • 电商数据采集电商,行业数据分析,平台数据获取|稳定的API接口数据
  • Linux-UDP套接字编程
  • 使用 NServiceBus 在 .NET 中构建分布式系统
  • 徽客松S1 | 合肥首场 AI 黑客松招募
  • 网络安全:从入门到精通,从概念到案例的全面解析
  • 文章记单词 | 第50篇(六级)
  • python实战项目66:抓取考研招生专业信息
  • 磁盘清理git gc
  • 【Python】Matplotlib:立体永生花绘制
  • 开发一个LabVIEW软件需要多少钱
  • 跨域问题(Cross-Origin Problem)
  • 浮点数:IEEE 754标准
  • 【MySQL数据库入门到精通-08 约束】
  • 气温“过山车”现象未来或更频繁且更剧烈
  • “天链”继续上新!长三乙火箭成功发射天链二号05星
  • 中央纪委办公厅公开通报3起整治形式主义为基层减负典型问题
  • 同款瑞幸咖啡竟差了6元,开了会员仍比别人贵!客服回应
  • 龚正会见巴基斯坦卡拉奇市市长穆尔塔扎·瓦哈卜、巴西圣保罗市市长里卡多·努内斯
  • 解放军仪仗司礼大队参加越南纪念南方解放50周年庆典活动