Skip to content

wake.ir.statements.inline_assembly module #

ExternalReference class #

Reference from an inline assembly block to a Solidity declaration.

Warning

This is not an IR node, but a helper class for InlineAssembly. Since this is not an IR node, there must still be a Yul IR node (YulIdentifier) in the source code that represents the identifier.

Source code in wake/ir/statements/inline_assembly.py
class ExternalReference:
    """
    Reference from an inline assembly block to a Solidity declaration.
    !!! warning
        This is not an IR node, but a helper class for [InlineAssembly][wake.ir.statements.inline_assembly.InlineAssembly].
        Since this is not an IR node, there must still be a Yul IR node ([YulIdentifier][wake.ir.yul.identifier.YulIdentifier]) in the source code that represents the identifier.
    """

    _inline_assembly: weakref.ReferenceType[InlineAssembly]
    _external_reference_model: ExternalReferenceModel
    _reference_resolver: ReferenceResolver
    _source: bytes
    _source_unit: weakref.ReferenceType[SourceUnit]

    _referenced_declaration_id: AstNodeId
    _value_size: int
    _suffix: Optional[InlineAssemblySuffix]
    _yul_identifier: Optional[YulIdentifier]

    def __init__(
        self,
        inline_assembly: InlineAssembly,
        init: IrInitTuple,
        external_reference_model: ExternalReferenceModel,
    ):
        self._inline_assembly = weakref.ref(inline_assembly)
        self._external_reference_model = external_reference_model
        self._reference_resolver = weakref.proxy(init.reference_resolver)
        self._source = init.source[self.byte_location[0] : self.byte_location[1]]
        assert init.source_unit is not None
        self._source_unit = weakref.ref(init.source_unit)

        self._referenced_declaration_id = external_reference_model.declaration
        assert self._referenced_declaration_id >= 0
        self._value_size = external_reference_model.value_size
        self._suffix = external_reference_model.suffix
        self._yul_identifier = None

        if external_reference_model.is_offset:
            self._suffix = InlineAssemblySuffix.OFFSET
        elif external_reference_model.is_slot:
            self._suffix = InlineAssemblySuffix.SLOT

        self._reference_resolver.register_post_process_callback(self._post_process)

    def __getstate__(self):
        state = self.__dict__.copy()
        del state["_inline_assembly"]
        del state["_source_unit"]
        del state["_reference_resolver"]
        return state

    def _post_process(self, callback_params: CallbackParams):
        referenced_declaration = self.referenced_declaration
        referenced_declaration.register_reference(self)
        self._reference_resolver.register_destroy_callback(
            self.source_unit.file, partial(self._destroy, referenced_declaration)
        )
        interval_tree = callback_params.interval_trees[self.source_unit.file]
        start, end = self.byte_location
        nodes = interval_tree[start:end]
        node = next(node for node in nodes if node.begin == start and node.end == end)
        assert isinstance(
            node.data, YulIdentifier
        ), f"Expected Identifier, got {type(node.data)}"
        self._yul_identifier = node.data
        self._yul_identifier._external_reference = weakref.ref(self)

    def _destroy(self, referenced_declaration: VariableDeclaration) -> None:
        referenced_declaration.unregister_reference(self)

    @property
    def source_unit(self) -> SourceUnit:
        """
        Returns:
            Source unit that contains this node.
        """
        return is_not_none(self._source_unit())

    @property
    def byte_location(self) -> Tuple[int, int]:
        """
        Returns:
            Byte offsets (start and end) of the external reference in the source file.
        """
        return (
            self._external_reference_model.src.byte_offset,
            self._external_reference_model.src.byte_offset
            + self._external_reference_model.src.byte_length,
        )

    @property
    @weak_self_lru_cache(maxsize=2048)
    def identifier_location(self) -> Tuple[int, int]:
        """
        !!! example
            Returns the byte location of `stateVar` on line 6, not `stateVar.slot`:
            ```solidity linenums="1"
            contract Foo {
                uint stateVar;

                function f() public pure {
                    assembly {
                        let x := stateVar.slot
                    }
                }
            }
            ```

        Returns:
            Byte offsets (start and end) of the identifier representing the external reference in the source file.
        """
        source = bytearray(self._source)
        _, stripped_sums = SoliditySourceParser.strip_comments(source)

        match = IDENTIFIER_RE.match(source)
        assert match

        if len(stripped_sums) == 0:
            stripped = 0
        else:
            index = bisect([s[0] for s in stripped_sums], match.start())
            if index == 0:
                stripped = 0
            else:
                stripped = stripped_sums[index - 1][1]

        start = self.byte_location[0] + match.start() + stripped
        end = self.byte_location[0] + match.end() + stripped
        return start, end

    @property
    def referenced_declaration(self) -> VariableDeclaration:
        """
        Returns:
            Solidity variable declaration referenced by this external reference.
        """
        node = self._reference_resolver.resolve_node(
            self._referenced_declaration_id, self.source_unit.cu_hash
        )
        assert isinstance(node, VariableDeclaration)
        return node

    @property
    def inline_assembly(self) -> InlineAssembly:
        """
        Returns:
            Inline assembly block this external references belongs to.
        """
        return is_not_none(self._inline_assembly())

    @property
    def yul_identifier(self) -> YulIdentifier:
        """
        Returns:
            Yul Identifier node representing this external reference.
        """
        assert isinstance(self._yul_identifier, YulIdentifier)
        return self._yul_identifier

    @property
    def value_size(self) -> int:
        # TODO document this?
        return self._value_size

    @property
    def suffix(self) -> Optional[InlineAssemblySuffix]:
        """
        - `.slot` and `.offset` suffixes may be applied to storage variables.
        - `.length` suffix may be applied to dynamically-sized `calldata` arrays, `calldata` bytes and `calldata` strings.
        - `.address` amd `.selector` suffixes may be applied to variables holding a reference to an external function (the variable has the [FunctionTypeName][wake.ir.type_names.function_type_name.FunctionTypeName] type name).

        Returns:
            Suffix of the external reference, if any.
        """
        return self._suffix

byte_location: Tuple[int, int] property #

Returns:

Type Description
Tuple[int, int]

Byte offsets (start and end) of the external reference in the source file.

identifier_location: Tuple[int, int] property #

Example

Returns the byte location of stateVar on line 6, not stateVar.slot:

1
2
3
4
5
6
7
8
9
contract Foo {
    uint stateVar;

    function f() public pure {
        assembly {
            let x := stateVar.slot
        }
    }
}

Returns:

Type Description
Tuple[int, int]

Byte offsets (start and end) of the identifier representing the external reference in the source file.

inline_assembly: InlineAssembly property #

Returns:

Type Description
InlineAssembly

Inline assembly block this external references belongs to.

referenced_declaration: VariableDeclaration property #

Returns:

Type Description
VariableDeclaration

Solidity variable declaration referenced by this external reference.

source_unit: SourceUnit property #

Returns:

Type Description
SourceUnit

Source unit that contains this node.

suffix: Optional[InlineAssemblySuffix] property #

  • .slot and .offset suffixes may be applied to storage variables.
  • .length suffix may be applied to dynamically-sized calldata arrays, calldata bytes and calldata strings.
  • .address amd .selector suffixes may be applied to variables holding a reference to an external function (the variable has the FunctionTypeName type name).

Returns:

Type Description
Optional[InlineAssemblySuffix]

Suffix of the external reference, if any.

yul_identifier: YulIdentifier property #

Returns:

Type Description
YulIdentifier

Yul Identifier node representing this external reference.

InlineAssembly class #

Bases: StatementAbc

Inline assembly block in Solidity.

Example

function f() public pure {
    assembly {
        let x := 1
        let y := 2
        let z := add(x, y)
    }
}
Source code in wake/ir/statements/inline_assembly.py
class InlineAssembly(StatementAbc):
    """
    Inline assembly block in Solidity.
    !!! example
        ```solidity
        function f() public pure {
            assembly {
                let x := 1
                let y := 2
                let z := add(x, y)
            }
        }
        ```
    """

    _ast_node: SolcInlineAssembly
    _parent: weakref.ReferenceType[
        Union[
            Block,
            DoWhileStatement,
            ForStatement,
            IfStatement,
            UncheckedBlock,
            WhileStatement,
        ]
    ]

    _yul_block: YulBlock
    _evm_version: InlineAssemblyEvmVersion
    _external_references: IntervalTree
    _flags: Set[InlineAssemblyFlag]

    def __init__(
        self,
        init: IrInitTuple,
        inline_assembly: SolcInlineAssembly,
        parent: SolidityAbc,
    ):
        super().__init__(init, inline_assembly, parent)
        init.inline_assembly = self

        self._yul_block = YulBlock(init, inline_assembly.ast, self)
        self._evm_version = inline_assembly.evm_version
        self._external_references = IntervalTree()
        self._flags = set()
        if inline_assembly.flags is not None:
            for flag in inline_assembly.flags:
                self._flags.add(InlineAssemblyFlag(flag))
        for external_reference in inline_assembly.external_references:
            start = external_reference.src.byte_offset
            end = start + external_reference.src.byte_length
            self._external_references[start:end] = ExternalReference(
                self, init, external_reference
            )

        init.inline_assembly = None

    def __iter__(self) -> Iterator[IrAbc]:
        yield self
        yield from self._yul_block

    @property
    def parent(
        self,
    ) -> Union[
        Block,
        DoWhileStatement,
        ForStatement,
        IfStatement,
        UncheckedBlock,
        WhileStatement,
    ]:
        """
        Returns:
            Parent IR node.
        """
        return super().parent

    @property
    def children(self) -> Iterator[YulBlock]:
        """
        Yields:
            Direct children of this node.
        """
        yield self._yul_block

    @property
    def yul_block(self) -> YulBlock:
        """
        Returns:
            Yul block containing Yul IR nodes ([YulAbc][wake.ir.yul.abc.YulAbc]).
        """
        return self._yul_block

    @property
    def evm_version(self) -> InlineAssemblyEvmVersion:
        """
        Depends on the version of the `solc` compiler used to compile the contract and settings passed to the compiler.

        Returns:
            EVM version used for the inline assembly block.
        """
        return self._evm_version

    @property
    def flags(self) -> FrozenSet[InlineAssemblyFlag]:
        """
        !!! example
            ```solidity
            function f() public pure {
                assembly ("memory-safe") {
                    let x := 1
                    let y := 2
                    let z := add(x, y)
                }
            }
            ```

        Returns:
            Flags decorating the inline assembly block.
        """
        return frozenset(self._flags)

    @property
    def external_references(self) -> Tuple[ExternalReference, ...]:
        """
        Returns:
            External references in the inline assembly block.
        """
        return tuple(interval.data for interval in self._external_references)

    def external_reference_at(self, byte_offset: int) -> Optional[ExternalReference]:
        """
        Args:
            byte_offset: Byte offset in the source file.

        Returns:
            External reference at the given byte offset, if any.
        """
        intervals = self._external_references.at(byte_offset)
        assert len(intervals) <= 1
        if len(intervals) == 0:
            return None
        return intervals.pop().data

    @property
    @weak_self_lru_cache(maxsize=2048)
    def modifies_state(
        self,
    ) -> Set[Tuple[Union[ExpressionAbc, StatementAbc, YulAbc], ModifiesStateFlag]]:
        return self.yul_block.modifies_state

children: Iterator[YulBlock] property #

Yields:

Type Description
YulBlock

Direct children of this node.

evm_version: InlineAssemblyEvmVersion property #

Depends on the version of the solc compiler used to compile the contract and settings passed to the compiler.

Returns:

Type Description
InlineAssemblyEvmVersion

EVM version used for the inline assembly block.

external_references: Tuple[ExternalReference, ...] property #

Returns:

Type Description
Tuple[ExternalReference, ...]

External references in the inline assembly block.

flags: FrozenSet[InlineAssemblyFlag] property #

Example

function f() public pure {
    assembly ("memory-safe") {
        let x := 1
        let y := 2
        let z := add(x, y)
    }
}

Returns:

Type Description
FrozenSet[InlineAssemblyFlag]

Flags decorating the inline assembly block.

parent: Union[Block, DoWhileStatement, ForStatement, IfStatement, UncheckedBlock, WhileStatement] property #

Returns:

Type Description
Union[Block, DoWhileStatement, ForStatement, IfStatement, UncheckedBlock, WhileStatement]

Parent IR node.

yul_block: YulBlock property #

Returns:

Type Description
YulBlock

Yul block containing Yul IR nodes (YulAbc).

external_reference_at(byte_offset) #

Parameters:

Name Type Description Default
byte_offset int

Byte offset in the source file.

required

Returns:

Type Description
Optional[ExternalReference]

External reference at the given byte offset, if any.

Source code in wake/ir/statements/inline_assembly.py
def external_reference_at(self, byte_offset: int) -> Optional[ExternalReference]:
    """
    Args:
        byte_offset: Byte offset in the source file.

    Returns:
        External reference at the given byte offset, if any.
    """
    intervals = self._external_references.at(byte_offset)
    assert len(intervals) <= 1
    if len(intervals) == 0:
        return None
    return intervals.pop().data