Universal Smart Contracts

triangle-exclamation

What is the Universal Smart Contract

Universal Smart Contracts (USC) act as an adapter which extends existing smart contracts with cross-chain capabilities, allowing contracts to query, retrieve, and verify data from external blockchains via the Creditcoin Decentralized Oracle.

Unlike traditional omnichain or cross-chain solutions that focus narrowly on token transfers or specific assets, USC provides a general-purpose execution layer. This enables contracts to act on externally verified data without needing to rewrite core logic. By adopting USC into their tech stack, developers can transform their contracts into universal components powered by seamless cross-chain data, allowing for novel patterns of interoperability across multiple blockchains.

USC Contract Architecture

USC contracts verify cross-chain proofs and execute business logic. DApp Business Logic Contracts are contracts deployed on Creditcoin that contain the DApp's state and business logic. In the example implementation (SimpleMinterUSC), the business logic (ERC20 token minting) is integrated directly into the USC contract itself. While, this combined pattern works well for simple use cases, for more complex DApps, developers can separate concerns by deploying distinct contracts: a USC contract that handles the core USC responsibilities and separate business logic contracts that the USC contract calls after verification succeeds. Both patterns are valid; the choice depends on the complexity and requirements of the DApp. Business logic contracts are considered part of the USC architecture since they execute based on verified cross-chain data provided by the USC contract.

How USC Works

USC contracts verify cross-chain transaction data using the Native Query Verifier Precompile (address 0x0FD2), a built-in runtime component that provides synchronous verification of Merkle and continuity proofs. USC contracts integrate with it by calling its verify() (or alternatively verifyAndEmit() ) function directly to verify proofs before processing cross-chain data. Once verified, USC contracts extract transaction and event data directly from the verified transaction bytes and execute DApp-specific business logic.

Key characteristics:

  • Synchronous verification: Proofs are verified in the same transaction, no async processing

  • Direct data extraction: Transaction and event data is extracted directly from verified transaction bytes

  • Replay protection: USC contracts implement mechanisms to prevent duplicate processing

  • Native-speed execution: The precompile runs as native Rust code for optimal performance

triangle-exclamation

Core USC Contract Pattern

A typical USC contract follows this pattern:

  1. Receives proofs and transaction data from an off-chain worker

  2. Implements replay protection to prevent duplicate processing

  3. Calls the Native Query Verifier Precompile to verify proofs synchronously

  4. Extracts transaction/event data from verified transaction bytes

  5. Executes business logic based on the verified data

Example USC Contract

See SimpleMinterUSC.solarrow-up-right for a complete USC implementation. The contract:

  • Receives proofs and transaction data from offchain worker

  • Implements replay protection using a processedQueries mapping

  • Uses the Native Query Verifier Precompile to verify proofs

  • Validates transaction type and receipt status (must be successful)

  • Extracts event data from verified transaction bytes using EvmV1Decoder

  • Executes business logic (ERC20 token minting) within the same contract that mints tokens once a burn event is verified from the source chain

Key function signature:

DApp Business Logic Contracts

DApp Business Logic Contracts are smart contracts deployed on Creditcoin that contain the DApp's state and business logic. They are considered part of the USC architecture because they execute based on verified cross-chain data provided by USC contracts.

In the example implementation (SimpleMinterUSC), the business logic is integrated directly into the USC contract. The contract:

  • Stores DApp state (e.g., token balances via ERC20)

  • Implements DApp-specific logic (e.g., minting tokens)

  • Executes business logic immediately after verifying cross-chain proofs and validating transaction contents

  • Validates inputs and updates state accordingly

Transaction Data Extraction

After verification succeeds, USC contracts extract transaction and event data from the encodedTransaction bytes as part of the transaction content validation process. The transaction encoding follows a deterministic format that includes:

  • Transaction fields: Type, chain ID, nonce, from address, to address, value, etc.

  • Receipt fields: Status, gas used, logs (events)

  • Event data: Topics and data from transaction receipt logs

USC contracts can use libraries like EvmV1Decoder to selectively extract specific events or transaction fields only needed for their business logic. This selective extraction allows USC contracts to efficiently validate specific events or transaction fields needed for their business logic without decoding the entire transaction structure.

Query Processing Flow

When an off-chain worker provides proof data for a source chain transaction:

  1. Worker generates proofs using the Proof Generation API server

  2. Worker calls USC contract with proofs and encoded transaction data

  3. USC contract verifies proofs synchronously using the Native Query Verifier Precompile

  4. USC contract extracts data from verified transaction bytes

  5. USC contract executes business logic immediately in the same transaction

All of this happens synchronously in a single transaction—there is no async query processing or result storage.

USC Contract Implementation Example

The following sections break down a complete USC contract implementation based on SimpleMinterUSC.solarrow-up-right

Contract Structure

Key components:

  • Inherits from ERC20: The contract uses the combined pattern—it's both a USC (verifies proofs) and a business logic contract (ERC20 token with minting logic)

  • VERIFIER: Immutable reference to the Native Query Verifier Precompile at address 0x0FD2

  • processedQueries: Mapping for replay protection, preventing duplicate processing of the same transaction

Precompile Interface and Helper Library

Description:

  • INativeQueryVerifier: Interface defining the precompile's verification function and proof structures

  • NativeQueryVerifierLib: Helper library that provides easy access to the precompile at address 0x0FD2

  • MerkleProofEntry: Represents a single node in the Merkle proof path (hash value and position)

  • ContinuityProof: Contains the chain of block attestations proving continuity from a checkpoint

Constructor and Initialization

Description:

  • Initializes the ERC20 token with name and symbol

  • Sets the VERIFIER immutable variable to the precompile instance

  • The precompile address is constant and always available

Replay Protection

Description:

  • processedQueries: Maps transaction keys to boolean values to track processed transactions

Transaction Index Calculation

Description:

  • _calculateTransactionIndex: Derives the transaction index from the Merkle proof path

    • The proof path encodes the position in the Merkle tree

    • Each isLeft flag indicates whether the sibling is on the left (1) or right (0)

    • The index is calculated by traversing the path from leaf to root

Proof Verification

Description:

  • Constructs the MerkleProof and ContinuityProof structs from the provided components

  • Calls the precompile's verifyAndEmit() function synchronously at address 0x0FD2

  • Returns true if both Merkle proof (transaction inclusion) and continuity proof (block attestation chain) are valid; reverts on failure (transaction reverts if verification fails)

  • Emits TransactionVerified event on successful verification

  • Verification happens in the same transaction - no async processing

Main Entry Point: mintFromQuery

Description:

  • Parameters: Receives all proof components and transaction data from the off-chain worker

  • Transaction Index Calculation: Calculates the transaction index from the Merkle proof path using _calculateTransactionIndex()

  • Transaction Key Generation: Creates a unique key from chainKey, blockHeight, and transactionIndex using assembly for gas efficiency

  • Replay Protection: Checks if this transaction has already been processed

  • Proof Verification: Calls _verifyProof() to verify the Merkle and continuity proofs synchronously

  • State Update (replay protection): Marks the transaction as processed in processedQueries mapping

  • Transaction Content Validation: Validates the transaction contents by checking transaction type and receipt status.

  • Business Logic Execution: If validation passes, executes business logic (minting tokens)

  • Event Emission: Emits TokensMinted event with the transaction details

Transaction Data Extraction

The contract includes helper functions for extracting and validating transaction data from encodedTransaction bytes:

Description:

  • Uses EvmV1Decoder library to decode the transaction bytes

  • Validates transaction type and receipt status

  • Extracts event logs matching the Transfer event signature

  • Validates that a burn transfer occurred (transfer to address < 128)

Complete Example

See SimpleMinterUSC.solarrow-up-right for the complete implementation with all helper functions and event processing logic. A corresponding helper script and instructions to use this code are available in the hello-bridge examplearrow-up-right.

Next Steps

DApp Design Patterns

Check out this tutorialarrow-up-right for an example of how to use the Creditcoin stack to set up a decentralized trustless bridge.

Last updated