Universal Smart Contracts
Please note that all information and code snippets provided in this section are for educational purposes only and not to be directly deployed in production.
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
The verification system does not validate if a transaction was successful or not. It only validates if a transaction is included in a block and that block is really a part of the confirmed source chain. Therefore, success of a transaction has to be verified in the smart contract itself.
Core USC Contract Pattern
A typical USC contract follows this pattern:
Receives proofs and transaction data from an off-chain worker
Implements replay protection to prevent duplicate processing
Calls the Native Query Verifier Precompile to verify proofs synchronously
Extracts transaction/event data from verified transaction bytes
Executes business logic based on the verified data
Example USC Contract
See SimpleMinterUSC.sol for a complete USC implementation. The contract:
Receives proofs and transaction data from offchain worker
Implements replay protection using a
processedQueriesmappingUses 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
EvmV1DecoderExecutes 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:
Worker generates proofs using the Proof Generation API server
Worker calls USC contract with proofs and encoded transaction data
USC contract verifies proofs synchronously using the Native Query Verifier Precompile
USC contract extracts data from verified transaction bytes
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.sol
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 (
ERC20token with minting logic)VERIFIER: Immutable reference to the Native Query Verifier Precompile at address
0x0FD2processedQueries: 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
0x0FD2MerkleProofEntry: 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
VERIFIERimmutable variable to the precompile instanceThe 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
isLeftflag 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
MerkleProofandContinuityProofstructs from the provided componentsCalls the precompile's
verifyAndEmit()function synchronously at address0x0FD2Returns
trueif both Merkle proof (transaction inclusion) and continuity proof (block attestation chain) are valid; reverts on failure (transaction reverts if verification fails)Emits
TransactionVerifiedevent on successful verificationVerification 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, andtransactionIndexusing assembly for gas efficiencyReplay Protection: Checks if this transaction has already been processed
Proof Verification: Calls
_verifyProof()to verify the Merkle and continuity proofs synchronouslyState Update (replay protection): Marks the transaction as processed in
processedQueriesmappingTransaction 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
TokensMintedevent with the transaction details
Transaction Data Extraction
The contract includes helper functions for extracting and validating transaction data from encodedTransaction bytes:
Description:
Uses
EvmV1Decoderlibrary to decode the transaction bytesValidates transaction type and receipt status
Extracts event logs matching the
Transferevent signatureValidates that a burn transfer occurred (transfer to address < 128)
Complete Example
See SimpleMinterUSC.sol 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 example.
Next Steps
Check out this tutorial for an example of how to use the Creditcoin stack to set up a decentralized trustless bridge.
Last updated