mirror of
https://github.com/pezkuwichain/revive.git
synced 2026-04-29 06:38:00 +00:00
dab29bc89b
Found in https://github.com/paritytech/contract-issues/issues/45, thanks @albertov19 and @sekisamu --------- Signed-off-by: xermicus <cyrill@parity.io> Signed-off-by: Cyrill Leutwiler <bigcyrill@hotmail.com>
177 lines
6.3 KiB
Rust
177 lines
6.3 KiB
Rust
//! Translates the contract creation instructions.
|
|
|
|
use inkwell::values::BasicValue;
|
|
use num::Zero;
|
|
|
|
use crate::polkavm::context::argument::Argument;
|
|
use crate::polkavm::context::code_type::CodeType;
|
|
use crate::polkavm::context::Context;
|
|
use crate::polkavm::Dependency;
|
|
|
|
/// Translates the contract `create` and `create2` instruction.
|
|
///
|
|
/// A `salt` value of `None` is equivalent to `create1`.
|
|
pub fn create<'ctx, D>(
|
|
context: &mut Context<'ctx, D>,
|
|
value: inkwell::values::IntValue<'ctx>,
|
|
input_offset: inkwell::values::IntValue<'ctx>,
|
|
input_length: inkwell::values::IntValue<'ctx>,
|
|
salt: Option<inkwell::values::IntValue<'ctx>>,
|
|
) -> anyhow::Result<inkwell::values::BasicValueEnum<'ctx>>
|
|
where
|
|
D: Dependency + Clone,
|
|
{
|
|
let input_offset = context.safe_truncate_int_to_xlen(input_offset)?;
|
|
let input_length = context.safe_truncate_int_to_xlen(input_length)?;
|
|
|
|
let code_hash_pointer = context.build_heap_gep(input_offset, input_length)?;
|
|
|
|
let value_pointer = context.build_alloca_at_entry(context.value_type(), "transferred_value");
|
|
context.build_store(value_pointer, value)?;
|
|
|
|
let salt_pointer = match salt {
|
|
Some(salt) => {
|
|
let salt_pointer = context.build_alloca_at_entry(context.word_type(), "salt_pointer");
|
|
let salt = context.build_byte_swap(salt.into())?;
|
|
context.build_store(salt_pointer, salt)?;
|
|
salt_pointer
|
|
}
|
|
None => context.sentinel_pointer(),
|
|
};
|
|
|
|
let address_pointer = context.build_alloca_at_entry(
|
|
context.integer_type(revive_common::BIT_LENGTH_ETH_ADDRESS),
|
|
"address_pointer",
|
|
);
|
|
context.build_store(address_pointer, context.word_const(0))?;
|
|
|
|
let deposit_pointer = context.build_alloca_at_entry(context.word_type(), "deposit_pointer");
|
|
context.build_store(deposit_pointer, context.word_type().const_all_ones())?;
|
|
|
|
let deposit_and_value = revive_runtime_api::calling_convention::pack_hi_lo_reg(
|
|
context.builder(),
|
|
context.llvm(),
|
|
deposit_pointer.to_int(context),
|
|
value_pointer.to_int(context),
|
|
"deposit_and_value",
|
|
)?;
|
|
let input_data = revive_runtime_api::calling_convention::pack_hi_lo_reg(
|
|
context.builder(),
|
|
context.llvm(),
|
|
input_length,
|
|
code_hash_pointer.to_int(context),
|
|
"input_data",
|
|
)?;
|
|
let address_and_salt = revive_runtime_api::calling_convention::pack_hi_lo_reg(
|
|
context.builder(),
|
|
context.llvm(),
|
|
address_pointer.to_int(context),
|
|
salt_pointer.to_int(context),
|
|
"output_data",
|
|
)?;
|
|
|
|
context.build_runtime_call(
|
|
revive_runtime_api::polkavm_imports::INSTANTIATE,
|
|
&[
|
|
context.register_type().const_all_ones().into(),
|
|
context.register_type().const_all_ones().into(),
|
|
deposit_and_value.into(),
|
|
input_data.into(),
|
|
context.register_type().const_all_ones().into(),
|
|
address_and_salt.into(),
|
|
],
|
|
);
|
|
|
|
let address = context.build_byte_swap(context.build_load(address_pointer, "address")?)?;
|
|
Ok(context
|
|
.builder()
|
|
.build_int_z_extend(
|
|
address.into_int_value(),
|
|
context.word_type(),
|
|
"address_zext",
|
|
)?
|
|
.into())
|
|
}
|
|
|
|
/// Translates the contract hash instruction, which is actually used to set the hash of the contract
|
|
/// being created, or other related auxiliary data.
|
|
/// Represents `dataoffset` in Yul and `PUSH [$]` in the EVM legacy assembly.
|
|
pub fn contract_hash<'ctx, D>(
|
|
context: &mut Context<'ctx, D>,
|
|
identifier: String,
|
|
) -> anyhow::Result<Argument<'ctx>>
|
|
where
|
|
D: Dependency + Clone,
|
|
{
|
|
let code_type = context
|
|
.code_type()
|
|
.ok_or_else(|| anyhow::anyhow!("The contract code part type is undefined"))?;
|
|
|
|
let parent = context.module().get_name().to_str().expect("Always valid");
|
|
|
|
let contract_path =
|
|
context
|
|
.resolve_path(identifier.as_str())
|
|
.map_err(|error| match code_type {
|
|
CodeType::Runtime if identifier.ends_with("_deployed") => {
|
|
anyhow::anyhow!("type({}).runtimeCode is not supported", identifier)
|
|
}
|
|
_ => error,
|
|
})?;
|
|
if contract_path.as_str() == parent {
|
|
return Ok(Argument::new_with_constant(
|
|
context.word_const(0).as_basic_value_enum(),
|
|
num::BigUint::zero(),
|
|
));
|
|
} else if identifier.ends_with("_deployed") && code_type == CodeType::Runtime {
|
|
anyhow::bail!("type({}).runtimeCode is not supported", identifier);
|
|
}
|
|
|
|
let hash_string = context.compile_dependency(identifier.as_str())?;
|
|
let hash_value = context
|
|
.word_const_str_hex(hash_string.as_str())
|
|
.as_basic_value_enum();
|
|
Ok(Argument::new_with_original(hash_value, hash_string))
|
|
}
|
|
|
|
/// Translates the deploy call header size instruction. the header consists of
|
|
/// the hash of the bytecode of the contract whose instance is being created.
|
|
/// Represents `datasize` in Yul and `PUSH #[$]` in the EVM legacy assembly.
|
|
pub fn header_size<'ctx, D>(
|
|
context: &mut Context<'ctx, D>,
|
|
identifier: String,
|
|
) -> anyhow::Result<Argument<'ctx>>
|
|
where
|
|
D: Dependency + Clone,
|
|
{
|
|
let code_type = context
|
|
.code_type()
|
|
.ok_or_else(|| anyhow::anyhow!("The contract code part type is undefined"))?;
|
|
|
|
let parent = context.module().get_name().to_str().expect("Always valid");
|
|
|
|
let contract_path =
|
|
context
|
|
.resolve_path(identifier.as_str())
|
|
.map_err(|error| match code_type {
|
|
CodeType::Runtime if identifier.ends_with("_deployed") => {
|
|
anyhow::anyhow!("type({}).runtimeCode is not supported", identifier)
|
|
}
|
|
_ => error,
|
|
})?;
|
|
if contract_path.as_str() == parent {
|
|
return Ok(Argument::new_with_constant(
|
|
context.word_const(0).as_basic_value_enum(),
|
|
num::BigUint::zero(),
|
|
));
|
|
} else if identifier.ends_with("_deployed") && code_type == CodeType::Runtime {
|
|
anyhow::bail!("type({}).runtimeCode is not supported", identifier);
|
|
}
|
|
|
|
let size_bigint = num::BigUint::from(crate::polkavm::DEPLOYER_CALL_HEADER_SIZE);
|
|
let size_value = context
|
|
.word_const(crate::polkavm::DEPLOYER_CALL_HEADER_SIZE as u64)
|
|
.as_basic_value_enum();
|
|
Ok(Argument::new_with_constant(size_value, size_bigint))
|
|
}
|