1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
use fuel_asm::Instruction;
use fuel_tx::{
    field::{Salt, Script},
    Bytes32, ConsensusParameters, InputRepr,
};
use fuel_types::{bytes::padded_len_usize, ContractId};

/// Gets the base offset for a script or a predicate. The offset depends on the `max_inputs`
/// field of the `ConsensusParameters` and the static offset.
pub fn base_offset_script(consensus_parameters: &ConsensusParameters) -> usize {
    consensus_parameters.tx_offset() + fuel_tx::Script::script_offset_static()
}

/// Gets the base offset for a script or a predicate. The offset depends on the `max_inputs`
/// field of the `ConsensusParameters` and the static offset.
pub fn base_offset_create(consensus_parameters: &ConsensusParameters) -> usize {
    // The easiest way to get the offset of `fuel_tx::Create` is to get the offset of the last field
    // of `Create` -- i.e. `salt` and skip it by adding its length.
    // This should be updated if `fuel_tx::Create` ever adds more fields after `salt`.
    consensus_parameters.tx_offset() + fuel_tx::Create::salt_offset_static() + Bytes32::LEN
}

/// Calculates the length of the script based on the number of contract calls it
/// has to make and returns the offset at which the script data begins
pub fn call_script_data_offset(
    consensus_parameters: &ConsensusParameters,
    calls_instructions_len: usize,
) -> usize {
    // Instruction::SIZE is a placeholder for the RET instruction which is added later for returning
    // from the script. This doesn't happen in the predicate.
    let opcode_len = Instruction::SIZE;

    base_offset_script(consensus_parameters) + padded_len_usize(calls_instructions_len + opcode_len)
}

pub fn coin_predicate_data_offset(code_len: usize) -> usize {
    InputRepr::Coin
        .coin_predicate_offset()
        .expect("should have predicate offset")
        + padded_len_usize(code_len)
}

pub fn message_predicate_data_offset(message_data_len: usize, code_len: usize) -> usize {
    InputRepr::Message
        .data_offset()
        .expect("should have data offset")
        + padded_len_usize(message_data_len)
        + padded_len_usize(code_len)
}

pub fn coin_signed_data_offset() -> usize {
    InputRepr::Coin
        .coin_predicate_offset()
        .expect("should have coin offset")
}

pub fn message_signed_data_offset(message_data_len: usize) -> usize {
    InputRepr::Message
        .data_offset()
        .expect("should have data offset")
        + padded_len_usize(message_data_len)
}

pub fn contract_input_offset() -> usize {
    // The easiest way to get the contract input offset is to get the offset of the last field of
    // `InputRepr::Contract` -- i.e. the `contract_id` and then add its len to skip the last field.
    // Care should be taken to update this should `InputRepr::Contract` ever get another field after
    // this last one.
    InputRepr::Contract.contract_id_offset().unwrap() + ContractId::LEN
}